user-orders.integration.test.ts 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628
  1. import { describe, it, expect, beforeEach } 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 { DeliveryAddress } from '@d8d/delivery-address-module';
  7. import { AreaEntity } from '@d8d/geo-areas';
  8. import { Merchant } from '@d8d/merchant-module';
  9. import { Supplier } from '@d8d/supplier-module';
  10. import { File } from '@d8d/file-module';
  11. import userOrderRoutes from '../../src/routes/user/orders';
  12. import { Order } from '../../src/entities';
  13. // 设置集成测试钩子
  14. setupIntegrationDatabaseHooksWithEntities([
  15. UserEntity, Role, Order, DeliveryAddress, Merchant, Supplier, File, AreaEntity
  16. ])
  17. describe('用户订单管理API集成测试', () => {
  18. let client: ReturnType<typeof testClient<typeof userOrderRoutes>>;
  19. let userToken: string;
  20. let otherUserToken: string;
  21. let testUser: UserEntity;
  22. let otherUser: UserEntity;
  23. let testDeliveryAddress: DeliveryAddress;
  24. let testMerchant: Merchant;
  25. let testSupplier: Supplier;
  26. beforeEach(async () => {
  27. // 创建测试客户端
  28. client = testClient(userOrderRoutes);
  29. // 获取数据源
  30. const dataSource = await IntegrationTestDatabase.getDataSource();
  31. // 创建测试用户
  32. const userRepository = dataSource.getRepository(UserEntity);
  33. testUser = userRepository.create({
  34. username: `test_user_${Math.floor(Math.random() * 100000)}`,
  35. password: 'test_password',
  36. nickname: '测试用户',
  37. registrationSource: 'web'
  38. });
  39. await userRepository.save(testUser);
  40. // 创建其他用户
  41. otherUser = userRepository.create({
  42. username: `other_user_${Math.floor(Math.random() * 100000)}`,
  43. password: 'other_password',
  44. nickname: '其他用户',
  45. registrationSource: 'web'
  46. });
  47. await userRepository.save(otherUser);
  48. // 生成测试用户的token
  49. userToken = JWTUtil.generateToken({
  50. id: testUser.id,
  51. username: testUser.username,
  52. roles: [{name:'user'}]
  53. });
  54. // 生成其他用户的token
  55. otherUserToken = JWTUtil.generateToken({
  56. id: otherUser.id,
  57. username: otherUser.username,
  58. roles: [{name:'user'}]
  59. });
  60. // 创建测试区域数据
  61. const areaRepository = dataSource.getRepository(AreaEntity);
  62. const testProvince = areaRepository.create({
  63. name: '广东省',
  64. code: '440000',
  65. level: 1,
  66. parentId: null
  67. });
  68. await areaRepository.save(testProvince);
  69. const testCity = areaRepository.create({
  70. name: '深圳市',
  71. code: '440300',
  72. level: 2,
  73. parentId: testProvince.id
  74. });
  75. await areaRepository.save(testCity);
  76. const testDistrict = areaRepository.create({
  77. name: '南山区',
  78. code: '440305',
  79. level: 3,
  80. parentId: testCity.id
  81. });
  82. await areaRepository.save(testDistrict);
  83. const testTown = areaRepository.create({
  84. name: '粤海街道',
  85. code: '440305001',
  86. level: 3,
  87. parentId: testDistrict.id
  88. });
  89. await areaRepository.save(testTown);
  90. // 创建测试配送地址
  91. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  92. testDeliveryAddress = deliveryAddressRepository.create({
  93. userId: testUser.id,
  94. name: '收货人姓名',
  95. phone: '13800138000',
  96. address: '测试地址',
  97. receiverProvince: testProvince.id,
  98. receiverCity: testCity.id,
  99. receiverDistrict: testDistrict.id,
  100. receiverTown: testTown.id,
  101. isDefault: 1,
  102. state: 1,
  103. createdBy: testUser.id
  104. });
  105. await deliveryAddressRepository.save(testDeliveryAddress);
  106. // 创建测试商户
  107. const merchantRepository = dataSource.getRepository(Merchant);
  108. testMerchant = merchantRepository.create({
  109. name: '测试商户',
  110. username: `test_merchant_${Math.floor(Math.random() * 100000)}`,
  111. password: 'password123',
  112. phone: '13800138001',
  113. realname: '测试商户',
  114. state: 1,
  115. createdBy: testUser.id
  116. });
  117. await merchantRepository.save(testMerchant);
  118. // 创建测试供应商
  119. const supplierRepository = dataSource.getRepository(Supplier);
  120. testSupplier = supplierRepository.create({
  121. name: '测试供应商',
  122. username: `test_supplier_${Math.floor(Math.random() * 100000)}`,
  123. password: 'password123',
  124. phone: '13800138002',
  125. realname: '测试供应商',
  126. state: 1,
  127. createdBy: testUser.id
  128. });
  129. await supplierRepository.save(testSupplier);
  130. });
  131. describe('GET /orders', () => {
  132. it('应该返回当前用户的订单列表', async () => {
  133. // 为测试用户创建一些订单
  134. const dataSource = await IntegrationTestDatabase.getDataSource();
  135. const orderRepository = dataSource.getRepository(Order);
  136. const userOrder1 = orderRepository.create({
  137. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  138. userId: testUser.id,
  139. amount: 100.00,
  140. costAmount: 80.00,
  141. payAmount: 100.00,
  142. orderType: 1,
  143. payType: 1,
  144. payState: 2,
  145. state: 0,
  146. addressId: testDeliveryAddress.id,
  147. merchantId: testMerchant.id,
  148. supplierId: testSupplier.id,
  149. createdBy: testUser.id
  150. });
  151. await orderRepository.save(userOrder1);
  152. const userOrder2 = orderRepository.create({
  153. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  154. userId: testUser.id,
  155. amount: 200.00,
  156. costAmount: 160.00,
  157. payAmount: 200.00,
  158. orderType: 1,
  159. payType: 1,
  160. payState: 2,
  161. state: 1,
  162. addressId: testDeliveryAddress.id,
  163. merchantId: testMerchant.id,
  164. supplierId: testSupplier.id,
  165. createdBy: testUser.id
  166. });
  167. await orderRepository.save(userOrder2);
  168. // 为其他用户创建一个订单,确保不会返回
  169. const otherUserOrder = orderRepository.create({
  170. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  171. userId: otherUser.id,
  172. amount: 300.00,
  173. costAmount: 240.00,
  174. payAmount: 300.00,
  175. orderType: 1,
  176. payType: 1,
  177. payState: 2,
  178. state: 0,
  179. addressId: testDeliveryAddress.id,
  180. merchantId: testMerchant.id,
  181. supplierId: testSupplier.id,
  182. createdBy: otherUser.id
  183. });
  184. await orderRepository.save(otherUserOrder);
  185. const response = await client.index.$get({
  186. query: {
  187. page: 1,
  188. pageSize: 10
  189. }
  190. }, {
  191. headers: {
  192. 'Authorization': `Bearer ${userToken}`
  193. }
  194. });
  195. console.debug('用户订单列表响应状态:', response.status);
  196. if (response.status !== 200) {
  197. const errorData = await response.json();
  198. console.debug('用户订单列表错误响应:', errorData);
  199. }
  200. expect(response.status).toBe(200);
  201. if (response.status === 200) {
  202. const data = await response.json();
  203. expect(data).toHaveProperty('data');
  204. expect(Array.isArray(data.data)).toBe(true);
  205. // 验证只返回当前用户的订单
  206. data.data.forEach((order: any) => {
  207. expect(order.userId).toBe(testUser.id);
  208. });
  209. // 验证不包含其他用户的订单
  210. const otherUserOrderInResponse = data.data.find((order: any) => order.userId === otherUser.id);
  211. expect(otherUserOrderInResponse).toBeUndefined();
  212. }
  213. });
  214. it('应该拒绝未认证用户的访问', async () => {
  215. const response = await client.index.$get({
  216. query: {
  217. page: 1,
  218. pageSize: 10
  219. }
  220. });
  221. expect(response.status).toBe(401);
  222. });
  223. });
  224. describe('POST /orders', () => {
  225. it('应该成功创建订单并自动设置当前用户权限', async () => {
  226. const createData = {
  227. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  228. amount: 150.00,
  229. costAmount: 120.00,
  230. payAmount: 150.00,
  231. orderType: 1,
  232. payType: 1,
  233. payState: 0,
  234. state: 0,
  235. addressId: testDeliveryAddress.id,
  236. merchantId: testMerchant.id,
  237. supplierId: testSupplier.id
  238. };
  239. const response = await client.index.$post({
  240. json: createData
  241. }, {
  242. headers: {
  243. 'Authorization': `Bearer ${userToken}`
  244. }
  245. });
  246. console.debug('用户创建订单响应状态:', response.status);
  247. if (response.status !== 201) {
  248. const errorData = await response.json();
  249. console.debug('用户创建订单错误响应:', errorData);
  250. }
  251. expect(response.status).toBe(201);
  252. if (response.status === 201) {
  253. const data = await response.json();
  254. expect(data).toHaveProperty('id');
  255. expect(data.orderNo).toBe(createData.orderNo);
  256. expect(parseFloat(data.amount)).toBe(createData.amount);
  257. expect(data.userId).toBe(testUser.id); // 验证自动设置当前用户权限
  258. expect(data.createdBy).toBe(testUser.id); // 验证自动设置创建用户
  259. }
  260. });
  261. it('应该验证创建订单的必填字段', async () => {
  262. const invalidData = {
  263. // 缺少必填字段
  264. amount: -1,
  265. orderType: -1
  266. };
  267. const response = await client.index.$post({
  268. json: invalidData
  269. }, {
  270. headers: {
  271. 'Authorization': `Bearer ${userToken}`
  272. }
  273. });
  274. expect(response.status).toBe(400);
  275. });
  276. });
  277. describe('GET /orders/:id', () => {
  278. it('应该返回当前用户的订单详情', async () => {
  279. // 先为测试用户创建一个订单
  280. const dataSource = await IntegrationTestDatabase.getDataSource();
  281. const orderRepository = dataSource.getRepository(Order);
  282. const testOrder = orderRepository.create({
  283. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  284. userId: testUser.id,
  285. amount: 100.00,
  286. costAmount: 80.00,
  287. payAmount: 100.00,
  288. orderType: 1,
  289. payType: 1,
  290. payState: 2,
  291. state: 0,
  292. addressId: testDeliveryAddress.id,
  293. merchantId: testMerchant.id,
  294. supplierId: testSupplier.id,
  295. createdBy: testUser.id
  296. });
  297. await orderRepository.save(testOrder);
  298. const response = await client[':id'].$get({
  299. param: { id: testOrder.id }
  300. }, {
  301. headers: {
  302. 'Authorization': `Bearer ${userToken}`
  303. }
  304. });
  305. console.debug('用户订单详情响应状态:', response.status);
  306. if (response.status !== 200) {
  307. const errorData = await response.json();
  308. console.debug('用户订单详情错误响应:', errorData);
  309. }
  310. expect(response.status).toBe(200);
  311. if (response.status === 200) {
  312. const data = await response.json();
  313. expect(data.id).toBe(testOrder.id);
  314. expect(data.orderNo).toBe(testOrder.orderNo);
  315. expect(data.userId).toBe(testUser.id);
  316. }
  317. });
  318. it('应该拒绝访问其他用户的订单', async () => {
  319. // 为其他用户创建一个订单
  320. const dataSource = await IntegrationTestDatabase.getDataSource();
  321. const orderRepository = dataSource.getRepository(Order);
  322. const otherUserOrder = orderRepository.create({
  323. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  324. userId: otherUser.id,
  325. amount: 100.00,
  326. costAmount: 80.00,
  327. payAmount: 100.00,
  328. orderType: 1,
  329. payType: 1,
  330. payState: 2,
  331. state: 0,
  332. addressId: testDeliveryAddress.id,
  333. merchantId: testMerchant.id,
  334. supplierId: testSupplier.id,
  335. createdBy: otherUser.id
  336. });
  337. await orderRepository.save(otherUserOrder);
  338. const response = await client[':id'].$get({
  339. param: { id: otherUserOrder.id }
  340. }, {
  341. headers: {
  342. 'Authorization': `Bearer ${userToken}`
  343. }
  344. });
  345. expect(response.status).toBe(403); // 数据权限控制返回403(权限不足)
  346. });
  347. it('应该处理不存在的订单', async () => {
  348. const response = await client[':id'].$get({
  349. param: { id: 999999 }
  350. }, {
  351. headers: {
  352. 'Authorization': `Bearer ${userToken}`
  353. }
  354. });
  355. expect(response.status).toBe(404);
  356. });
  357. });
  358. describe('PUT /orders/:id', () => {
  359. it('应该成功更新当前用户的订单', async () => {
  360. // 先为测试用户创建一个订单
  361. const dataSource = await IntegrationTestDatabase.getDataSource();
  362. const orderRepository = dataSource.getRepository(Order);
  363. const testOrder = orderRepository.create({
  364. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  365. userId: testUser.id,
  366. amount: 100.00,
  367. costAmount: 80.00,
  368. payAmount: 100.00,
  369. orderType: 1,
  370. payType: 1,
  371. payState: 2,
  372. state: 0,
  373. addressId: testDeliveryAddress.id,
  374. merchantId: testMerchant.id,
  375. supplierId: testSupplier.id,
  376. createdBy: testUser.id
  377. });
  378. await orderRepository.save(testOrder);
  379. const updateData = {
  380. state: 1,
  381. payState: 1
  382. };
  383. const response = await client[':id'].$put({
  384. param: { id: testOrder.id },
  385. json: updateData
  386. }, {
  387. headers: {
  388. 'Authorization': `Bearer ${userToken}`
  389. }
  390. });
  391. console.debug('用户更新订单响应状态:', response.status);
  392. expect(response.status).toBe(200);
  393. if (response.status === 200) {
  394. const data = await response.json();
  395. expect(data.state).toBe(updateData.state);
  396. expect(data.payState).toBe(updateData.payState);
  397. expect(data.updatedBy).toBe(testUser.id); // 验证自动设置更新用户
  398. }
  399. });
  400. it('应该拒绝更新其他用户的订单', async () => {
  401. // 为其他用户创建一个订单
  402. const dataSource = await IntegrationTestDatabase.getDataSource();
  403. const orderRepository = dataSource.getRepository(Order);
  404. const otherUserOrder = orderRepository.create({
  405. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  406. userId: otherUser.id,
  407. amount: 100.00,
  408. costAmount: 80.00,
  409. payAmount: 100.00,
  410. orderType: 1,
  411. payType: 1,
  412. payState: 2,
  413. state: 0,
  414. addressId: testDeliveryAddress.id,
  415. merchantId: testMerchant.id,
  416. supplierId: testSupplier.id,
  417. createdBy: otherUser.id
  418. });
  419. await orderRepository.save(otherUserOrder);
  420. const updateData = {
  421. state: 1
  422. };
  423. const response = await client[':id'].$put({
  424. param: { id: otherUserOrder.id },
  425. json: updateData
  426. }, {
  427. headers: {
  428. 'Authorization': `Bearer ${userToken}`
  429. }
  430. });
  431. expect(response.status).toBe(403); // 数据权限控制返回403
  432. });
  433. });
  434. describe('DELETE /orders/:id', () => {
  435. it('应该成功删除当前用户的订单', async () => {
  436. // 先为测试用户创建一个订单
  437. const dataSource = await IntegrationTestDatabase.getDataSource();
  438. const orderRepository = dataSource.getRepository(Order);
  439. const testOrder = orderRepository.create({
  440. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  441. userId: testUser.id,
  442. amount: 100.00,
  443. costAmount: 80.00,
  444. payAmount: 100.00,
  445. orderType: 1,
  446. payType: 1,
  447. payState: 2,
  448. state: 0,
  449. addressId: testDeliveryAddress.id,
  450. merchantId: testMerchant.id,
  451. supplierId: testSupplier.id,
  452. createdBy: testUser.id
  453. });
  454. await orderRepository.save(testOrder);
  455. const response = await client[':id'].$delete({
  456. param: { id: testOrder.id }
  457. }, {
  458. headers: {
  459. 'Authorization': `Bearer ${userToken}`
  460. }
  461. });
  462. console.debug('用户删除订单响应状态:', response.status);
  463. expect(response.status).toBe(204);
  464. });
  465. it('应该拒绝删除其他用户的订单', async () => {
  466. // 为其他用户创建一个订单
  467. const dataSource = await IntegrationTestDatabase.getDataSource();
  468. const orderRepository = dataSource.getRepository(Order);
  469. const otherUserOrder = orderRepository.create({
  470. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  471. userId: otherUser.id,
  472. amount: 100.00,
  473. costAmount: 80.00,
  474. payAmount: 100.00,
  475. orderType: 1,
  476. payType: 1,
  477. payState: 2,
  478. state: 0,
  479. addressId: testDeliveryAddress.id,
  480. merchantId: testMerchant.id,
  481. supplierId: testSupplier.id,
  482. createdBy: otherUser.id
  483. });
  484. await orderRepository.save(otherUserOrder);
  485. const response = await client[':id'].$delete({
  486. param: { id: otherUserOrder.id }
  487. }, {
  488. headers: {
  489. 'Authorization': `Bearer ${userToken}`
  490. }
  491. });
  492. expect(response.status).toBe(403); // 数据权限控制返回403
  493. });
  494. });
  495. describe('数据权限配置测试', () => {
  496. it('应该验证dataPermission配置正确工作', async () => {
  497. // 这个测试验证数据权限配置是否正常工作
  498. // 用户只能访问自己创建的订单
  499. const dataSource = await IntegrationTestDatabase.getDataSource();
  500. const orderRepository = dataSource.getRepository(Order);
  501. // 创建测试用户和其他用户的订单
  502. const userOrder = orderRepository.create({
  503. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  504. userId: testUser.id,
  505. amount: 100.00,
  506. costAmount: 80.00,
  507. payAmount: 100.00,
  508. orderType: 1,
  509. payType: 1,
  510. payState: 2,
  511. state: 0,
  512. addressId: testDeliveryAddress.id,
  513. merchantId: testMerchant.id,
  514. supplierId: testSupplier.id,
  515. createdBy: testUser.id
  516. });
  517. await orderRepository.save(userOrder);
  518. const otherUserOrder = orderRepository.create({
  519. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  520. userId: otherUser.id,
  521. amount: 200.00,
  522. costAmount: 160.00,
  523. payAmount: 200.00,
  524. orderType: 1,
  525. payType: 1,
  526. payState: 2,
  527. state: 0,
  528. addressId: testDeliveryAddress.id,
  529. merchantId: testMerchant.id,
  530. supplierId: testSupplier.id,
  531. createdBy: otherUser.id
  532. });
  533. await orderRepository.save(otherUserOrder);
  534. // 使用测试用户token获取列表
  535. const response = await client.index.$get({
  536. query: {
  537. page: 1,
  538. pageSize: 10
  539. }
  540. }, {
  541. headers: {
  542. 'Authorization': `Bearer ${userToken}`
  543. }
  544. });
  545. if (response.status !== 200) {
  546. const errorData = await response.json();
  547. console.debug('数据权限配置测试错误响应:', errorData);
  548. }
  549. expect(response.status).toBe(200);
  550. const data = await response.json();
  551. // 类型检查确保data属性存在
  552. if ('data' in data && Array.isArray(data.data)) {
  553. // 验证只返回测试用户的订单
  554. const userOrderInResponse = data.data.filter((order: any) => order.userId === testUser.id);
  555. const otherUserOrderInResponse = data.data.filter((order: any) => order.userId === otherUser.id);
  556. expect(userOrderInResponse.length).toBeGreaterThan(0);
  557. expect(otherUserOrderInResponse.length).toBe(0);
  558. } else {
  559. // 如果响应是错误格式,应该失败
  560. expect(data).toHaveProperty('data');
  561. }
  562. });
  563. });
  564. });