user-orders.integration.test.ts 19 KB

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