2
0

admin-routes.integration.test.ts 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659
  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: 1, // 使用有效的正整数
  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. if (response.status !== 201) {
  121. const errorData = await response.json();
  122. console.debug('创建配送地址错误响应:', errorData);
  123. }
  124. expect(response.status).toBe(201);
  125. if (response.status === 201) {
  126. const data = await response.json();
  127. expect(data).toHaveProperty('id');
  128. expect(data.name).toBe(createData.name);
  129. expect(data.phone).toBe(createData.phone);
  130. expect(data.address).toBe(createData.address);
  131. expect(data.receiverProvince).toBe(createData.receiverProvince);
  132. expect(data.receiverCity).toBe(createData.receiverCity);
  133. expect(data.receiverDistrict).toBe(createData.receiverDistrict);
  134. expect(data.isDefault).toBe(createData.isDefault);
  135. }
  136. });
  137. it('应该验证创建配送地址的必填字段', async () => {
  138. const invalidData = {
  139. // 缺少必填字段
  140. userId: testUser.id,
  141. name: '',
  142. phone: '',
  143. address: '',
  144. receiverProvince: 0,
  145. receiverCity: 0,
  146. receiverDistrict: 0
  147. };
  148. const response = await client.index.$post({
  149. json: invalidData
  150. }, {
  151. headers: {
  152. 'Authorization': `Bearer ${adminToken}`
  153. }
  154. });
  155. expect(response.status).toBe(400);
  156. });
  157. it('应该验证地区层级关系', async () => {
  158. // 使用不存在的地区ID
  159. const invalidAreaData = {
  160. userId: testUser.id,
  161. name: '李四',
  162. phone: '13900139000',
  163. address: '测试地址',
  164. receiverProvince: 999999, // 不存在的省份
  165. receiverCity: testCity.id,
  166. receiverDistrict: testDistrict.id,
  167. receiverTown: 1,
  168. state: 1,
  169. isDefault: 0
  170. };
  171. const response = await client.index.$post({
  172. json: invalidAreaData
  173. }, {
  174. headers: {
  175. 'Authorization': `Bearer ${adminToken}`
  176. }
  177. });
  178. console.debug('地区验证响应状态:', response.status);
  179. expect(response.status).toBe(400);
  180. });
  181. });
  182. describe('GET /delivery-address/:id', () => {
  183. it('应该返回指定配送地址的详情', async () => {
  184. // 先创建一个配送地址
  185. const dataSource = await IntegrationTestDatabase.getDataSource();
  186. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  187. const testDeliveryAddress = deliveryAddressRepository.create({
  188. userId: testUser.id,
  189. name: '王五',
  190. phone: '13600136000',
  191. address: '海淀区中关村大街1号',
  192. receiverProvince: testProvince.id,
  193. receiverCity: testCity.id,
  194. receiverDistrict: testDistrict.id,
  195. receiverTown: 1,
  196. state: 1,
  197. isDefault: 0,
  198. createdBy: testUser.id
  199. });
  200. await deliveryAddressRepository.save(testDeliveryAddress);
  201. const response = await client[':id'].$get({
  202. param: { id: testDeliveryAddress.id }
  203. }, {
  204. headers: {
  205. 'Authorization': `Bearer ${adminToken}`
  206. }
  207. });
  208. console.debug('配送地址详情响应状态:', response.status);
  209. expect(response.status).toBe(200);
  210. if (response.status === 200) {
  211. const data = await response.json();
  212. expect(data.id).toBe(testDeliveryAddress.id);
  213. expect(data.name).toBe(testDeliveryAddress.name);
  214. expect(data.phone).toBe(testDeliveryAddress.phone);
  215. expect(data.address).toBe(testDeliveryAddress.address);
  216. // 验证地区数据关联
  217. if (data.province) {
  218. expect(data.province.id).toBe(testProvince.id);
  219. }
  220. if (data.city) {
  221. expect(data.city.id).toBe(testCity.id);
  222. }
  223. if (data.district) {
  224. expect(data.district.id).toBe(testDistrict.id);
  225. }
  226. }
  227. });
  228. it('应该处理不存在的配送地址', async () => {
  229. const response = await client[':id'].$get({
  230. param: { id: 999999 }
  231. }, {
  232. headers: {
  233. 'Authorization': `Bearer ${adminToken}`
  234. }
  235. });
  236. expect(response.status).toBe(404);
  237. });
  238. });
  239. describe('PUT /delivery-address/:id', () => {
  240. it('应该成功更新配送地址', async () => {
  241. // 先创建一个配送地址
  242. const dataSource = await IntegrationTestDatabase.getDataSource();
  243. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  244. const testDeliveryAddress = deliveryAddressRepository.create({
  245. userId: testUser.id,
  246. name: '原始姓名',
  247. phone: '13500135000',
  248. address: '原始地址',
  249. receiverProvince: testProvince.id,
  250. receiverCity: testCity.id,
  251. receiverDistrict: testDistrict.id,
  252. receiverTown: 1,
  253. state: 1,
  254. isDefault: 0,
  255. createdBy: testUser.id
  256. });
  257. await deliveryAddressRepository.save(testDeliveryAddress);
  258. const updateData = {
  259. name: '更新后的姓名',
  260. phone: '13700137000',
  261. address: '更新后的地址',
  262. isDefault: 1
  263. };
  264. const response = await client[':id'].$put({
  265. param: { id: testDeliveryAddress.id },
  266. json: updateData
  267. }, {
  268. headers: {
  269. 'Authorization': `Bearer ${adminToken}`
  270. }
  271. });
  272. console.debug('更新配送地址响应状态:', response.status);
  273. expect(response.status).toBe(200);
  274. if (response.status === 200) {
  275. const data = await response.json();
  276. expect(data.name).toBe(updateData.name);
  277. expect(data.phone).toBe(updateData.phone);
  278. expect(data.address).toBe(updateData.address);
  279. expect(data.isDefault).toBe(updateData.isDefault);
  280. }
  281. });
  282. });
  283. describe('DELETE /delivery-address/:id', () => {
  284. it('应该成功删除配送地址', async () => {
  285. // 先创建一个配送地址
  286. const dataSource = await IntegrationTestDatabase.getDataSource();
  287. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  288. const testDeliveryAddress = deliveryAddressRepository.create({
  289. userId: testUser.id,
  290. name: '待删除地址',
  291. phone: '13400134000',
  292. address: '待删除地址',
  293. receiverProvince: testProvince.id,
  294. receiverCity: testCity.id,
  295. receiverDistrict: testDistrict.id,
  296. receiverTown: 1,
  297. state: 1,
  298. isDefault: 0,
  299. createdBy: testUser.id
  300. });
  301. await deliveryAddressRepository.save(testDeliveryAddress);
  302. const response = await client[':id'].$delete({
  303. param: { id: testDeliveryAddress.id }
  304. }, {
  305. headers: {
  306. 'Authorization': `Bearer ${adminToken}`
  307. }
  308. });
  309. console.debug('删除配送地址响应状态:', response.status);
  310. expect(response.status).toBe(204);
  311. // 验证配送地址确实被删除
  312. const deletedDeliveryAddress = await deliveryAddressRepository.findOne({
  313. where: { id: testDeliveryAddress.id }
  314. });
  315. expect(deletedDeliveryAddress).toBeNull();
  316. });
  317. });
  318. describe('省市区关联测试', () => {
  319. it('应该正确关联省市区数据', async () => {
  320. // 创建配送地址
  321. const dataSource = await IntegrationTestDatabase.getDataSource();
  322. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  323. const testDeliveryAddress = deliveryAddressRepository.create({
  324. userId: testUser.id,
  325. name: '关联测试',
  326. phone: '13300133000',
  327. address: '关联测试地址',
  328. receiverProvince: testProvince.id,
  329. receiverCity: testCity.id,
  330. receiverDistrict: testDistrict.id,
  331. receiverTown: 1,
  332. state: 1,
  333. isDefault: 0,
  334. createdBy: testUser.id
  335. });
  336. await deliveryAddressRepository.save(testDeliveryAddress);
  337. // 查询配送地址详情,验证地区关联
  338. const response = await client[':id'].$get({
  339. param: { id: testDeliveryAddress.id }
  340. }, {
  341. headers: {
  342. 'Authorization': `Bearer ${adminToken}`
  343. }
  344. });
  345. expect(response.status).toBe(200);
  346. const data = await response.json();
  347. // 验证省市区关联数据
  348. if ('province' in data && data.province) {
  349. expect(data.province.name).toBe('北京市');
  350. expect(data.province.level).toBe(AreaLevel.PROVINCE);
  351. }
  352. if ('city' in data && data.city) {
  353. expect(data.city.name).toBe('北京市');
  354. expect(data.city.level).toBe(AreaLevel.CITY);
  355. }
  356. if ('district' in data && data.district) {
  357. expect(data.district.name).toBe('朝阳区');
  358. expect(data.district.level).toBe(AreaLevel.DISTRICT);
  359. }
  360. });
  361. it('应该验证地区层级关系', async () => {
  362. // 创建另一个区级地区,但父级不是市级
  363. const dataSource = await IntegrationTestDatabase.getDataSource();
  364. const areaRepository = dataSource.getRepository(AreaEntity);
  365. const invalidDistrict = areaRepository.create({
  366. name: '无效区',
  367. code: '999999',
  368. level: AreaLevel.DISTRICT,
  369. parentId: testProvince.id // 父级是省,不是市
  370. });
  371. await areaRepository.save(invalidDistrict);
  372. // 尝试使用无效的地区层级关系创建地址
  373. const createData = {
  374. userId: testUser.id,
  375. name: '层级测试',
  376. phone: '13200132000',
  377. address: '层级测试地址',
  378. receiverProvince: testProvince.id,
  379. receiverCity: testCity.id,
  380. receiverDistrict: invalidDistrict.id, // 这个区的父级不是testCity
  381. receiverTown: 1,
  382. state: 1,
  383. isDefault: 0
  384. };
  385. const response = await client.index.$post({
  386. json: createData
  387. }, {
  388. headers: {
  389. 'Authorization': `Bearer ${adminToken}`
  390. }
  391. });
  392. console.debug('地区层级验证响应状态:', response.status);
  393. // 这里期望返回400,因为地区层级关系不匹配
  394. expect(response.status).toBe(400);
  395. });
  396. });
  397. describe('管理员权限测试', () => {
  398. it('管理员应该可以为其他用户创建地址', async () => {
  399. const createData = {
  400. userId: testUser.id, // 为其他用户创建地址
  401. name: '其他用户地址',
  402. phone: '13800138001',
  403. address: '其他用户地址',
  404. receiverProvince: testProvince.id,
  405. receiverCity: testCity.id,
  406. receiverDistrict: testDistrict.id,
  407. receiverTown: 1,
  408. state: 1,
  409. isDefault: 0
  410. };
  411. const response = await client.index.$post({
  412. json: createData
  413. }, {
  414. headers: {
  415. 'Authorization': `Bearer ${adminToken}`
  416. }
  417. });
  418. console.debug('管理员为其他用户创建地址响应状态:', response.status);
  419. expect(response.status).toBe(201);
  420. if (response.status === 201) {
  421. const data = await response.json();
  422. expect(data.userId).toBe(testUser.id); // 验证地址确实属于其他用户
  423. expect(data.name).toBe(createData.name);
  424. }
  425. });
  426. it('管理员应该可以访问所有用户的地址', async () => {
  427. // 为测试用户创建一些地址
  428. const dataSource = await IntegrationTestDatabase.getDataSource();
  429. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  430. const userAddress1 = deliveryAddressRepository.create({
  431. userId: testUser.id,
  432. name: '用户地址1',
  433. phone: '13800138002',
  434. address: '用户地址1',
  435. receiverProvince: testProvince.id,
  436. receiverCity: testCity.id,
  437. receiverDistrict: testDistrict.id,
  438. receiverTown: 1,
  439. state: 1,
  440. isDefault: 0,
  441. createdBy: testUser.id
  442. });
  443. await deliveryAddressRepository.save(userAddress1);
  444. const userAddress2 = deliveryAddressRepository.create({
  445. userId: testUser.id,
  446. name: '用户地址2',
  447. phone: '13800138003',
  448. address: '用户地址2',
  449. receiverProvince: testProvince.id,
  450. receiverCity: testCity.id,
  451. receiverDistrict: testDistrict.id,
  452. receiverTown: 1,
  453. state: 1,
  454. isDefault: 0,
  455. createdBy: testUser.id
  456. });
  457. await deliveryAddressRepository.save(userAddress2);
  458. // 管理员应该能看到所有地址
  459. const response = await client.index.$get({
  460. query: {}
  461. }, {
  462. headers: {
  463. 'Authorization': `Bearer ${adminToken}`
  464. }
  465. });
  466. expect(response.status).toBe(200);
  467. const data = await response.json();
  468. if (data && 'data' in data) {
  469. expect(Array.isArray(data.data)).toBe(true);
  470. expect(data.data.length).toBeGreaterThanOrEqual(2); // 至少包含我们创建的两个地址
  471. }
  472. });
  473. it('管理员应该可以更新其他用户的地址', async () => {
  474. // 先为测试用户创建一个地址
  475. const dataSource = await IntegrationTestDatabase.getDataSource();
  476. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  477. const testDeliveryAddress = deliveryAddressRepository.create({
  478. userId: testUser.id,
  479. name: '原始地址',
  480. phone: '13800138004',
  481. address: '原始地址',
  482. receiverProvince: testProvince.id,
  483. receiverCity: testCity.id,
  484. receiverDistrict: testDistrict.id,
  485. receiverTown: 1,
  486. state: 1,
  487. isDefault: 0,
  488. createdBy: testUser.id
  489. });
  490. await deliveryAddressRepository.save(testDeliveryAddress);
  491. const updateData = {
  492. name: '管理员更新的地址',
  493. phone: '13900139000',
  494. address: '管理员更新的地址'
  495. };
  496. const response = await client[':id'].$put({
  497. param: { id: testDeliveryAddress.id },
  498. json: updateData
  499. }, {
  500. headers: {
  501. 'Authorization': `Bearer ${adminToken}`
  502. }
  503. });
  504. console.debug('管理员更新其他用户地址响应状态:', response.status);
  505. expect(response.status).toBe(200);
  506. if (response.status === 200) {
  507. const data = await response.json();
  508. expect(data.name).toBe(updateData.name);
  509. expect(data.phone).toBe(updateData.phone);
  510. expect(data.address).toBe(updateData.address);
  511. }
  512. });
  513. it('管理员应该可以删除其他用户的地址', async () => {
  514. // 先为测试用户创建一个地址
  515. const dataSource = await IntegrationTestDatabase.getDataSource();
  516. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  517. const testDeliveryAddress = deliveryAddressRepository.create({
  518. userId: testUser.id,
  519. name: '待删除地址',
  520. phone: '13800138005',
  521. address: '待删除地址',
  522. receiverProvince: testProvince.id,
  523. receiverCity: testCity.id,
  524. receiverDistrict: testDistrict.id,
  525. receiverTown: 1,
  526. state: 1,
  527. isDefault: 0,
  528. createdBy: testUser.id
  529. });
  530. await deliveryAddressRepository.save(testDeliveryAddress);
  531. const response = await client[':id'].$delete({
  532. param: { id: testDeliveryAddress.id }
  533. }, {
  534. headers: {
  535. 'Authorization': `Bearer ${adminToken}`
  536. }
  537. });
  538. console.debug('管理员删除其他用户地址响应状态:', response.status);
  539. expect(response.status).toBe(204);
  540. // 验证地址确实被删除
  541. const deletedDeliveryAddress = await deliveryAddressRepository.findOne({
  542. where: { id: testDeliveryAddress.id }
  543. });
  544. expect(deletedDeliveryAddress).toBeNull();
  545. });
  546. it('管理员应该可以查询指定用户的地址', async () => {
  547. // 为测试用户创建一些地址
  548. const dataSource = await IntegrationTestDatabase.getDataSource();
  549. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  550. const userAddress = deliveryAddressRepository.create({
  551. userId: testUser.id,
  552. name: '指定用户地址',
  553. phone: '13800138006',
  554. address: '指定用户地址',
  555. receiverProvince: testProvince.id,
  556. receiverCity: testCity.id,
  557. receiverDistrict: testDistrict.id,
  558. receiverTown: 1,
  559. state: 1,
  560. isDefault: 0,
  561. createdBy: testUser.id
  562. });
  563. await deliveryAddressRepository.save(userAddress);
  564. // 管理员可以查询指定用户的地址
  565. const response = await client.index.$get({
  566. query: { filters: JSON.stringify({ userId: testUser.id }) }
  567. }, {
  568. headers: {
  569. 'Authorization': `Bearer ${adminToken}`
  570. }
  571. });
  572. expect(response.status).toBe(200);
  573. const data = await response.json();
  574. if (data && 'data' in data) {
  575. expect(Array.isArray(data.data)).toBe(true);
  576. // 验证返回的地址都属于指定用户
  577. if (data.data.length > 0) {
  578. data.data.forEach((address: any) => {
  579. expect(address.userId).toBe(testUser.id);
  580. });
  581. }
  582. }
  583. });
  584. });
  585. });