person-add-sync.spec.ts 13 KB

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