create-order.integration.test.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  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 { Goods, GoodsCategory } from '@d8d/goods-module';
  9. import { Supplier } from '@d8d/supplier-module';
  10. import { File } from '@d8d/file-module';
  11. import { Merchant } from '@d8d/merchant-module';
  12. import { createOrderRoutes } from '../../src/routes/create-order';
  13. import { Order, OrderGoods } from '../../src/entities';
  14. // 设置集成测试钩子
  15. setupIntegrationDatabaseHooksWithEntities([
  16. UserEntity, Role, Order, OrderGoods, Goods, GoodsCategory, File, Supplier, DeliveryAddress, Merchant, AreaEntity
  17. ])
  18. describe('订单创建API集成测试', () => {
  19. let client: ReturnType<typeof testClient<typeof createOrderRoutes>>;
  20. let userToken: string;
  21. let testUser: UserEntity;
  22. let testGoods: Goods;
  23. let testSupplier: Supplier;
  24. let testFile: File;
  25. let testDeliveryAddress: DeliveryAddress;
  26. beforeEach(async () => {
  27. // 创建测试客户端
  28. client = testClient(createOrderRoutes);
  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. // 生成测试用户的token
  41. userToken = JWTUtil.generateToken({
  42. id: testUser.id,
  43. username: testUser.username,
  44. roles: [{name:'user'}]
  45. });
  46. // 创建测试商品
  47. const goodsRepository = dataSource.getRepository(Goods);
  48. testGoods = goodsRepository.create({
  49. name: '测试商品',
  50. price: 100.00,
  51. costPrice: 80.00,
  52. categoryId1: 1,
  53. categoryId2: 1,
  54. categoryId3: 1,
  55. goodsType: 1,
  56. supplierId: 1,
  57. state: 1,
  58. stock: 100,
  59. lowestBuy: 1,
  60. createdBy: testUser.id
  61. });
  62. await goodsRepository.save(testGoods);
  63. // 创建测试供应商
  64. const supplierRepository = dataSource.getRepository(Supplier);
  65. testSupplier = supplierRepository.create({
  66. name: '测试供应商',
  67. username: `test_supplier_${Math.floor(Math.random() * 100000)}`,
  68. password: 'password123',
  69. phone: '13800138000',
  70. realname: '测试供应商',
  71. state: 1,
  72. createdBy: testUser.id
  73. });
  74. await supplierRepository.save(testSupplier);
  75. // 创建测试文件
  76. const fileRepository = dataSource.getRepository(File);
  77. testFile = fileRepository.create({
  78. name: 'test_image.jpg',
  79. type: 'image/jpeg',
  80. size: 102400,
  81. path: 'images/test_image.jpg',
  82. uploadUserId: testUser.id,
  83. uploadTime: new Date(),
  84. createdAt: new Date(),
  85. updatedAt: new Date()
  86. });
  87. await fileRepository.save(testFile);
  88. // 创建测试配送地址
  89. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  90. testDeliveryAddress = deliveryAddressRepository.create({
  91. userId: testUser.id,
  92. name: '收货人姓名',
  93. phone: '13800138000',
  94. province: '广东省',
  95. city: '深圳市',
  96. district: '南山区',
  97. address: '测试地址',
  98. isDefault: 1,
  99. state: 1,
  100. createdBy: testUser.id
  101. });
  102. await deliveryAddressRepository.save(testDeliveryAddress);
  103. });
  104. describe('POST /create-order', () => {
  105. it('应该成功创建订单并关联商品', async () => {
  106. const createData = {
  107. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  108. amount: 150.00,
  109. costAmount: 120.00,
  110. payAmount: 150.00,
  111. orderType: 1,
  112. payType: 1,
  113. payState: 0,
  114. state: 0,
  115. deliveryAddressId: testDeliveryAddress.id,
  116. merchantId: 1,
  117. supplierId: testSupplier.id,
  118. orderGoods: [
  119. {
  120. goodsId: testGoods.id,
  121. goodsName: '测试商品1',
  122. price: 50.00,
  123. num: 2,
  124. state: 0,
  125. supplierId: testSupplier.id,
  126. imageFileId: testFile.id
  127. },
  128. {
  129. goodsId: testGoods.id,
  130. goodsName: '测试商品2',
  131. price: 25.00,
  132. num: 2,
  133. state: 0,
  134. supplierId: testSupplier.id,
  135. imageFileId: testFile.id
  136. }
  137. ]
  138. };
  139. const response = await client.index.$post({
  140. json: createData
  141. }, {
  142. headers: {
  143. 'Authorization': `Bearer ${userToken}`
  144. }
  145. });
  146. console.debug('订单创建响应状态:', response.status);
  147. if (response.status !== 201) {
  148. const errorData = await response.json();
  149. console.debug('订单创建错误响应:', errorData);
  150. }
  151. expect(response.status).toBe(201);
  152. if (response.status === 201) {
  153. const data = await response.json();
  154. expect(data).toHaveProperty('id');
  155. expect(data.orderNo).toBe(createData.orderNo);
  156. expect(parseFloat(data.amount)).toBe(createData.amount);
  157. expect(data.userId).toBe(testUser.id); // 验证自动设置当前用户权限
  158. expect(data.createdBy).toBe(testUser.id); // 验证自动设置创建用户
  159. // 验证订单商品被正确创建
  160. expect(data.orderGoods).toBeDefined();
  161. expect(Array.isArray(data.orderGoods)).toBe(true);
  162. expect(data.orderGoods.length).toBe(2);
  163. // 验证商品信息
  164. data.orderGoods.forEach((orderGoods: any, index: number) => {
  165. expect(orderGoods.goodsName).toBe(createData.orderGoods[index].goodsName);
  166. expect(parseFloat(orderGoods.price)).toBe(createData.orderGoods[index].price);
  167. expect(orderGoods.num).toBe(createData.orderGoods[index].num);
  168. expect(orderGoods.createdBy).toBe(testUser.id); // 验证商品创建用户
  169. });
  170. }
  171. });
  172. it('应该验证创建订单的必填字段', async () => {
  173. const invalidData = {
  174. // 缺少必填字段
  175. amount: -1,
  176. orderType: -1
  177. };
  178. const response = await client.index.$post({
  179. json: invalidData
  180. }, {
  181. headers: {
  182. 'Authorization': `Bearer ${userToken}`
  183. }
  184. });
  185. expect(response.status).toBe(400);
  186. });
  187. it('应该处理商品库存不足的情况', async () => {
  188. // 创建一个库存为0的商品
  189. const dataSource = await IntegrationTestDatabase.getDataSource();
  190. const goodsRepository = dataSource.getRepository(Goods);
  191. const outOfStockGoods = goodsRepository.create({
  192. name: '库存不足商品',
  193. price: 100.00,
  194. costPrice: 80.00,
  195. categoryId1: 1,
  196. categoryId2: 1,
  197. categoryId3: 1,
  198. goodsType: 1,
  199. supplierId: 1,
  200. state: 1,
  201. stock: 0, // 库存为0
  202. lowestBuy: 1,
  203. createdBy: testUser.id
  204. });
  205. await goodsRepository.save(outOfStockGoods);
  206. const createData = {
  207. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  208. amount: 100.00,
  209. costAmount: 80.00,
  210. payAmount: 100.00,
  211. orderType: 1,
  212. payType: 1,
  213. payState: 0,
  214. state: 0,
  215. deliveryAddressId: testDeliveryAddress.id,
  216. merchantId: 1,
  217. supplierId: testSupplier.id,
  218. orderGoods: [
  219. {
  220. goodsId: outOfStockGoods.id,
  221. goodsName: '库存不足商品',
  222. price: 100.00,
  223. num: 1,
  224. state: 0,
  225. supplierId: testSupplier.id,
  226. imageFileId: testFile.id
  227. }
  228. ]
  229. };
  230. const response = await client.index.$post({
  231. json: createData
  232. }, {
  233. headers: {
  234. 'Authorization': `Bearer ${userToken}`
  235. }
  236. });
  237. // 应该返回错误状态(库存不足)
  238. expect(response.status).toBe(400);
  239. });
  240. it('应该拒绝未认证用户的访问', async () => {
  241. const createData = {
  242. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  243. amount: 150.00,
  244. costAmount: 120.00,
  245. payAmount: 150.00,
  246. orderType: 1,
  247. payType: 1,
  248. payState: 0,
  249. state: 0,
  250. deliveryAddressId: testDeliveryAddress.id,
  251. merchantId: 1,
  252. supplierId: testSupplier.id,
  253. orderGoods: [
  254. {
  255. goodsId: testGoods.id,
  256. goodsName: '测试商品',
  257. price: 50.00,
  258. num: 3,
  259. state: 0,
  260. supplierId: testSupplier.id,
  261. imageFileId: testFile.id
  262. }
  263. ]
  264. };
  265. const response = await client.index.$post({
  266. json: createData
  267. });
  268. expect(response.status).toBe(401);
  269. });
  270. });
  271. describe('事务性测试', () => {
  272. it('应该确保订单和商品创建的事务一致性', async () => {
  273. const createData = {
  274. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  275. amount: 200.00,
  276. costAmount: 160.00,
  277. payAmount: 200.00,
  278. orderType: 1,
  279. payType: 1,
  280. payState: 0,
  281. state: 0,
  282. deliveryAddressId: testDeliveryAddress.id,
  283. merchantId: 1,
  284. supplierId: testSupplier.id,
  285. orderGoods: [
  286. {
  287. goodsId: testGoods.id,
  288. goodsName: '测试商品1',
  289. price: 100.00,
  290. num: 2,
  291. state: 0,
  292. supplierId: testSupplier.id,
  293. imageFileId: testFile.id
  294. }
  295. ]
  296. };
  297. const response = await client.index.$post({
  298. json: createData
  299. }, {
  300. headers: {
  301. 'Authorization': `Bearer ${userToken}`
  302. }
  303. });
  304. expect(response.status).toBe(201);
  305. if (response.status === 201) {
  306. const data = await response.json();
  307. // 验证订单和商品都被正确创建
  308. const dataSource = await IntegrationTestDatabase.getDataSource();
  309. const orderRepository = dataSource.getRepository(Order);
  310. const orderGoodsRepository = dataSource.getRepository(OrderGoods);
  311. // 验证订单存在
  312. const createdOrder = await orderRepository.findOne({
  313. where: { id: data.id },
  314. relations: ['orderGoods']
  315. });
  316. expect(createdOrder).toBeDefined();
  317. expect(createdOrder?.orderNo).toBe(createData.orderNo);
  318. expect(createdOrder?.userId).toBe(testUser.id);
  319. // 验证订单商品存在
  320. const createdOrderGoods = await orderGoodsRepository.find({
  321. where: { orderId: data.id }
  322. });
  323. expect(createdOrderGoods.length).toBe(1);
  324. expect(createdOrderGoods[0].goodsName).toBe(createData.orderGoods[0].goodsName);
  325. expect(createdOrderGoods[0].num).toBe(createData.orderGoods[0].num);
  326. }
  327. });
  328. it('应该验证库存更新功能', async () => {
  329. const initialStock = testGoods.stock;
  330. const purchaseQuantity = 2;
  331. const createData = {
  332. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  333. amount: 200.00,
  334. costAmount: 160.00,
  335. payAmount: 200.00,
  336. orderType: 1,
  337. payType: 1,
  338. payState: 0,
  339. state: 0,
  340. deliveryAddressId: testDeliveryAddress.id,
  341. merchantId: 1,
  342. supplierId: testSupplier.id,
  343. orderGoods: [
  344. {
  345. goodsId: testGoods.id,
  346. goodsName: '测试商品',
  347. price: 100.00,
  348. num: purchaseQuantity,
  349. state: 0,
  350. supplierId: testSupplier.id,
  351. imageFileId: testFile.id
  352. }
  353. ]
  354. };
  355. const response = await client.index.$post({
  356. json: createData
  357. }, {
  358. headers: {
  359. 'Authorization': `Bearer ${userToken}`
  360. }
  361. });
  362. expect(response.status).toBe(201);
  363. if (response.status === 201) {
  364. // 验证商品库存被正确更新
  365. const dataSource = await IntegrationTestDatabase.getDataSource();
  366. const goodsRepository = dataSource.getRepository(Goods);
  367. const updatedGoods = await goodsRepository.findOne({
  368. where: { id: testGoods.id }
  369. });
  370. expect(updatedGoods).toBeDefined();
  371. expect(updatedGoods?.stock).toBe(initialStock - purchaseQuantity);
  372. }
  373. });
  374. });
  375. describe('数据权限测试', () => {
  376. it('应该自动设置当前用户权限', async () => {
  377. const createData = {
  378. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  379. amount: 100.00,
  380. costAmount: 80.00,
  381. payAmount: 100.00,
  382. orderType: 1,
  383. payType: 1,
  384. payState: 0,
  385. state: 0,
  386. deliveryAddressId: testDeliveryAddress.id,
  387. merchantId: 1,
  388. supplierId: testSupplier.id,
  389. orderGoods: [
  390. {
  391. goodsId: testGoods.id,
  392. goodsName: '测试商品',
  393. price: 100.00,
  394. num: 1,
  395. state: 0,
  396. supplierId: testSupplier.id,
  397. imageFileId: testFile.id
  398. }
  399. ]
  400. };
  401. const response = await client.index.$post({
  402. json: createData
  403. }, {
  404. headers: {
  405. 'Authorization': `Bearer ${userToken}`
  406. }
  407. });
  408. expect(response.status).toBe(201);
  409. if (response.status === 201) {
  410. const data = await response.json();
  411. // 验证订单和商品都设置了正确的用户权限
  412. expect(data.userId).toBe(testUser.id);
  413. expect(data.createdBy).toBe(testUser.id);
  414. // 验证订单商品也设置了正确的用户权限
  415. data.orderGoods.forEach((orderGoods: any) => {
  416. expect(orderGoods.createdBy).toBe(testUser.id);
  417. });
  418. }
  419. });
  420. });
  421. });