create-order.integration.test.ts 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515
  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 goodsCategoryRepository = dataSource.getRepository(GoodsCategory);
  48. const testCategory1 = goodsCategoryRepository.create({
  49. name: '一级分类',
  50. level: 1,
  51. sort: 1,
  52. state: 1,
  53. createdBy: testUser.id
  54. });
  55. await goodsCategoryRepository.save(testCategory1);
  56. const testCategory2 = goodsCategoryRepository.create({
  57. name: '二级分类',
  58. level: 2,
  59. sort: 1,
  60. state: 1,
  61. createdBy: testUser.id
  62. });
  63. await goodsCategoryRepository.save(testCategory2);
  64. const testCategory3 = goodsCategoryRepository.create({
  65. name: '三级分类',
  66. level: 3,
  67. sort: 1,
  68. state: 1,
  69. createdBy: testUser.id
  70. });
  71. await goodsCategoryRepository.save(testCategory3);
  72. // 创建测试供应商
  73. const supplierRepository = dataSource.getRepository(Supplier);
  74. testSupplier = supplierRepository.create({
  75. name: '测试供应商',
  76. username: `test_supplier_${Math.floor(Math.random() * 100000)}`,
  77. password: 'password123',
  78. phone: '13800138000',
  79. realname: '测试供应商',
  80. state: 1,
  81. createdBy: testUser.id
  82. });
  83. await supplierRepository.save(testSupplier);
  84. // 创建测试商户
  85. const merchantRepository = dataSource.getRepository(Merchant);
  86. const testMerchant = merchantRepository.create({
  87. name: '测试商户',
  88. username: `test_merchant_${Math.floor(Math.random() * 100000)}`,
  89. password: 'password123',
  90. phone: '13800138000',
  91. realname: '测试商户',
  92. state: 1,
  93. createdBy: testUser.id
  94. });
  95. await merchantRepository.save(testMerchant);
  96. // 创建测试商品
  97. const goodsRepository = dataSource.getRepository(Goods);
  98. testGoods = goodsRepository.create({
  99. name: '测试商品',
  100. price: 100.00,
  101. costPrice: 80.00,
  102. categoryId1: testCategory1.id,
  103. categoryId2: testCategory2.id,
  104. categoryId3: testCategory3.id,
  105. goodsType: 1,
  106. supplierId: testSupplier.id,
  107. merchantId: testMerchant.id,
  108. state: 1,
  109. stock: 100,
  110. lowestBuy: 1,
  111. createdBy: testUser.id
  112. });
  113. await goodsRepository.save(testGoods);
  114. // 创建测试文件
  115. const fileRepository = dataSource.getRepository(File);
  116. testFile = fileRepository.create({
  117. name: 'test_image.jpg',
  118. type: 'image/jpeg',
  119. size: 102400,
  120. path: 'images/test_image.jpg',
  121. uploadUserId: testUser.id,
  122. uploadTime: new Date(),
  123. createdAt: new Date(),
  124. updatedAt: new Date()
  125. });
  126. await fileRepository.save(testFile);
  127. // 创建测试地区数据
  128. const areaRepository = dataSource.getRepository(AreaEntity);
  129. const testProvince = areaRepository.create({
  130. name: '广东省',
  131. level: 1,
  132. code: '440000',
  133. isDisabled: 0,
  134. isDeleted: 0,
  135. createdBy: testUser.id
  136. });
  137. await areaRepository.save(testProvince);
  138. const testCity = areaRepository.create({
  139. name: '深圳市',
  140. level: 2,
  141. code: '440300',
  142. parentId: testProvince.id,
  143. isDisabled: 0,
  144. isDeleted: 0,
  145. createdBy: testUser.id
  146. });
  147. await areaRepository.save(testCity);
  148. const testDistrict = areaRepository.create({
  149. name: '南山区',
  150. level: 3,
  151. code: '440305',
  152. parentId: testCity.id,
  153. isDisabled: 0,
  154. isDeleted: 0,
  155. createdBy: testUser.id
  156. });
  157. await areaRepository.save(testDistrict);
  158. const testTown = areaRepository.create({
  159. name: '粤海街道',
  160. level: 4,
  161. code: '440305001',
  162. parentId: testDistrict.id,
  163. isDisabled: 0,
  164. isDeleted: 0,
  165. createdBy: testUser.id
  166. });
  167. await areaRepository.save(testTown);
  168. // 创建测试配送地址
  169. const deliveryAddressRepository = dataSource.getRepository(DeliveryAddress);
  170. testDeliveryAddress = deliveryAddressRepository.create({
  171. userId: testUser.id,
  172. name: '收货人姓名',
  173. phone: '13800138000',
  174. receiverProvince: testProvince.id,
  175. receiverCity: testCity.id,
  176. receiverDistrict: testDistrict.id,
  177. receiverTown: testTown.id,
  178. address: '测试地址',
  179. isDefault: 1,
  180. state: 1,
  181. createdBy: testUser.id
  182. });
  183. await deliveryAddressRepository.save(testDeliveryAddress);
  184. });
  185. describe('POST /create-order', () => {
  186. it('应该成功创建订单并关联商品', async () => {
  187. const createData = {
  188. addressId: testDeliveryAddress.id,
  189. productOwn: '自营',
  190. consumeFrom: '积分兑换',
  191. products: [
  192. {
  193. id: testGoods.id,
  194. num: 2
  195. },
  196. {
  197. id: testGoods.id,
  198. num: 1
  199. }
  200. ]
  201. };
  202. const response = await client.index.$post({
  203. json: createData
  204. }, {
  205. headers: {
  206. 'Authorization': `Bearer ${userToken}`
  207. }
  208. });
  209. console.debug('订单创建响应状态:', response.status);
  210. if (response.status !== 201) {
  211. const errorData = await response.json();
  212. console.debug('订单创建错误响应:', errorData);
  213. }
  214. expect(response.status).toBe(201);
  215. if (response.status === 201) {
  216. const data = await response.json();
  217. expect(data).toHaveProperty('orderId');
  218. expect(data).toHaveProperty('orderNo');
  219. expect(data).toHaveProperty('amount');
  220. expect(data).toHaveProperty('payAmount');
  221. // 验证订单商品被正确创建(需要从数据库查询验证)
  222. const dataSource = await IntegrationTestDatabase.getDataSource();
  223. const orderGoodsRepository = dataSource.getRepository(OrderGoods);
  224. const createdOrderGoods = await orderGoodsRepository.find({
  225. where: { orderId: data.orderId }
  226. });
  227. expect(createdOrderGoods).toBeDefined();
  228. expect(Array.isArray(createdOrderGoods)).toBe(true);
  229. expect(createdOrderGoods.length).toBe(2);
  230. // 验证商品信息
  231. createdOrderGoods.forEach((orderGoods: any, index: number) => {
  232. expect(orderGoods.goodsName).toBe(testGoods.name);
  233. expect(orderGoods.price).toBe(testGoods.price);
  234. expect(orderGoods.num).toBe(createData.products[index].num);
  235. expect(orderGoods.createdBy).toBe(testUser.id); // 验证商品创建用户
  236. });
  237. }
  238. });
  239. it('应该验证创建订单的必填字段', async () => {
  240. const invalidData = {
  241. // 缺少必填字段
  242. amount: -1,
  243. orderType: -1
  244. };
  245. const response = await client.index.$post({
  246. json: invalidData
  247. }, {
  248. headers: {
  249. 'Authorization': `Bearer ${userToken}`
  250. }
  251. });
  252. expect(response.status).toBe(400);
  253. });
  254. it('应该处理商品库存不足的情况', async () => {
  255. // 创建一个库存为0的商品
  256. const dataSource = await IntegrationTestDatabase.getDataSource();
  257. const goodsRepository = dataSource.getRepository(Goods);
  258. const outOfStockGoods = goodsRepository.create({
  259. name: '库存不足商品',
  260. price: 100.00,
  261. costPrice: 80.00,
  262. categoryId1: 1,
  263. categoryId2: 1,
  264. categoryId3: 1,
  265. goodsType: 1,
  266. supplierId: 1,
  267. state: 1,
  268. stock: 0, // 库存为0
  269. lowestBuy: 1,
  270. createdBy: testUser.id
  271. });
  272. await goodsRepository.save(outOfStockGoods);
  273. const createData = {
  274. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  275. amount: 100.00,
  276. costAmount: 80.00,
  277. payAmount: 100.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: outOfStockGoods.id,
  288. goodsName: '库存不足商品',
  289. price: 100.00,
  290. num: 1,
  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. // 应该返回错误状态(库存不足)
  305. expect(response.status).toBe(400);
  306. });
  307. it('应该拒绝未认证用户的访问', async () => {
  308. const createData = {
  309. orderNo: `ORDER_${Math.floor(Math.random() * 100000)}`,
  310. amount: 150.00,
  311. costAmount: 120.00,
  312. payAmount: 150.00,
  313. orderType: 1,
  314. payType: 1,
  315. payState: 0,
  316. state: 0,
  317. deliveryAddressId: testDeliveryAddress.id,
  318. merchantId: 1,
  319. supplierId: testSupplier.id,
  320. orderGoods: [
  321. {
  322. goodsId: testGoods.id,
  323. goodsName: '测试商品',
  324. price: 50.00,
  325. num: 3,
  326. state: 0,
  327. supplierId: testSupplier.id,
  328. imageFileId: testFile.id
  329. }
  330. ]
  331. };
  332. const response = await client.index.$post({
  333. json: createData
  334. });
  335. expect(response.status).toBe(401);
  336. });
  337. });
  338. describe('事务性测试', () => {
  339. it('应该确保订单和商品创建的事务一致性', async () => {
  340. const createData = {
  341. addressId: testDeliveryAddress.id,
  342. productOwn: '自营',
  343. consumeFrom: '积分兑换',
  344. products: [
  345. {
  346. id: testGoods.id,
  347. num: 2
  348. }
  349. ]
  350. };
  351. const response = await client.index.$post({
  352. json: createData
  353. }, {
  354. headers: {
  355. 'Authorization': `Bearer ${userToken}`
  356. }
  357. });
  358. expect(response.status).toBe(201);
  359. if (response.status === 201) {
  360. const data = await response.json();
  361. // 验证订单和商品都被正确创建
  362. const dataSource = await IntegrationTestDatabase.getDataSource();
  363. const orderRepository = dataSource.getRepository(Order);
  364. const orderGoodsRepository = dataSource.getRepository(OrderGoods);
  365. // 验证订单存在
  366. const createdOrder = await orderRepository.findOne({
  367. where: { id: data.orderId }
  368. });
  369. expect(createdOrder).toBeDefined();
  370. expect(createdOrder?.orderNo).toBe(data.orderNo);
  371. expect(createdOrder?.userId).toBe(testUser.id);
  372. // 验证订单商品存在
  373. const createdOrderGoods = await orderGoodsRepository.find({
  374. where: { orderId: data.orderId }
  375. });
  376. expect(createdOrderGoods.length).toBe(1);
  377. expect(createdOrderGoods[0].goodsName).toBe(testGoods.name);
  378. expect(createdOrderGoods[0].num).toBe(createData.products[0].num);
  379. }
  380. });
  381. it('应该验证库存更新功能', async () => {
  382. const initialStock = testGoods.stock;
  383. const purchaseQuantity = 2;
  384. const createData = {
  385. addressId: testDeliveryAddress.id,
  386. productOwn: '自营',
  387. consumeFrom: '积分兑换',
  388. products: [
  389. {
  390. id: testGoods.id,
  391. num: purchaseQuantity
  392. }
  393. ]
  394. };
  395. const response = await client.index.$post({
  396. json: createData
  397. }, {
  398. headers: {
  399. 'Authorization': `Bearer ${userToken}`
  400. }
  401. });
  402. expect(response.status).toBe(201);
  403. if (response.status === 201) {
  404. // 验证商品库存被正确更新
  405. const dataSource = await IntegrationTestDatabase.getDataSource();
  406. const goodsRepository = dataSource.getRepository(Goods);
  407. const updatedGoods = await goodsRepository.findOne({
  408. where: { id: testGoods.id }
  409. });
  410. expect(updatedGoods).toBeDefined();
  411. expect(Number(updatedGoods?.stock)).toBe(initialStock - purchaseQuantity);
  412. }
  413. });
  414. });
  415. describe('数据权限测试', () => {
  416. it('应该自动设置当前用户权限', async () => {
  417. const createData = {
  418. addressId: testDeliveryAddress.id,
  419. productOwn: '自营',
  420. consumeFrom: '积分兑换',
  421. products: [
  422. {
  423. id: testGoods.id,
  424. num: 1
  425. }
  426. ]
  427. };
  428. const response = await client.index.$post({
  429. json: createData
  430. }, {
  431. headers: {
  432. 'Authorization': `Bearer ${userToken}`
  433. }
  434. });
  435. expect(response.status).toBe(201);
  436. if (response.status === 201) {
  437. const data = await response.json();
  438. // 验证订单和商品都设置了正确的用户权限(需要从数据库查询验证)
  439. const dataSource = await IntegrationTestDatabase.getDataSource();
  440. const orderRepository = dataSource.getRepository(Order);
  441. const orderGoodsRepository = dataSource.getRepository(OrderGoods);
  442. // 验证订单权限
  443. const createdOrder = await orderRepository.findOne({
  444. where: { id: data.orderId }
  445. });
  446. expect(createdOrder?.userId).toBe(testUser.id);
  447. expect(createdOrder?.createdBy).toBe(testUser.id);
  448. // 验证订单商品权限
  449. const createdOrderGoods = await orderGoodsRepository.find({
  450. where: { orderId: data.orderId }
  451. });
  452. createdOrderGoods.forEach((orderGoods: any) => {
  453. expect(orderGoods.createdBy).toBe(testUser.id);
  454. });
  455. }
  456. });
  457. });
  458. });