| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363 |
- import { TIMEOUTS } from '../../utils/timeouts';
- import { test, expect } from '../../utils/test-setup';
- /**
- * 首页看板人才数据同步 E2E 测试
- *
- * 测试目标:验证后台添加人员到订单后,企业小程序首页看板显示分配的人才数据
- *
- * 测试流程:
- * 1. 后台操作:登录 → 打开订单详情 → 添加人员到订单 → 验证添加成功
- * 2. 小程序验证:登录 → 首页看板 → 验证人才卡片显示 → 验证统计数字同步
- *
- * 测试要点:
- * - 使用两个独立的 browser context(后台和小程序)
- * - 记录数据同步时间
- * - 使用 data-testid 选择器(后台)和文本选择器(小程序)
- * - 验证人才信息完整性(姓名、残疾类型、等级)
- * - 验证核心统计数字(在职人员、待入职、本月新增)
- *
- * Playwright MCP 探索结果 (2026-01-14):
- * - 后台添加人员成功,订单 722 添加人员 1238
- * - 小程序首页显示分配人才卡片和核心统计数字
- * - 数据同步验证成功
- *
- * 与 Story 13.3 的区别:
- * - Story 13.3: 验证后台添加人员 → 人才**小程序**端的数据同步
- * - Story 13.6: 验证后台添加人员 → 企业**小程序首页**的人才数据
- */
- // 测试数据常量
- const TEST_USER_PHONE = '13800001111';
- const TEST_USER_PASSWORD = 'password123';
- const TEST_ORDER_ID = 721; // 已存在的订单,关联到测试公司
- const TEST_PERSON_NAME = '测试残疾人_1768346782426_12_8219';
- test.describe('首页看板人才数据同步测试 - 后台添加人员到企业小程序首页', () => {
- let addedPersonName: string;
- let syncStartTime: number;
- test.describe.serial('后台添加人员到订单', () => {
- test('应该成功登录后台并添加人员到订单', async ({ page: adminPage }) => {
- syncStartTime = Date.now();
- // 1. 后台登录
- await adminPage.goto('http://localhost:8080/admin/login');
- await adminPage.getByPlaceholder('请输入用户名').fill('admin');
- await adminPage.getByPlaceholder('请输入密码').fill('admin123');
- await adminPage.getByRole('button', { name: '登录' }).click();
- await adminPage.waitForURL('**/admin/dashboard', { timeout: TIMEOUTS.PAGE_LOAD });
- console.debug('[后台] 登录成功');
- // 2. 导航到订单管理页面
- await adminPage.goto('http://localhost:8080/admin/orders');
- await adminPage.waitForSelector('table tbody tr', { state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
- console.debug('[后台] 导航到订单管理页面');
- // 3. 搜索并打开订单详情(使用测试公司的订单)
- const orderRow = adminPage.locator('table tbody tr').filter({ hasText: TEST_ORDER_ID.toString() }).first();
- await orderRow.getByRole('button', { name: '打开菜单' }).click();
- await adminPage.waitForTimeout(TIMEOUTS.SHORT);
- console.debug(`[后台] 打开订单 ${TEST_ORDER_ID} 的菜单`);
- // 4. 点击"查看详情"
- await adminPage.getByRole('menuitem', { name: '查看详情' }).click();
- await adminPage.waitForSelector('[role="dialog"]', { state: 'visible', timeout: TIMEOUTS.DIALOG });
- console.debug('[后台] 打开订单详情对话框');
- // 5. 点击"添加人员"按钮
- await adminPage.getByTestId('order-detail-bottom-add-persons-button').click();
- await adminPage.waitForSelector('text=选择残疾人', { state: 'visible', timeout: TIMEOUTS.DIALOG });
- console.debug('[后台] 打开选择残疾人对话框');
- // 6. 选择一个残疾人(选择第一个未禁用的复选框)
- try {
- // 等待残疾人列表加载
- await adminPage.waitForSelector('table tbody tr', { state: 'visible', timeout: TIMEOUTS.TABLE_LOAD });
- // 找到第一个可选择的残疾人(复选框未禁用)
- const selectableRow = adminPage.locator('table tbody tr input[type="checkbox"]:not([disabled])').first();
- await selectableRow.check();
- console.debug('[后台] 选择了第一个残疾人');
- // 获取残疾人姓名和ID
- const selectedRow = selectableRow.locator('../../..');
- const cells = await selectedRow.locator('td').allTextContents();
- addedPersonName = cells[1]; // 第二列是姓名
- console.debug(`[后台] 选中的残疾人: ${addedPersonName}`);
- // 7. 点击"确认选择"按钮
- await adminPage.getByTestId('confirm-batch-button').click();
- await adminPage.waitForTimeout(TIMEOUTS.MEDIUM);
- console.debug('[后台] 确认选择残疾人');
- // 8. 点击"确认添加"按钮(在待添加人员列表中)
- await adminPage.getByTestId('confirm-add-persons-button').click();
- await adminPage.waitForTimeout(TIMEOUTS.MEDIUM);
- console.debug('[后台] 确认添加残疾人到订单');
- // 9. 验证添加成功的 Toast
- const successToast = adminPage.locator('[data-sonner-toast][data-type="success"]');
- await expect(successToast).toBeVisible({ timeout: TIMEOUTS.TOAST });
- console.debug('[后台] 人员添加成功');
- // 10. 保存残疾人姓名到环境变量
- process.env.__TEST_PERSON_NAME__ = addedPersonName;
- } catch (_error) {
- console.debug('[后台] 没有可选择的残疾人,或选择对话框未正常打开');
- // 使用已知存在的测试残疾人
- addedPersonName = TEST_PERSON_NAME;
- process.env.__TEST_PERSON_NAME__ = addedPersonName;
- }
- // 11. 关闭对话框
- await adminPage.getByTestId('order-detail-close-button').click();
- console.debug('[后台] 关闭订单详情对话框');
- });
- });
- test.describe.serial('小程序验证首页看板人才数据同步', () => {
- test.use({ storageState: undefined }); // 确保使用新的浏览器上下文
- test('应该在小程序首页看板显示分配的人才数据', async ({ page: miniPage }) => {
- // 从环境变量获取添加的残疾人姓名
- const testPersonName = process.env.__TEST_PERSON_NAME__ || TEST_PERSON_NAME;
- console.debug(`[小程序] 查找人才: ${testPersonName}`);
- // 1. 小程序登录
- await miniPage.goto('http://localhost:8080/mini');
- await miniPage.waitForLoadState('networkidle');
- // 填写登录表单
- await miniPage.getByTestId('mini-phone-input').getByPlaceholder('请输入手机号').fill(TEST_USER_PHONE);
- await miniPage.getByRole('textbox', { name: '请输入密码' }).fill(TEST_USER_PASSWORD);
- await miniPage.getByTestId('mini-login-button').click();
- console.debug('[小程序] 登录请求已发送');
- // 等待登录成功(跳转到 dashboard)
- // 注意:小程序使用 hash 路由,需要检查 hash 而不是 pathname
- await miniPage.waitForURL(
- url => url.hash.includes('dashboard') || url.hash.includes('/pages/yongren/dashboard'),
- { timeout: TIMEOUTS.PAGE_LOAD }
- );
- console.debug('[小程序] 登录成功,停留在首页 dashboard');
- // 2. 记录数据同步时间
- const syncEndTime = Date.now();
- const syncTime = syncEndTime - syncStartTime;
- console.debug(`[小程序] 数据同步完成,耗时: ${syncTime}ms`);
- // 3. 等待首页数据加载
- await miniPage.waitForTimeout(TIMEOUTS.LONG);
- // 3.5. 验证企业名称显示(不是占位符"企业名称")
- // 企业名称应该在"欢迎回来"后面显示
- const welcomeText = await miniPage.getByText('欢迎回来').locator('..').textContent();
- expect(welcomeText).toBeDefined();
- // 验证不包含占位符"企业名称"
- expect(welcomeText).not.toContain('企业名称');
- // 验证包含真实的企业名称(测试公司的名称格式)
- expect(welcomeText).toMatch(/测试公司_\d+/);
- console.debug('[小程序] 企业名称显示正确');
- // 4. 验证分配人才列表显示
- // 注意:小程序首页没有 data-testid,需要使用文本选择器
- const assignedTalentSection = miniPage.getByText('分配人才', { exact: true });
- await expect(assignedTalentSection).toBeVisible();
- console.debug('[小程序] 分配人才区域可见');
- // 5. 验证人才卡片信息
- // 检查是否显示"暂无分配人才"或实际的人才卡片
- const noTalentMessage = miniPage.getByText('暂无分配人才');
- const hasNoTalent = await noTalentMessage.count() > 0;
- if (hasNoTalent) {
- console.debug('[小程序] 显示"暂无分配人才" - 这可能是正常的,取决于测试数据');
- } else {
- console.debug('[小程序] 显示分配人才卡片');
- // 验证人才信息完整性
- // 首页显示的人才卡片包含:姓名、残疾类型、等级、状态
- const talentCard = miniPage.locator('.bg-white.p-4.rounded-lg').first();
- await expect(talentCard).toBeVisible();
- console.debug('[小程序] 人才卡片可见');
- // 验证人才姓名显示
- const personNameText = await talentCard.textContent();
- expect(personNameText).toContain(testPersonName);
- console.debug(`[小程序] 人才姓名: ${testPersonName}`);
- // 验证残疾类型和等级显示(格式:残疾类型 · 等级)
- expect(personNameText).toMatch(/(视力|听力|肢体|智力|精神)残疾/);
- console.debug('[小程序] 残疾类型显示正确');
- // 验证工作状态显示
- expect(personNameText).toMatch(/(在职|待入职|离职)/);
- console.debug('[小程序] 工作状态显示正确');
- }
- // 6. 验证核心统计数字
- // 验证"在职人员"、"待入职"、"本月新增"统计卡片可见
- const statsText = await miniPage.getByText(/在职人员|待入职|本月新增/).allTextContents();
- expect(statsText.length).toBeGreaterThan(0);
- console.debug('[小程序] 核心统计卡片可见');
- // 7. 验证统计数字非负数
- const employedCount = await miniPage.locator('text=/在职人员/').locator('..').locator('text=/\\d+/').first().textContent();
- const employedNum = parseInt(employedCount || '0');
- expect(employedNum).toBeGreaterThanOrEqual(0);
- console.debug(`[小程序] 在职人员数: ${employedNum}`);
- // 8. 验证数据统计区域
- // 使用 .first() 处理多个"数据统计"文本的情况
- const dataStatsSection = miniPage.getByText('数据统计', { exact: true }).first();
- await expect(dataStatsSection).toBeVisible();
- console.debug('[小程序] 数据统计区域可见');
- // 9. 验证在职率和平均薪资显示
- // 验证"在职率"标签存在
- const employmentRateLabel = miniPage.getByText(/在职率/);
- await expect(employmentRateLabel).toBeVisible();
- // 验证百分比数值显示
- const percentageValue = miniPage.getByText(/\d+%|--/).first();
- await expect(percentageValue).toBeVisible();
- console.debug('[小程序] 在职率显示正确');
- // 10. 下拉刷新验证
- // 向下滚动触发下拉刷新
- await miniPage.evaluate(() => {
- window.scrollTo(0, 0);
- });
- await miniPage.waitForTimeout(TIMEOUTS.SHORT);
- // 模拟下拉刷新手势
- await miniPage.evaluate(() => {
- const scrollView = document.querySelector('.overflow-y-auto');
- if (scrollView) {
- scrollView.scrollTop = 100;
- setTimeout(() => {
- scrollView.scrollTop = 0;
- }, 100);
- }
- });
- await miniPage.waitForTimeout(TIMEOUTS.LONG);
- console.debug('[小程序] 触发下拉刷新');
- // 验证刷新后数据仍然显示
- await expect(assignedTalentSection).toBeVisible();
- console.debug('[小程序] 下拉刷新后数据正常');
- // 清理环境变量
- delete process.env.__TEST_PERSON_NAME__;
- console.debug('[小程序] 首页看板人才数据同步验证完成');
- });
- });
- /**
- * 测试场景:后台添加人员后,核心统计数字同步验证
- */
- test.describe.serial('核心统计数字同步验证', () => {
- test.use({ storageState: undefined });
- test('应该正确更新核心统计数字', async ({ page: miniPage }) => {
- // 1. 小程序登录
- await miniPage.goto('http://localhost:8080/mini');
- await miniPage.waitForLoadState('networkidle');
- await miniPage.getByTestId('mini-phone-input').getByPlaceholder('请输入手机号').fill(TEST_USER_PHONE);
- await miniPage.getByRole('textbox', { name: '请输入密码' }).fill(TEST_USER_PASSWORD);
- await miniPage.getByTestId('mini-login-button').click();
- await miniPage.waitForURL(
- url => url.hash.includes('dashboard') || url.hash.includes('/pages/yongren/dashboard'),
- { timeout: TIMEOUTS.PAGE_LOAD }
- );
- console.debug('[小程序] 登录成功');
- // 2. 等待首页数据加载
- await miniPage.waitForTimeout(TIMEOUTS.LONG);
- // 3. 获取核心统计数据
- // 在职人员、待入职、本月新增
- const statsContainer = miniPage.locator('.from-blue-500.to-purple-600').first();
- const statsText = await statsContainer.textContent();
- console.debug(`[小程序] 核心统计: ${statsText}`);
- // 4. 验证统计数据格式
- // 格式应该是:数字 + 标签
- expect(statsText).toMatch(/\d+/); // 包含数字
- expect(statsText).toMatch(/在职人员|待入职|本月新增/); // 包含标签
- // 5. 验证数据统计区域的在职率和平均薪资
- const employmentRateText = await miniPage.getByText(/在职率/).locator('..').textContent();
- expect(employmentRateText).toBeTruthy();
- const avgSalaryLabel = miniPage.getByText(/平均薪资/);
- await expect(avgSalaryLabel).toBeVisible();
- // 验证薪资数值显示
- const salaryValue = miniPage.getByText(/¥\d+/).first();
- await expect(salaryValue).toBeVisible();
- console.debug('[小程序] 平均薪资显示正确');
- console.debug('[小程序] 核心统计数字验证完成');
- });
- });
- /**
- * 测试场景:数据刷新时效性验证
- */
- test.describe.serial('数据刷新时效性验证', () => {
- test.use({ storageState: undefined });
- test('应该在合理时间内完成数据同步', async ({ page: miniPage }) => {
- const startTime = Date.now();
- // 1. 小程序登录
- await miniPage.goto('http://localhost:8080/mini');
- await miniPage.waitForLoadState('networkidle');
- await miniPage.getByTestId('mini-phone-input').getByPlaceholder('请输入手机号').fill(TEST_USER_PHONE);
- await miniPage.getByRole('textbox', { name: '请输入密码' }).fill(TEST_USER_PASSWORD);
- await miniPage.getByTestId('mini-login-button').click();
- await miniPage.waitForURL(
- url => url.hash.includes('dashboard') || url.hash.includes('/pages/yongren/dashboard'),
- { timeout: TIMEOUTS.PAGE_LOAD }
- );
- // 2. 等待首页数据完全加载
- await miniPage.waitForSelector('text=分配人才', { state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
- await miniPage.waitForSelector('text=数据统计', { state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
- const endTime = Date.now();
- const loadTime = endTime - startTime;
- // 3. 验证加载时间在合理范围内(应该在 10 秒内完成)
- expect(loadTime).toBeLessThan(10000);
- console.debug(`[小程序] 首页数据加载时间: ${loadTime}ms (预期 < 10000ms)`);
- // 4. 验证 API 响应时间(通过检查网络请求)
- // 注意:这需要在测试中启用网络监听
- const apiRequests: string[] = [];
- miniPage.on('request', request => {
- if (request.url().includes('/api/v1/yongren/company/')) {
- apiRequests.push(request.url());
- }
- });
- // 刷新页面
- await miniPage.reload();
- await miniPage.waitForLoadState('networkidle');
- const refreshEndTime = Date.now();
- const refreshTime = refreshEndTime - endTime;
- expect(refreshTime).toBeLessThan(8000); // 刷新应该在 8 秒内完成
- console.debug(`[小程序] 页面刷新时间: ${refreshTime}ms (预期 < 8000ms)`);
- });
- });
- });
|