disability-person-bankcard.spec.ts 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416
  1. import { test, expect } from '../../utils/test-setup';
  2. import { readFileSync } from 'fs';
  3. import { join, dirname } from 'path';
  4. import { fileURLToPath } from 'url';
  5. const __filename = fileURLToPath(import.meta.url);
  6. const __dirname = dirname(__filename);
  7. const testUsers = JSON.parse(readFileSync(join(__dirname, '../../fixtures/test-users.json'), 'utf-8'));
  8. // 超时配置常量
  9. const TIMEOUTS = {
  10. SHORT: 300,
  11. MEDIUM: 500,
  12. LONG: 1000,
  13. DIALOG: 5000,
  14. } as const;
  15. /**
  16. * 生成唯一的测试数据(使用更大范围的随机值提高唯一性)
  17. */
  18. function generateUniqueTestData(suffix: string) {
  19. const randomPart = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER);
  20. const timestamp = Date.now();
  21. return {
  22. name: `银行卡${suffix}_${timestamp}_${randomPart}`,
  23. gender: randomPart % 2 === 0 ? '男' : '女',
  24. idCard: `42010119900101${String(randomPart % 10000).padStart(4, '0')}`,
  25. disabilityId: `511001199001${String(randomPart % 10000).padStart(4, '0')}`,
  26. disabilityType: ['视力残疾', '听力残疾', '言语残疾', '肢体残疾', '智力残疾', '精神残疾'][randomPart % 6],
  27. disabilityLevel: ['一级', '二级', '三级', '四级'][randomPart % 4],
  28. phone: `138${String(randomPart % 100000000).padStart(8, '0')}`,
  29. idAddress: `湖北省武汉市测试街道${randomPart % 100}号`,
  30. province: '湖北省',
  31. city: '武汉市'
  32. };
  33. }
  34. test.describe('残疾人管理 - 银行卡管理功能', () => {
  35. // 测试级别的时间戳,用于生成唯一数据
  36. const TEST_TIMESTAMP = Date.now();
  37. const TEST_PREFIX = `bankcard_${TEST_TIMESTAMP}`;
  38. // 用于跟踪已创建的测试数据,便于清理
  39. const createdTestData: Array<{ name: string; idCard: string }> = [];
  40. test.beforeEach(async ({ adminLoginPage, disabilityPersonPage }) => {
  41. // 每次测试前重置数据存储
  42. createdTestData.length = 0;
  43. // 以管理员身份登录后台
  44. await adminLoginPage.goto();
  45. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  46. await adminLoginPage.expectLoginSuccess();
  47. await disabilityPersonPage.goto();
  48. });
  49. test.afterEach(async ({ disabilityPersonPage, page }) => {
  50. // 清理测试数据(添加超时保护以避免 afterEach 超过 60 秒)
  51. for (const data of createdTestData) {
  52. try {
  53. await disabilityPersonPage.goto().catch(() => {});
  54. await disabilityPersonPage.searchByName(data.name);
  55. // 为每个清理操作设置较短的超时时间
  56. const deleteButton = page.getByRole('button', { name: '删除' }).first();
  57. if (await deleteButton.count({ timeout: TIMEOUTS.VERY_LONG }) > 0) {
  58. await deleteButton.click({ timeout: TIMEOUTS.DIALOG });
  59. await page.getByRole('button', { name: '确认' }).click({ timeout: TIMEOUTS.DIALOG }).catch(() => {});
  60. await page.waitForTimeout(TIMEOUTS.MEDIUM);
  61. }
  62. } catch (error) {
  63. console.debug(` ⚠ 清理数据失败: ${data.name}`, error);
  64. }
  65. }
  66. createdTestData.length = 0;
  67. });
  68. test('应该成功添加单张银行卡', async ({ disabilityPersonPage, page }) => {
  69. const testData = generateUniqueTestData('单张银行卡');
  70. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  71. console.debug('\n========== 单张银行卡添加测试 ==========');
  72. // 打开对话框并填写基本信息
  73. await disabilityPersonPage.openCreateDialog();
  74. await disabilityPersonPage.fillBasicForm(testData);
  75. // 滚动到银行卡管理区域
  76. await disabilityPersonPage.scrollToSection('银行卡管理');
  77. // 添加银行卡
  78. const bankCard = {
  79. bankName: '中国工商银行',
  80. subBankName: '北京分行朝阳支行',
  81. cardNumber: '6222021234567890123',
  82. cardholderName: testData.name,
  83. isDefault: true,
  84. };
  85. await disabilityPersonPage.addBankCard(bankCard);
  86. // 验证银行卡已添加
  87. const bankCardList = await disabilityPersonPage.getBankCardList();
  88. expect(bankCardList).toHaveLength(1);
  89. console.debug(' ✓ 银行卡列表数量: 1');
  90. // 验证默认银行卡
  91. const defaultIndex = await disabilityPersonPage.getDefaultBankCardIndex();
  92. expect(defaultIndex).toBe(0);
  93. console.debug(' ✓ 默认银行卡索引: 0');
  94. console.debug('✅ 单张银行卡添加测试通过');
  95. });
  96. test('应该成功编辑银行卡信息', async ({ disabilityPersonPage, page }) => {
  97. const testData = generateUniqueTestData('编辑银行卡');
  98. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  99. console.debug('\n========== 银行卡编辑测试 ==========');
  100. // 打开对话框并填写基本信息
  101. await disabilityPersonPage.openCreateDialog();
  102. await disabilityPersonPage.fillBasicForm(testData);
  103. // 添加原始银行卡
  104. const originalCard = {
  105. bankName: '中国工商银行',
  106. subBankName: '北京分行朝阳支行',
  107. cardNumber: '6222021234567890123',
  108. cardholderName: testData.name,
  109. };
  110. await disabilityPersonPage.addBankCard(originalCard);
  111. // 编辑银行卡
  112. const updatedCard = {
  113. bankName: '中国建设银行',
  114. subBankName: '上海分行浦东支行',
  115. cardNumber: '6227001234567890123',
  116. };
  117. await disabilityPersonPage.editBankCard(0, updatedCard);
  118. // 验证更新后的信息
  119. const bankCardList = await disabilityPersonPage.getBankCardList();
  120. expect(bankCardList).toHaveLength(1);
  121. // 验证银行卡号已更新(格式化后可能包含空格)
  122. const cardNumberInput = page.locator('[data-testid="card-number-input-0"]');
  123. const cardNumberValue = await cardNumberInput.inputValue();
  124. expect(cardNumberValue).toContain('6227');
  125. console.debug(' ✓ 银行卡号已更新');
  126. console.debug('✅ 银行卡编辑测试通过');
  127. });
  128. test('应该成功删除银行卡', async ({ disabilityPersonPage, page }) => {
  129. const testData = generateUniqueTestData('删除银行卡');
  130. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  131. console.debug('\n========== 银行卡删除测试 ==========');
  132. // 打开对话框并填写基本信息
  133. await disabilityPersonPage.openCreateDialog();
  134. await disabilityPersonPage.fillBasicForm(testData);
  135. // 添加银行卡
  136. const bankCard = {
  137. bankName: '中国工商银行',
  138. subBankName: '北京分行朝阳支行',
  139. cardNumber: '6222021234567890123',
  140. cardholderName: testData.name,
  141. };
  142. await disabilityPersonPage.addBankCard(bankCard);
  143. // 验证银行卡存在
  144. let bankCardList = await disabilityPersonPage.getBankCardList();
  145. expect(bankCardList).toHaveLength(1);
  146. console.debug(' ✓ 银行卡已添加');
  147. // 删除银行卡
  148. await disabilityPersonPage.deleteBankCard(0);
  149. // 验证银行卡已被删除
  150. bankCardList = await disabilityPersonPage.getBankCardList();
  151. expect(bankCardList).toHaveLength(0);
  152. console.debug(' ✓ 银行卡已删除');
  153. console.debug('✅ 银行卡删除测试通过');
  154. });
  155. test('应该支持添加多张银行卡', async ({ disabilityPersonPage, page }) => {
  156. const testData = generateUniqueTestData('多张银行卡');
  157. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  158. console.debug('\n========== 多张银行卡管理测试 ==========');
  159. // 打开对话框并填写基本信息
  160. await disabilityPersonPage.openCreateDialog();
  161. await disabilityPersonPage.fillBasicForm(testData);
  162. // 滚动到银行卡管理区域
  163. await disabilityPersonPage.scrollToSection('银行卡管理');
  164. // 添加三张银行卡
  165. await disabilityPersonPage.addBankCard({
  166. bankName: '中国工商银行',
  167. subBankName: '北京分行朝阳支行',
  168. cardNumber: '6222021234567890123',
  169. cardholderName: testData.name,
  170. });
  171. await disabilityPersonPage.addBankCard({
  172. bankName: '中国建设银行',
  173. subBankName: '上海分行浦东支行',
  174. cardNumber: '6227001234567890123',
  175. cardholderName: testData.name,
  176. });
  177. await disabilityPersonPage.addBankCard({
  178. bankName: '中国农业银行',
  179. subBankName: '广州分行天河支行',
  180. cardNumber: '6228481234567890123',
  181. cardholderName: testData.name,
  182. });
  183. // 验证所有银行卡都显示
  184. const bankCardList = await disabilityPersonPage.getBankCardList();
  185. expect(bankCardList).toHaveLength(3);
  186. console.debug(' ✓ 银行卡数量: 3');
  187. console.debug('✅ 多张银行卡管理测试通过');
  188. });
  189. test('应该能够设置默认银行卡', async ({ disabilityPersonPage, page }) => {
  190. const testData = generateUniqueTestData('默认银行卡');
  191. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  192. console.debug('\n========== 默认银行卡设置测试 ==========');
  193. // 打开对话框并填写基本信息
  194. await disabilityPersonPage.openCreateDialog();
  195. await disabilityPersonPage.fillBasicForm(testData);
  196. // 滚动到银行卡管理区域
  197. await disabilityPersonPage.scrollToSection('银行卡管理');
  198. // 添加第一张银行卡(设为默认)
  199. await disabilityPersonPage.addBankCard({
  200. bankName: '中国工商银行',
  201. subBankName: '北京分行朝阳支行',
  202. cardNumber: '6222021234567890123',
  203. cardholderName: testData.name,
  204. isDefault: true,
  205. });
  206. // 添加第二张银行卡(不设为默认)
  207. await disabilityPersonPage.addBankCard({
  208. bankName: '中国建设银行',
  209. subBankName: '上海分行浦东支行',
  210. cardNumber: '6227001234567890123',
  211. cardholderName: testData.name,
  212. isDefault: false,
  213. });
  214. // 验证默认银行卡是第一张(索引0)
  215. const defaultIndex = await disabilityPersonPage.getDefaultBankCardIndex();
  216. expect(defaultIndex).toBe(0);
  217. console.debug(' ✓ 默认银行卡是第一张');
  218. // 将第二张设为默认
  219. await disabilityPersonPage.editBankCard(1, { isDefault: true });
  220. await page.waitForTimeout(TIMEOUTS.SHORT);
  221. // 验证默认银行卡变为第二张,第一张不再是默认
  222. const defaultSwitch0 = page.locator('[data-testid="default-card-switch-0"]');
  223. const defaultSwitch1 = page.locator('[data-testid="default-card-switch-1"]');
  224. const isDefault0 = await defaultSwitch0.isChecked();
  225. const isDefault1 = await defaultSwitch1.isChecked();
  226. expect(isDefault0).toBe(false);
  227. expect(isDefault1).toBe(true);
  228. console.debug(' ✓ 默认银行卡已切换到第二张');
  229. console.debug('✅ 默认银行卡设置测试通过');
  230. });
  231. test('应该支持选择银行卡类型', async ({ disabilityPersonPage, page }) => {
  232. const testData = generateUniqueTestData('银行卡类型');
  233. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  234. console.debug('\n========== 银行卡类型选择测试 ==========');
  235. // 打开对话框并填写基本信息
  236. await disabilityPersonPage.openCreateDialog();
  237. await disabilityPersonPage.fillBasicForm(testData);
  238. // 添加一类卡
  239. await disabilityPersonPage.addBankCard({
  240. bankName: '中国工商银行',
  241. subBankName: '北京分行朝阳支行',
  242. cardNumber: '6222021234567890123',
  243. cardholderName: testData.name,
  244. cardType: '一类卡',
  245. });
  246. // 验证银行卡类型
  247. const cardTypeTrigger = page.locator('[data-testid="card-type-select-0"]');
  248. const selectedText = await cardTypeTrigger.textContent();
  249. expect(selectedText).toContain('一类卡');
  250. console.debug(' ✓ 银行卡类型: 一类卡');
  251. console.debug('✅ 银行卡类型选择测试通过');
  252. });
  253. test('应该限制最多添加5张银行卡', async ({ disabilityPersonPage, page }) => {
  254. const testData = generateUniqueTestData('银行卡数量限制');
  255. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  256. console.debug('\n========== 银行卡数量限制测试 ==========');
  257. // 打开对话框并填写基本信息
  258. await disabilityPersonPage.openCreateDialog();
  259. await disabilityPersonPage.fillBasicForm(testData);
  260. // 滚动到银行卡管理区域
  261. await disabilityPersonPage.scrollToSection('银行卡管理');
  262. // 尝试添加5张银行卡
  263. const banks = ['中国工商银行', '中国建设银行', '中国农业银行', '中国银行', '交通银行'];
  264. for (let i = 0; i < 5; i++) {
  265. await disabilityPersonPage.addBankCard({
  266. bankName: banks[i],
  267. subBankName: `${i}分行`,
  268. cardNumber: `622202123456789${String(i).padStart(3, '0')}`,
  269. cardholderName: testData.name,
  270. });
  271. }
  272. // 验证有5张银行卡
  273. let bankCardCount = await disabilityPersonPage.getBankCardCount();
  274. expect(bankCardCount).toBe(5);
  275. console.debug(' ✓ 已添加 5 张银行卡');
  276. // 尝试添加第6张银行卡
  277. const isDisabled = await disabilityPersonPage.isAddBankCardButtonDisabled();
  278. expect(isDisabled).toBe(true);
  279. console.debug(' ✓ 添加按钮已禁用(达到最大数量)');
  280. console.debug('✅ 银行卡数量限制测试通过');
  281. });
  282. test('完整流程:添加多张银行卡并提交', async ({ disabilityPersonPage, page }) => {
  283. const testData = generateUniqueTestData('完整流程');
  284. createdTestData.push({ name: testData.name, idCard: testData.idCard });
  285. console.debug('\n========== 银行卡完整流程测试 ==========');
  286. // 打开对话框并填写基本信息
  287. await disabilityPersonPage.openCreateDialog();
  288. await disabilityPersonPage.fillBasicForm(testData);
  289. // 滚动到银行卡管理区域
  290. await disabilityPersonPage.scrollToSection('银行卡管理');
  291. // 添加多张银行卡
  292. await disabilityPersonPage.addBankCard({
  293. bankName: '中国工商银行',
  294. subBankName: '北京分行朝阳支行',
  295. cardNumber: '6222021234567890123',
  296. cardholderName: testData.name,
  297. cardType: '一类卡',
  298. isDefault: true,
  299. });
  300. await disabilityPersonPage.addBankCard({
  301. bankName: '中国建设银行',
  302. subBankName: '上海分行浦东支行',
  303. cardNumber: '6227001234567890123',
  304. cardholderName: testData.name,
  305. cardType: '二类卡',
  306. });
  307. // 验证银行卡数量
  308. const bankCardList = await disabilityPersonPage.getBankCardList();
  309. expect(bankCardList).toHaveLength(2);
  310. // 提交表单
  311. const result = await disabilityPersonPage.submitForm();
  312. console.debug('\n========== 测试结果 ==========');
  313. console.debug('有错误提示:', result.hasError);
  314. console.debug('有成功提示:', result.hasSuccess);
  315. if (result.hasError) {
  316. console.debug('错误消息:', result.errorMessage);
  317. }
  318. if (result.hasSuccess) {
  319. console.debug('成功消息:', result.successMessage);
  320. }
  321. // 验证对话框关闭
  322. await disabilityPersonPage.waitForDialogClosed();
  323. // 刷新页面并验证数据创建成功(修复稳定性问题)
  324. await page.reload();
  325. await page.waitForLoadState('networkidle');
  326. await disabilityPersonPage.goto();
  327. await disabilityPersonPage.searchByName(testData.name);
  328. // 增加等待时间以确保数据已持久化
  329. await page.waitForTimeout(TIMEOUTS.EXTENDED);
  330. const personExists = await disabilityPersonPage.personExists(testData.name);
  331. console.debug('数据创建成功:', personExists);
  332. expect(personExists).toBe(true);
  333. console.debug('✅ 银行卡完整流程测试完成');
  334. });
  335. });