admin-routes.integration.test.ts 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655
  1. import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
  2. import { testClient } from 'hono/testing';
  3. import { IntegrationTestDatabase, setupIntegrationDatabaseHooksWithEntities } from '@d8d/shared-test-util';
  4. import { JWTUtil } from '@d8d/shared-utils';
  5. import { UserEntity, Role } from '@d8d/user-module';
  6. import { AreaEntity, AreaLevel } from '@d8d/geo-areas';
  7. import { File } from '@d8d/file-module';
  8. import { adminDeliveryAddressRoutes } from '../../src/routes';
  9. import { DeliveryAddress } from '../../src/entities';
  10. // 设置集成测试钩子
  11. setupIntegrationDatabaseHooksWithEntities([UserEntity, Role, AreaEntity, DeliveryAddress, File])
  12. describe('管理员配送地址管理API集成测试', () => {
  13. let client: ReturnType<typeof testClient<typeof adminDeliveryAddressRoutes>>;
  14. let adminToken: string;
  15. let testUser: UserEntity;
  16. let testAdmin: UserEntity;
  17. let testProvince: AreaEntity;
  18. let testCity: AreaEntity;
  19. let testDistrict: AreaEntity;
  20. beforeEach(async () => {
  21. // 创建测试客户端
  22. client = testClient(adminDeliveryAddressRoutes);
  23. // 获取数据源
  24. const dataSource = await IntegrationTestDatabase.getDataSource();
  25. // 创建测试用户
  26. const userRepository = dataSource.getRepository(UserEntity);
  27. testUser = userRepository.create({
  28. username: `test_user_${Date.now()}`,
  29. password: 'test_password',
  30. nickname: '测试用户',
  31. registrationSource: 'web'
  32. });
  33. await userRepository.save(testUser);
  34. // 创建测试管理员用户
  35. testAdmin = userRepository.create({
  36. username: `test_admin_${Date.now()}`,
  37. password: 'admin_password',
  38. nickname: '测试管理员',
  39. registrationSource: 'web'
  40. });
  41. await userRepository.save(testAdmin);
  42. // 创建测试地区数据 - 省
  43. const areaRepository = dataSource.getRepository(AreaEntity);
  44. testProvince = areaRepository.create({
  45. name: '北京市',
  46. code: '110000',
  47. level: AreaLevel.PROVINCE,
  48. parentId: null
  49. });
  50. await areaRepository.save(testProvince);
  51. // 创建测试地区数据 - 市
  52. testCity = areaRepository.create({
  53. name: '北京市',
  54. code: '110100',
  55. level: AreaLevel.CITY,
  56. parentId: testProvince.id
  57. });
  58. await areaRepository.save(testCity);
  59. // 创建测试地区数据 - 区
  60. testDistrict = areaRepository.create({
  61. name: '朝阳区',
  62. code: '110105',
  63. level: AreaLevel.DISTRICT,
  64. parentId: testCity.id
  65. });
  66. await areaRepository.save(testDistrict);
  67. // 生成测试管理员的token
  68. adminToken = JWTUtil.generateToken({
  69. id: testAdmin.id,
  70. username: testAdmin.username,
  71. roles: [{name:'admin'}]
  72. });
  73. });
  74. describe('GET /delivery-address', () => {
  75. it('应该返回配送地址列表', async () => {
  76. const response = await client.index.$get({
  77. query: {}
  78. }, {
  79. headers: {
  80. 'Authorization': `Bearer ${adminToken}`
  81. }
  82. });
  83. console.debug('配送地址列表响应状态:', response.status);
  84. expect(response.status).toBe(200);
  85. if (response.status === 200) {
  86. const data = await response.json();
  87. expect(data).toHaveProperty('data');
  88. expect(Array.isArray(data.data)).toBe(true);
  89. }
  90. });
  91. it('应该拒绝未认证用户的访问', async () => {
  92. const response = await client.index.$get({
  93. query: {}
  94. });
  95. expect(response.status).toBe(401);
  96. });
  97. });
  98. describe('POST /delivery-address', () => {
  99. it('应该成功创建配送地址', async () => {
  100. const createData = {
  101. userId: testUser.id,
  102. name: '张三',
  103. phone: '13800138000',
  104. address: '朝阳区建国路88号',
  105. receiverProvince: testProvince.id,
  106. receiverCity: testCity.id,
  107. receiverDistrict: testDistrict.id,
  108. receiverTown: 0,
  109. state: 1,
  110. isDefault: 1
  111. };
  112. const response = await client.index.$post({
  113. json: createData
  114. }, {
  115. headers: {
  116. 'Authorization': `Bearer ${adminToken}`
  117. }
  118. });
  119. console.debug('创建配送地址响应状态:', response.status);
  120. expect(response.status).toBe(201);
  121. if (response.status === 201) {
  122. const data = await response.json();
  123. expect(data).toHaveProperty('id');
  124. expect(data.name).toBe(createData.name);
  125. expect(data.phone).toBe(createData.phone);
  126. expect(data.address).toBe(createData.address);
  127. expect(data.receiverProvince).toBe(createData.receiverProvince);
  128. expect(data.receiverCity).toBe(createData.receiverCity);
  129. expect(data.receiverDistrict).toBe(createData.receiverDistrict);
  130. expect(data.isDefault).toBe(createData.isDefault);
  131. }
  132. });
  133. it('应该验证创建配送地址的必填字段', async () => {
  134. const invalidData = {
  135. // 缺少必填字段
  136. userId: testUser.id,
  137. name: '',
  138. phone: '',
  139. address: '',
  140. receiverProvince: 0,
  141. receiverCity: 0,
  142. receiverDistrict: 0
  143. };
  144. const response = await client.index.$post({
  145. json: invalidData
  146. }, {
  147. headers: {
  148. 'Authorization': `Bearer ${adminToken}`
  149. }
  150. });
  151. expect(response.status).toBe(400);
  152. });
  153. it('应该验证地区层级关系', async () => {
  154. // 使用不存在的地区ID
  155. const invalidAreaData = {
  156. userId: testUser.id,
  157. name: '李四',
  158. phone: '13900139000',
  159. address: '测试地址',
  160. receiverProvince: 999999, // 不存在的省份
  161. receiverCity: testCity.id,
  162. receiverDistrict: testDistrict.id,
  163. receiverTown: 0,
  164. state: 1,
  165. isDefault: 0
  166. };
  167. const response = await client.index.$post({
  168. json: invalidAreaData
  169. }, {
  170. headers: {
  171. 'Authorization': `Bearer ${adminToken}`
  172. }
  173. });
  174. console.debug('地区验证响应状态:', response.status);
  175. expect(response.status).toBe(400);
  176. });
  177. });
  178. describe('GET /delivery-address/:id', () => {
  179. it('应该返回指定配送地址的详情', async () => {
  180. // 先创建一个配送地址
  181. const dataSource = await IntegrationTestDatabase.getDataSource();
  182. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  183. const testDeliveryAddress = deliveryAddressRepository.create({
  184. userId: testUser.id,
  185. name: '王五',
  186. phone: '13600136000',
  187. address: '海淀区中关村大街1号',
  188. receiverProvince: testProvince.id,
  189. receiverCity: testCity.id,
  190. receiverDistrict: testDistrict.id,
  191. receiverTown: 0,
  192. state: 1,
  193. isDefault: 0,
  194. createdBy: testUser.id
  195. });
  196. await deliveryAddressRepository.save(testDeliveryAddress);
  197. const response = await client[':id'].$get({
  198. param: { id: testDeliveryAddress.id }
  199. }, {
  200. headers: {
  201. 'Authorization': `Bearer ${adminToken}`
  202. }
  203. });
  204. console.debug('配送地址详情响应状态:', response.status);
  205. expect(response.status).toBe(200);
  206. if (response.status === 200) {
  207. const data = await response.json();
  208. expect(data.id).toBe(testDeliveryAddress.id);
  209. expect(data.name).toBe(testDeliveryAddress.name);
  210. expect(data.phone).toBe(testDeliveryAddress.phone);
  211. expect(data.address).toBe(testDeliveryAddress.address);
  212. // 验证地区数据关联
  213. if (data.province) {
  214. expect(data.province.id).toBe(testProvince.id);
  215. }
  216. if (data.city) {
  217. expect(data.city.id).toBe(testCity.id);
  218. }
  219. if (data.district) {
  220. expect(data.district.id).toBe(testDistrict.id);
  221. }
  222. }
  223. });
  224. it('应该处理不存在的配送地址', async () => {
  225. const response = await client[':id'].$get({
  226. param: { id: 999999 }
  227. }, {
  228. headers: {
  229. 'Authorization': `Bearer ${adminToken}`
  230. }
  231. });
  232. expect(response.status).toBe(404);
  233. });
  234. });
  235. describe('PUT /delivery-address/:id', () => {
  236. it('应该成功更新配送地址', async () => {
  237. // 先创建一个配送地址
  238. const dataSource = await IntegrationTestDatabase.getDataSource();
  239. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  240. const testDeliveryAddress = deliveryAddressRepository.create({
  241. userId: testUser.id,
  242. name: '原始姓名',
  243. phone: '13500135000',
  244. address: '原始地址',
  245. receiverProvince: testProvince.id,
  246. receiverCity: testCity.id,
  247. receiverDistrict: testDistrict.id,
  248. receiverTown: 0,
  249. state: 1,
  250. isDefault: 0,
  251. createdBy: testUser.id
  252. });
  253. await deliveryAddressRepository.save(testDeliveryAddress);
  254. const updateData = {
  255. name: '更新后的姓名',
  256. phone: '13700137000',
  257. address: '更新后的地址',
  258. isDefault: 1
  259. };
  260. const response = await client[':id'].$put({
  261. param: { id: testDeliveryAddress.id },
  262. json: updateData
  263. }, {
  264. headers: {
  265. 'Authorization': `Bearer ${adminToken}`
  266. }
  267. });
  268. console.debug('更新配送地址响应状态:', response.status);
  269. expect(response.status).toBe(200);
  270. if (response.status === 200) {
  271. const data = await response.json();
  272. expect(data.name).toBe(updateData.name);
  273. expect(data.phone).toBe(updateData.phone);
  274. expect(data.address).toBe(updateData.address);
  275. expect(data.isDefault).toBe(updateData.isDefault);
  276. }
  277. });
  278. });
  279. describe('DELETE /delivery-address/:id', () => {
  280. it('应该成功删除配送地址', async () => {
  281. // 先创建一个配送地址
  282. const dataSource = await IntegrationTestDatabase.getDataSource();
  283. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  284. const testDeliveryAddress = deliveryAddressRepository.create({
  285. userId: testUser.id,
  286. name: '待删除地址',
  287. phone: '13400134000',
  288. address: '待删除地址',
  289. receiverProvince: testProvince.id,
  290. receiverCity: testCity.id,
  291. receiverDistrict: testDistrict.id,
  292. receiverTown: 0,
  293. state: 1,
  294. isDefault: 0,
  295. createdBy: testUser.id
  296. });
  297. await deliveryAddressRepository.save(testDeliveryAddress);
  298. const response = await client[':id'].$delete({
  299. param: { id: testDeliveryAddress.id }
  300. }, {
  301. headers: {
  302. 'Authorization': `Bearer ${adminToken}`
  303. }
  304. });
  305. console.debug('删除配送地址响应状态:', response.status);
  306. expect(response.status).toBe(204);
  307. // 验证配送地址确实被删除
  308. const deletedDeliveryAddress = await deliveryAddressRepository.findOne({
  309. where: { id: testDeliveryAddress.id }
  310. });
  311. expect(deletedDeliveryAddress).toBeNull();
  312. });
  313. });
  314. describe('省市区关联测试', () => {
  315. it('应该正确关联省市区数据', async () => {
  316. // 创建配送地址
  317. const dataSource = await IntegrationTestDatabase.getDataSource();
  318. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  319. const testDeliveryAddress = deliveryAddressRepository.create({
  320. userId: testUser.id,
  321. name: '关联测试',
  322. phone: '13300133000',
  323. address: '关联测试地址',
  324. receiverProvince: testProvince.id,
  325. receiverCity: testCity.id,
  326. receiverDistrict: testDistrict.id,
  327. receiverTown: 0,
  328. state: 1,
  329. isDefault: 0,
  330. createdBy: testUser.id
  331. });
  332. await deliveryAddressRepository.save(testDeliveryAddress);
  333. // 查询配送地址详情,验证地区关联
  334. const response = await client[':id'].$get({
  335. param: { id: testDeliveryAddress.id }
  336. }, {
  337. headers: {
  338. 'Authorization': `Bearer ${adminToken}`
  339. }
  340. });
  341. expect(response.status).toBe(200);
  342. const data = await response.json();
  343. // 验证省市区关联数据
  344. if ('province' in data && data.province) {
  345. expect(data.province.name).toBe('北京市');
  346. expect(data.province.level).toBe(AreaLevel.PROVINCE);
  347. }
  348. if ('city' in data && data.city) {
  349. expect(data.city.name).toBe('北京市');
  350. expect(data.city.level).toBe(AreaLevel.CITY);
  351. }
  352. if ('district' in data && data.district) {
  353. expect(data.district.name).toBe('朝阳区');
  354. expect(data.district.level).toBe(AreaLevel.DISTRICT);
  355. }
  356. });
  357. it('应该验证地区层级关系', async () => {
  358. // 创建另一个区级地区,但父级不是市级
  359. const dataSource = await IntegrationTestDatabase.getDataSource();
  360. const areaRepository = dataSource.getRepository(AreaEntity);
  361. const invalidDistrict = areaRepository.create({
  362. name: '无效区',
  363. code: '999999',
  364. level: AreaLevel.DISTRICT,
  365. parentId: testProvince.id // 父级是省,不是市
  366. });
  367. await areaRepository.save(invalidDistrict);
  368. // 尝试使用无效的地区层级关系创建地址
  369. const createData = {
  370. userId: testUser.id,
  371. name: '层级测试',
  372. phone: '13200132000',
  373. address: '层级测试地址',
  374. receiverProvince: testProvince.id,
  375. receiverCity: testCity.id,
  376. receiverDistrict: invalidDistrict.id, // 这个区的父级不是testCity
  377. receiverTown: 0,
  378. state: 1,
  379. isDefault: 0
  380. };
  381. const response = await client.index.$post({
  382. json: createData
  383. }, {
  384. headers: {
  385. 'Authorization': `Bearer ${adminToken}`
  386. }
  387. });
  388. console.debug('地区层级验证响应状态:', response.status);
  389. // 这里期望返回400,因为地区层级关系不匹配
  390. expect(response.status).toBe(400);
  391. });
  392. });
  393. describe('管理员权限测试', () => {
  394. it('管理员应该可以为其他用户创建地址', async () => {
  395. const createData = {
  396. userId: testUser.id, // 为其他用户创建地址
  397. name: '其他用户地址',
  398. phone: '13800138001',
  399. address: '其他用户地址',
  400. receiverProvince: testProvince.id,
  401. receiverCity: testCity.id,
  402. receiverDistrict: testDistrict.id,
  403. receiverTown: 0,
  404. state: 1,
  405. isDefault: 0
  406. };
  407. const response = await client.index.$post({
  408. json: createData
  409. }, {
  410. headers: {
  411. 'Authorization': `Bearer ${adminToken}`
  412. }
  413. });
  414. console.debug('管理员为其他用户创建地址响应状态:', response.status);
  415. expect(response.status).toBe(201);
  416. if (response.status === 201) {
  417. const data = await response.json();
  418. expect(data.userId).toBe(testUser.id); // 验证地址确实属于其他用户
  419. expect(data.name).toBe(createData.name);
  420. }
  421. });
  422. it('管理员应该可以访问所有用户的地址', async () => {
  423. // 为测试用户创建一些地址
  424. const dataSource = await IntegrationTestDatabase.getDataSource();
  425. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  426. const userAddress1 = deliveryAddressRepository.create({
  427. userId: testUser.id,
  428. name: '用户地址1',
  429. phone: '13800138002',
  430. address: '用户地址1',
  431. receiverProvince: testProvince.id,
  432. receiverCity: testCity.id,
  433. receiverDistrict: testDistrict.id,
  434. receiverTown: 0,
  435. state: 1,
  436. isDefault: 0,
  437. createdBy: testUser.id
  438. });
  439. await deliveryAddressRepository.save(userAddress1);
  440. const userAddress2 = deliveryAddressRepository.create({
  441. userId: testUser.id,
  442. name: '用户地址2',
  443. phone: '13800138003',
  444. address: '用户地址2',
  445. receiverProvince: testProvince.id,
  446. receiverCity: testCity.id,
  447. receiverDistrict: testDistrict.id,
  448. receiverTown: 0,
  449. state: 1,
  450. isDefault: 0,
  451. createdBy: testUser.id
  452. });
  453. await deliveryAddressRepository.save(userAddress2);
  454. // 管理员应该能看到所有地址
  455. const response = await client.index.$get({
  456. query: {}
  457. }, {
  458. headers: {
  459. 'Authorization': `Bearer ${adminToken}`
  460. }
  461. });
  462. expect(response.status).toBe(200);
  463. const data = await response.json();
  464. if (data && 'data' in data) {
  465. expect(Array.isArray(data.data)).toBe(true);
  466. expect(data.data.length).toBeGreaterThanOrEqual(2); // 至少包含我们创建的两个地址
  467. }
  468. });
  469. it('管理员应该可以更新其他用户的地址', async () => {
  470. // 先为测试用户创建一个地址
  471. const dataSource = await IntegrationTestDatabase.getDataSource();
  472. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  473. const testDeliveryAddress = deliveryAddressRepository.create({
  474. userId: testUser.id,
  475. name: '原始地址',
  476. phone: '13800138004',
  477. address: '原始地址',
  478. receiverProvince: testProvince.id,
  479. receiverCity: testCity.id,
  480. receiverDistrict: testDistrict.id,
  481. receiverTown: 0,
  482. state: 1,
  483. isDefault: 0,
  484. createdBy: testUser.id
  485. });
  486. await deliveryAddressRepository.save(testDeliveryAddress);
  487. const updateData = {
  488. name: '管理员更新的地址',
  489. phone: '13900139000',
  490. address: '管理员更新的地址'
  491. };
  492. const response = await client[':id'].$put({
  493. param: { id: testDeliveryAddress.id },
  494. json: updateData
  495. }, {
  496. headers: {
  497. 'Authorization': `Bearer ${adminToken}`
  498. }
  499. });
  500. console.debug('管理员更新其他用户地址响应状态:', response.status);
  501. expect(response.status).toBe(200);
  502. if (response.status === 200) {
  503. const data = await response.json();
  504. expect(data.name).toBe(updateData.name);
  505. expect(data.phone).toBe(updateData.phone);
  506. expect(data.address).toBe(updateData.address);
  507. }
  508. });
  509. it('管理员应该可以删除其他用户的地址', async () => {
  510. // 先为测试用户创建一个地址
  511. const dataSource = await IntegrationTestDatabase.getDataSource();
  512. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  513. const testDeliveryAddress = deliveryAddressRepository.create({
  514. userId: testUser.id,
  515. name: '待删除地址',
  516. phone: '13800138005',
  517. address: '待删除地址',
  518. receiverProvince: testProvince.id,
  519. receiverCity: testCity.id,
  520. receiverDistrict: testDistrict.id,
  521. receiverTown: 0,
  522. state: 1,
  523. isDefault: 0,
  524. createdBy: testUser.id
  525. });
  526. await deliveryAddressRepository.save(testDeliveryAddress);
  527. const response = await client[':id'].$delete({
  528. param: { id: testDeliveryAddress.id }
  529. }, {
  530. headers: {
  531. 'Authorization': `Bearer ${adminToken}`
  532. }
  533. });
  534. console.debug('管理员删除其他用户地址响应状态:', response.status);
  535. expect(response.status).toBe(204);
  536. // 验证地址确实被删除
  537. const deletedDeliveryAddress = await deliveryAddressRepository.findOne({
  538. where: { id: testDeliveryAddress.id }
  539. });
  540. expect(deletedDeliveryAddress).toBeNull();
  541. });
  542. it('管理员应该可以查询指定用户的地址', async () => {
  543. // 为测试用户创建一些地址
  544. const dataSource = await IntegrationTestDatabase.getDataSource();
  545. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  546. const userAddress = deliveryAddressRepository.create({
  547. userId: testUser.id,
  548. name: '指定用户地址',
  549. phone: '13800138006',
  550. address: '指定用户地址',
  551. receiverProvince: testProvince.id,
  552. receiverCity: testCity.id,
  553. receiverDistrict: testDistrict.id,
  554. receiverTown: 0,
  555. state: 1,
  556. isDefault: 0,
  557. createdBy: testUser.id
  558. });
  559. await deliveryAddressRepository.save(userAddress);
  560. // 管理员可以查询指定用户的地址
  561. const response = await client.index.$get({
  562. query: { filters: JSON.stringify({ userId: testUser.id }) }
  563. }, {
  564. headers: {
  565. 'Authorization': `Bearer ${adminToken}`
  566. }
  567. });
  568. expect(response.status).toBe(200);
  569. const data = await response.json();
  570. if (data && 'data' in data) {
  571. expect(Array.isArray(data.data)).toBe(true);
  572. // 验证返回的地址都属于指定用户
  573. if (data.data.length > 0) {
  574. data.data.forEach((address: any) => {
  575. expect(address.userId).toBe(testUser.id);
  576. });
  577. }
  578. }
  579. });
  580. });
  581. });