order-create-sync.spec.ts 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  1. import { TIMEOUTS } from '../../utils/timeouts';
  2. import { test, expect } from '../../utils/test-setup';
  3. /**
  4. * 跨端数据同步 E2E 测试
  5. *
  6. * 测试目标:验证后台创建订单后,企业小程序能否正确显示该订单
  7. *
  8. * 测试流程:
  9. * 1. 后台操作:登录 → 导航到订单管理 → 创建测试订单 → 验证创建成功
  10. * 2. 小程序验证:登录 → 导航到订单列表 → 验证订单存在 → 查看详情
  11. *
  12. * 测试要点:
  13. * - 使用两个独立的 browser context(后台和小程序)
  14. * - 记录数据同步时间
  15. * - 使用 data-testid 选择器
  16. * - 遵循项目测试规范
  17. */
  18. // 测试常量
  19. const TEST_SYNC_TIMEOUT = 3000; // 数据同步等待时间(ms)
  20. const TEST_COMPANY_NAME = '测试公司_1768346782396'; // 测试公司名称
  21. const MINI_LOGIN_PHONE = '13800001111'; // 小程序登录手机号
  22. const MINI_LOGIN_PASSWORD = 'password123'; // 小程序登录密码
  23. test.describe('跨端数据同步测试 - 后台创建订单到企业小程序', () => {
  24. let orderName: string;
  25. test.describe.serial('后台创建订单', () => {
  26. test('应该成功登录后台并创建订单', async ({ page: adminPage, testUsers }) => {
  27. // 记录开始时间
  28. const startTime = Date.now();
  29. // 1. 后台登录
  30. await adminPage.goto('/admin/login');
  31. await adminPage.getByPlaceholder('请输入用户名').fill(testUsers.admin.username);
  32. await adminPage.getByPlaceholder('请输入密码').fill(testUsers.admin.password);
  33. await adminPage.getByRole('button', { name: '登录' }).click();
  34. await adminPage.waitForURL('**/admin/dashboard', { timeout: TIMEOUTS.PAGE_LOAD });
  35. console.debug('[后台] 登录成功');
  36. // 2. 导航到订单管理页面
  37. await adminPage.goto('/admin/orders');
  38. await adminPage.waitForSelector('table tbody tr', { state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
  39. console.debug('[后台] 导航到订单管理页面');
  40. // 3. 点击创建订单按钮
  41. await adminPage.getByTestId('create-order-button').click();
  42. await adminPage.waitForSelector('[role="dialog"]', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  43. console.debug('[后台] 打开创建订单对话框');
  44. // 4. 填写订单表单
  45. const timestamp = Date.now();
  46. orderName = `跨端同步测试_${timestamp}`;
  47. // 填写订单名称
  48. await adminPage.getByLabel('订单名称').fill(orderName);
  49. console.debug(`[后台] 订单名称: ${orderName}`);
  50. // 选择平台(选择第一个可用平台)
  51. await adminPage.getByTestId('platform-selector-create').click();
  52. await adminPage.waitForTimeout(TIMEOUTS.SHORT);
  53. const firstPlatformOption = adminPage.locator('[role="option"]').first();
  54. const platformName = await firstPlatformOption.textContent();
  55. await firstPlatformOption.click();
  56. console.debug(`[后台] 选择平台: ${platformName}`);
  57. // 选择公司(选择与测试用户关联的公司:测试公司_1768346782396)
  58. await adminPage.getByTestId('company-selector-create').click();
  59. await adminPage.waitForTimeout(TIMEOUTS.SHORT);
  60. // 选择测试公司_1768346782396(与 13800001111 用户关联的公司)
  61. const companyOption = adminPage.getByRole('option', { name: TEST_COMPANY_NAME });
  62. await companyOption.click();
  63. console.debug(`[后台] 选择公司: ${TEST_COMPANY_NAME}`);
  64. // 5. 选择残疾人
  65. // 首先检查是否已经有残疾人被选择(可能在之前的测试中)
  66. // 通过检查是否存在显示残疾人详情的区域来判断
  67. const hasExistingPerson = await adminPage.getByText('残疾证号:', { exact: false }).count() > 0;
  68. if (hasExistingPerson) {
  69. console.debug('[后台] 订单中已有残疾人,跳过选择步骤');
  70. } else {
  71. // 需要选择残疾人,点击"选择残疾人"按钮
  72. await adminPage.getByTestId('select-persons-button').click();
  73. console.debug('[后台] 点击选择残疾人按钮');
  74. // 等待残疾人选择对话框出现
  75. await adminPage.waitForSelector('text=选择残疾人', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  76. console.debug('[后台] 选择残疾人对话框已打开');
  77. // 等待残疾人列表加载
  78. await adminPage.waitForSelector('table tbody tr', { state: 'visible', timeout: TIMEOUTS.TABLE_LOAD });
  79. // 选择第一个残疾人 - 使用复选框选择器
  80. const firstCheckbox = adminPage.locator('[data-testid^="person-checkbox-"]').first();
  81. await firstCheckbox.click();
  82. console.debug('[后台] 已勾选第一个残疾人');
  83. // 等待确认按钮变为可用状态
  84. await adminPage.waitForSelector('[data-testid="confirm-batch-button"]:not([disabled])', { timeout: TIMEOUTS.SHORT });
  85. // 点击确认选择按钮
  86. await adminPage.getByTestId('confirm-batch-button').click();
  87. await adminPage.waitForTimeout(TIMEOUTS.MEDIUM);
  88. console.debug('[后台] 已确认选择残疾人');
  89. }
  90. // 等待 UI 稳定
  91. await adminPage.waitForTimeout(1000);
  92. // 6. 提交表单
  93. await adminPage.getByTestId('order-create-submit-button').click();
  94. await adminPage.waitForLoadState('domcontentloaded');
  95. await adminPage.waitForTimeout(TIMEOUTS.LONG);
  96. console.debug('[后台] 提交订单表单');
  97. // 7. 验证创建成功的 Toast
  98. const successToast = adminPage.locator('[data-sonner-toast][data-type="success"]');
  99. const toastCount = await successToast.count();
  100. expect(toastCount).toBeGreaterThan(0);
  101. console.debug('[后台] 订单创建成功');
  102. // 8. 验证订单出现在列表中
  103. const orderExists = adminPage.locator('table tbody tr').filter({ hasText: orderName }).count() > 0;
  104. expect(orderExists).toBe(true);
  105. console.debug(`[后台] 订单 "${orderName}" 出现在列表中`);
  106. // 9. 记录创建完成时间
  107. const endTime = Date.now();
  108. const syncTime = endTime - startTime;
  109. console.debug(`[后台] 订单创建完成,耗时: ${syncTime}ms`);
  110. // 保存订单名称到环境变量,供后续测试使用
  111. process.env.__TEST_ORDER_NAME__ = orderName;
  112. });
  113. });
  114. test.describe.serial('小程序验证订单同步', () => {
  115. test.use({ storageState: undefined }); // 确保使用新的浏览器上下文
  116. test('应该在小程序中显示后台创建的订单', async ({ page: miniPage }) => {
  117. // 从环境变量获取订单名称
  118. const testOrderName = process.env.__TEST_ORDER_NAME__;
  119. if (!testOrderName) {
  120. throw new Error('未找到测试订单名称,请先运行后台创建订单测试');
  121. }
  122. console.debug(`[小程序] 查找订单: ${testOrderName}`);
  123. // 等待一段时间,确保数据同步完成
  124. await new Promise(resolve => setTimeout(resolve, TEST_SYNC_TIMEOUT));
  125. // 1. 小程序登录
  126. await miniPage.goto('/mini');
  127. await miniPage.waitForLoadState('networkidle');
  128. // 填写登录表单(使用企业账号)
  129. await miniPage.getByTestId('mini-phone-input').getByPlaceholder('请输入手机号').fill(MINI_LOGIN_PHONE);
  130. await miniPage.getByRole('textbox', { name: '请输入密码' }).fill(MINI_LOGIN_PASSWORD);
  131. await miniPage.getByTestId('mini-login-button').click();
  132. console.debug('[小程序] 登录请求已发送');
  133. // 等待登录成功(跳转到 dashboard)
  134. await miniPage.waitForURL(
  135. url => url.pathname.includes('/dashboard') || url.pathname.includes('/pages/yongren/dashboard'),
  136. { timeout: TIMEOUTS.PAGE_LOAD }
  137. );
  138. console.debug('[小程序] 登录成功');
  139. // 2. 导航到订单列表页面
  140. await miniPage.getByText('订单', { exact: true }).click();
  141. await miniPage.waitForLoadState('domcontentloaded');
  142. console.debug('[小程序] 导航到订单列表页面');
  143. // 3. 等待订单列表加载
  144. await miniPage.waitForTimeout(TIMEOUTS.LONG);
  145. // 4. 验证订单出现在小程序列表中
  146. // 使用多种方法验证订单存在
  147. const orderExistsByLocator = await miniPage.locator('text=' + testOrderName).count() > 0;
  148. const orderExistsByGetByText = await miniPage.getByText(testOrderName).count() > 0;
  149. const orderExists = orderExistsByLocator || orderExistsByGetByText;
  150. expect(orderExists).toBe(true);
  151. console.debug(`[小程序] 订单 "${testOrderName}" 已同步到小程序`);
  152. // 5. 点击查看详情
  153. const detailButton = miniPage.getByText('查看详情').first();
  154. await detailButton.click();
  155. await miniPage.waitForURL(/\/detail/, { timeout: TIMEOUTS.PAGE_LOAD });
  156. console.debug('[小程序] 打开订单详情');
  157. // 6. 验证订单详情信息
  158. // 验证订单名称显示
  159. const orderNameElement = miniPage.getByText(testOrderName);
  160. await expect(orderNameElement).toBeVisible();
  161. console.debug('[小程序] 订单详情显示正确');
  162. // 7. 记录数据同步完成时间
  163. const syncEndTime = Date.now();
  164. console.debug(`[小程序] 数据同步验证完成,时间戳: ${syncEndTime}`);
  165. // 清理环境变量
  166. delete process.env.__TEST_ORDER_NAME__;
  167. });
  168. });
  169. });