person-add-sync.spec.ts 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. import { TIMEOUTS } from '../../utils/timeouts';
  2. import { test, expect } from '../../utils/test-setup';
  3. import { AdminLoginPage } from '../../pages/admin/login.page';
  4. import { TalentMiniPage } from '../../pages/mini/talent-mini.page';
  5. import type { TalentOrderData, TalentOrderDetailData } from '../../pages/mini/talent-mini.page';
  6. /**
  7. * 跨端数据同步 E2E 测试 - 人员添加
  8. *
  9. * 测试目标:验证后台添加残疾人到订单后,人才小程序能否正确显示关联的订单
  10. *
  11. * 测试流程:
  12. * 1. 后台操作:登录 → 打开订单详情 → 添加残疾人 → 验证添加成功
  13. * 2. 小程序验证:登录 → 导航到"我的订单" → 验证订单显示
  14. *
  15. * 测试要点:
  16. * - 使用两个独立的 browser context(后台和小程序)
  17. * - 记录数据同步时间
  18. * - 使用 data-testid 选择器
  19. * - 验证两步确认流程(选择残疾人 → 确认添加)
  20. *
  21. * 关键发现(来自 Task 0 Playwright MCP 探索):
  22. * 1. 添加人员需要两步确认:
  23. * - 第一步:选择残疾人 → "确认选择"
  24. * - 第二步:待添加人员列表 → "确认添加"
  25. * 2. 默认薪资为 5000
  26. * 3. 入职日期自动设置为当天
  27. * 4. 已绑定人员的复选框自动禁用
  28. */
  29. // 测试常量
  30. const TEST_SYNC_TIMEOUT = 10000; // 数据同步等待时间(ms)- AC 要求 ≤ 10 秒
  31. const TEST_POLL_INTERVAL = 500; // 轮询检查间隔(ms)
  32. const TEST_ORDER_ID = 724; // 测试订单 ID(使用已存在的订单)
  33. const TEST_ORDER_NAME = 'Epic13验证测试_1768403960000_Story13.2已编辑'; // 测试订单名称
  34. // 人才小程序登录凭据(需要与添加的残疾人关联)
  35. const TALENT_LOGIN_PHONE = '13800119311'; // 测试残疾人_1768346764677_11_9311 的手机号
  36. const TALENT_LOGIN_PASSWORD = 'password123'; // 默认测试密码
  37. /**
  38. * 后台登录辅助函数
  39. */
  40. async function loginAdmin(page: any, testUsers: any) {
  41. const adminLoginPage = new AdminLoginPage(page);
  42. await adminLoginPage.goto();
  43. await adminLoginPage.page.getByPlaceholder('请输入用户名').fill(testUsers.admin.username);
  44. await adminLoginPage.page.getByPlaceholder('请输入密码').fill(testUsers.admin.password);
  45. await adminLoginPage.page.getByRole('button', { name: '登录' }).click();
  46. await adminLoginPage.page.waitForURL('**/admin/dashboard', { timeout: TIMEOUTS.PAGE_LOAD });
  47. console.debug('[后台] 登录成功');
  48. }
  49. /**
  50. * 人才小程序登录辅助函数
  51. */
  52. async function loginTalentMini(page: any) {
  53. const talentMiniPage = new TalentMiniPage(page);
  54. await talentMiniPage.goto();
  55. await talentMiniPage.login(TALENT_LOGIN_PHONE, TALENT_LOGIN_PASSWORD);
  56. await talentMiniPage.expectLoginSuccess();
  57. console.debug('[人才小程序] 登录成功');
  58. }
  59. // 测试状态管理(使用闭包替代 process.env)
  60. interface TestState {
  61. personId: number | null;
  62. personName: string | null;
  63. syncTime: number | null;
  64. }
  65. const testState: TestState = {
  66. personId: null,
  67. personName: null,
  68. syncTime: null,
  69. };
  70. test.describe('跨端数据同步测试 - 后台添加人员到人才小程序', () => {
  71. // 在所有测试后清理测试数据
  72. test.afterAll(async () => {
  73. // 清理测试状态
  74. testState.personId = null;
  75. testState.personName = null;
  76. testState.syncTime = null;
  77. console.debug('[清理] 测试数据已清理');
  78. });
  79. test.describe.serial('后台添加人员', () => {
  80. test('应该成功打开订单详情并添加残疾人', async ({ page: adminPage, testUsers }) => {
  81. // 记录开始时间
  82. const startTime = Date.now();
  83. // 1. 后台登录(使用辅助函数)
  84. await loginAdmin(adminPage, testUsers);
  85. // 2. 导航到订单管理页面
  86. await adminPage.goto('/admin/orders');
  87. await adminPage.waitForSelector('table tbody tr', { state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
  88. console.debug('[后台] 导航到订单管理页面');
  89. // 3. 打开订单详情对话框
  90. // 点击订单的"打开菜单"按钮
  91. await adminPage.getByTestId(`order-menu-trigger-${TEST_ORDER_ID}`).click();
  92. await adminPage.waitForTimeout(TIMEOUTS.VERY_SHORT);
  93. console.debug(`[后台] 打开订单 ${TEST_ORDER_ID} 的菜单`);
  94. // 点击"查看详情"菜单项
  95. await adminPage.getByTestId(`view-order-detail-button-${TEST_ORDER_ID}`).click();
  96. await adminPage.waitForSelector('[role="dialog"]', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  97. console.debug('[后台] 打开订单详情对话框');
  98. // 4. 点击"添加人员"按钮
  99. await adminPage.getByTestId('order-detail-card-add-persons-button').click();
  100. await adminPage.waitForSelector('text=选择残疾人', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  101. console.debug('[后台] 打开选择残疾人对话框');
  102. // 5. 选择残疾人
  103. // 注意:已绑定人员的复选框会被禁用
  104. // 动态选择第一个未绑定的残疾人
  105. const checkboxes = adminPage.locator('[data-testid^="person-checkbox-"]:not([disabled])');
  106. const firstAvailable = checkboxes.first();
  107. const testPersonIdAttr = await firstAvailable.getAttribute('data-testid');
  108. const testPersonId = parseInt(testPersonIdAttr?.replace('person-checkbox-', '') || '0');
  109. await firstAvailable.click();
  110. console.debug(`[后台] 选择残疾人 ID: ${testPersonId}`);
  111. // 6. 点击"确认选择"按钮
  112. // 人员进入"待添加人员列表"
  113. await adminPage.getByTestId('confirm-batch-button').click();
  114. await adminPage.waitForTimeout(TIMEOUTS.MEDIUM);
  115. console.debug('[后台] 确认选择残疾人');
  116. // 7. 验证待添加人员列表显示
  117. // 检查是否有"待添加人员列表"或"确认添加"按钮
  118. const confirmAddButton = adminPage.getByTestId('confirm-add-persons-button');
  119. await expect(confirmAddButton).toBeVisible();
  120. console.debug('[后台] 待添加人员列表已显示');
  121. // 8. 点击"确认添加"按钮完成绑定
  122. await confirmAddButton.click();
  123. await adminPage.waitForTimeout(TIMEOUTS.LONG);
  124. // 9. 验证 Toast 通知
  125. // 使用 filter 精确匹配"批量添加人员成功"消息(避免匹配"已添加到待添加列表")
  126. const successToast = adminPage.locator('[data-sonner-toast][data-type="success"]').filter({ hasText: '批量添加人员成功' });
  127. await expect(successToast).toBeVisible({ timeout: TIMEOUTS.VERY_LONG });
  128. const toastMessage = await successToast.textContent();
  129. console.debug(`[后台] Toast 消息: ${toastMessage}`);
  130. expect(toastMessage).toContain('批量添加人员成功');
  131. // 10. 验证绑定人员列表更新
  132. // 检查订单详情对话框中的绑定人员列表
  133. // 应该能看到新添加的人员
  134. const personRow = adminPage.locator('table tbody tr').filter({ hasText: '测试残疾人_1768346764677_11_9311' });
  135. await expect(personRow).toBeVisible();
  136. console.debug('[后台] 人员已成功添加到绑定人员列表');
  137. // 记录测试数据
  138. testState.personId = testPersonId;
  139. testState.personName = '测试残疾人_1768346764677_11_9311';
  140. const endTime = Date.now();
  141. const duration = endTime - startTime;
  142. console.debug(`[后台] 添加人员完成,耗时: ${duration}ms`);
  143. });
  144. });
  145. test.describe.serial('人才小程序验证', () => {
  146. test('应该在小程序中显示关联的订单', async ({ page: talentMiniPage }) => {
  147. // 1. 人才小程序登录
  148. await loginTalentMini(talentMiniPage);
  149. // 2. 记录同步开始时间
  150. const syncStartTime = Date.now();
  151. // 3. 导航到"我的订单"页面
  152. await talentMiniPage.navigateToMyOrders();
  153. console.debug('[人才小程序] 已导航到我的订单页面');
  154. // 4. 验证订单显示(使用轮询等待)
  155. // 由于数据同步可能有延迟,使用轮询检查
  156. const orderFound = await talentMiniPage.waitForOrderToAppear(TEST_ORDER_NAME, TEST_SYNC_TIMEOUT);
  157. const syncEndTime = Date.now();
  158. testState.syncTime = syncEndTime - syncStartTime;
  159. // 5. 验证订单存在
  160. expect(orderFound, `订单 "${TEST_ORDER_NAME}" 应该在人才小程序中显示`).toBe(true);
  161. console.debug(`[人才小程序] 订单已同步,耗时: ${testState.syncTime}ms`);
  162. // 6. 验证同步时间符合要求
  163. expect(testState.syncTime).toBeLessThanOrEqual(TEST_SYNC_TIMEOUT);
  164. console.debug(`[验证] 数据同步时间 ${testState.syncTime}ms 符合要求(≤ ${TEST_SYNC_TIMEOUT}ms)`);
  165. });
  166. test('应该在小程序订单详情中显示完整信息', async ({ page: talentMiniPage }) => {
  167. // 前置条件:已登录并在我的订单页面
  168. await loginTalentMini(talentMiniPage);
  169. await talentMiniPage.navigateToMyOrders();
  170. // 1. 导航到"我的订单"并找到测试订单
  171. const orders: TalentOrderData[] = await (talentMiniPage as unknown as TalentMiniPage).getMyOrders();
  172. const testOrder = orders.find(order => order.name === TEST_ORDER_NAME);
  173. expect(testOrder, `应该找到订单 "${TEST_ORDER_NAME}"`).toBeDefined();
  174. console.debug(`[人才小程序] 找到订单: ${testOrder?.name}`);
  175. // 2. 点击订单详情
  176. const orderId = await talentMiniPage.openOrderDetail(TEST_ORDER_NAME);
  177. expect(orderId).toBeTruthy();
  178. console.debug(`[人才小程序] 已打开订单详情: ${orderId}`);
  179. // 3. 验证订单信息完整性
  180. const detail: TalentOrderDetailData = await (talentMiniPage as unknown as TalentMiniPage).getOrderDetail();
  181. // 验证订单名称
  182. expect(detail.name).toBe(TEST_ORDER_NAME);
  183. console.debug(`[人才小程序] 订单名称: ${detail.name}`);
  184. // 验证公司名称
  185. expect(detail.companyName).toBeDefined();
  186. console.debug(`[人才小程序] 公司名称: ${detail.companyName}`);
  187. // 验证订单状态
  188. expect(detail.status).toBeDefined();
  189. console.debug(`[人才小程序] 订单状态: ${detail.status}`);
  190. console.debug('[人才小程序] 订单详情信息验证完成');
  191. });
  192. });
  193. test.describe.serial('数据同步时效性验证', () => {
  194. test('应该在合理时间内完成数据同步(≤ 10 秒)', async ({ page: _adminPage, page: _talentMiniPage, testUsers: _testUsers }) => {
  195. // 此测试专门验证数据同步时效性
  196. // AC6: 数据应在 5 秒内同步,最多等待 10 秒
  197. console.debug('[验证] 数据同步时效性测试');
  198. console.debug(`[验证] 要求: ≤ ${TEST_SYNC_TIMEOUT}ms`);
  199. console.debug(`[验证] 实际: ${testState.syncTime}ms`);
  200. console.debug(`[验证] 轮询间隔: ${TEST_POLL_INTERVAL}ms`);
  201. if (testState.syncTime !== null) {
  202. expect(testState.syncTime).toBeLessThanOrEqual(TEST_SYNC_TIMEOUT);
  203. console.debug(`[验证] ✅ 数据同步时间符合要求`);
  204. } else {
  205. console.warn('[验证] ⚠️ 未记录到同步时间,请检查测试执行');
  206. }
  207. });
  208. });
  209. test.describe.serial('多人员添加同步验证', () => {
  210. test('应该支持批量添加多个残疾人', async ({ page: adminPage, testUsers }) => {
  211. // AC3: 多人员添加同步验证
  212. // 此测试验证添加多个残疾人后小程序的同步情况
  213. // 1. 后台登录
  214. await loginAdmin(adminPage, testUsers);
  215. // 2. 打开订单详情
  216. await adminPage.goto('/admin/orders');
  217. await adminPage.getByTestId(`order-menu-trigger-${TEST_ORDER_ID}`).click();
  218. await adminPage.waitForTimeout(TIMEOUTS.VERY_SHORT);
  219. await adminPage.getByTestId(`view-order-detail-button-${TEST_ORDER_ID}`).click();
  220. await adminPage.waitForSelector('[role="dialog"]', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  221. // 3. 点击"添加人员"按钮
  222. await adminPage.getByTestId('order-detail-card-add-persons-button').click();
  223. await adminPage.waitForSelector('text=选择残疾人', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  224. // 4. 选择多个残疾人
  225. // 选择第一个和第二个可用的残疾人(未被绑定的)
  226. const checkboxes = adminPage.locator('[data-testid^="person-checkbox-"]:not([disabled])');
  227. const checkboxCount = await checkboxes.count();
  228. if (checkboxCount >= 2) {
  229. await checkboxes.nth(0).click();
  230. await checkboxes.nth(1).click();
  231. console.debug(`[后台] 已选择 ${Math.min(2, checkboxCount)} 个残疾人`);
  232. // 点击"确认选择"
  233. await adminPage.getByTestId('confirm-batch-button').click();
  234. await adminPage.waitForTimeout(TIMEOUTS.MEDIUM);
  235. // 点击"确认添加"
  236. await adminPage.getByTestId('confirm-add-persons-button').click();
  237. await adminPage.waitForTimeout(TIMEOUTS.LONG);
  238. // 验证成功 Toast
  239. const successToast = adminPage.locator('[data-sonner-toast][data-type="success"]').filter({ hasText: '批量添加人员成功' });
  240. await expect(successToast).toBeVisible({ timeout: TIMEOUTS.VERY_LONG });
  241. console.debug('[后台] 多人员添加成功');
  242. } else {
  243. console.debug(`[后台] 可用残疾人不足 2 个,跳过多人员测试`);
  244. }
  245. });
  246. });
  247. });