admin-goods-routes.integration.test.ts 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552
  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 { File } from '@d8d/file-module';
  7. import { Supplier } from '@d8d/supplier-module';
  8. import { Merchant } from '@d8d/merchant-module';
  9. import { adminGoodsRoutes } from '../../src/routes';
  10. import { Goods, GoodsCategory } from '../../src/entities';
  11. // 设置集成测试钩子
  12. setupIntegrationDatabaseHooksWithEntities([
  13. UserEntity, Role, Goods, GoodsCategory, File, Supplier, Merchant
  14. ])
  15. describe('管理员商品管理API集成测试', () => {
  16. let client: ReturnType<typeof testClient<typeof adminGoodsRoutes>>;
  17. let adminToken: string;
  18. let testUser: UserEntity;
  19. let testAdmin: UserEntity;
  20. let testCategory: GoodsCategory;
  21. let testSupplier: Supplier;
  22. let testMerchant: Merchant;
  23. beforeEach(async () => {
  24. // 创建测试客户端
  25. client = testClient(adminGoodsRoutes);
  26. // 获取数据源
  27. const dataSource = await IntegrationTestDatabase.getDataSource();
  28. // 创建测试用户
  29. const userRepository = dataSource.getRepository(UserEntity);
  30. testUser = userRepository.create({
  31. username: `test_user_${Math.floor(Math.random() * 100000)}`,
  32. password: 'test_password',
  33. nickname: '测试用户',
  34. registrationSource: 'web'
  35. });
  36. await userRepository.save(testUser);
  37. // 创建测试管理员用户
  38. testAdmin = userRepository.create({
  39. username: `test_admin_${Math.floor(Math.random() * 100000)}`,
  40. password: 'admin_password',
  41. nickname: '测试管理员',
  42. registrationSource: 'web'
  43. });
  44. await userRepository.save(testAdmin);
  45. // 生成测试管理员的token
  46. adminToken = JWTUtil.generateToken({
  47. id: testAdmin.id,
  48. username: testAdmin.username,
  49. roles: [{name:'admin'}]
  50. });
  51. // 创建测试商品分类
  52. const categoryRepository = dataSource.getRepository(GoodsCategory);
  53. testCategory = categoryRepository.create({
  54. name: '测试分类',
  55. parentId: 0,
  56. level: 1,
  57. state: 1,
  58. createdBy: testUser.id
  59. });
  60. await categoryRepository.save(testCategory);
  61. // 创建测试供应商
  62. const supplierRepository = dataSource.getRepository(Supplier);
  63. testSupplier = supplierRepository.create({
  64. name: '测试供应商',
  65. username: `test_supplier_${Math.floor(Math.random() * 100000)}`,
  66. password: 'password123',
  67. phone: '13800138000',
  68. realname: '测试供应商',
  69. state: 1,
  70. createdBy: testUser.id
  71. });
  72. await supplierRepository.save(testSupplier);
  73. // 创建测试商户
  74. const merchantRepository = dataSource.getRepository(Merchant);
  75. testMerchant = merchantRepository.create({
  76. name: '测试商户',
  77. username: `test_merchant_${Math.floor(Math.random() * 100000)}`,
  78. password: 'password123',
  79. phone: '13800138001',
  80. realname: '测试商户',
  81. state: 1,
  82. createdBy: testUser.id
  83. });
  84. await merchantRepository.save(testMerchant);
  85. });
  86. describe('GET /goods', () => {
  87. it('应该返回所有商品列表', async () => {
  88. // 创建多个用户的商品
  89. const dataSource = await IntegrationTestDatabase.getDataSource();
  90. const goodsRepository = dataSource.getRepository(Goods);
  91. const userGoods1 = goodsRepository.create({
  92. name: '用户商品1',
  93. price: 100.00,
  94. costPrice: 80.00,
  95. categoryId1: testCategory.id,
  96. categoryId2: testCategory.id,
  97. categoryId3: testCategory.id,
  98. goodsType: 1,
  99. supplierId: testSupplier.id,
  100. merchantId: testMerchant.id,
  101. state: 1,
  102. stock: 100,
  103. lowestBuy: 1,
  104. createdBy: testUser.id
  105. });
  106. await goodsRepository.save(userGoods1);
  107. const userGoods2 = goodsRepository.create({
  108. name: '用户商品2',
  109. price: 200.00,
  110. costPrice: 160.00,
  111. categoryId1: testCategory.id,
  112. categoryId2: testCategory.id,
  113. categoryId3: testCategory.id,
  114. goodsType: 1,
  115. supplierId: testSupplier.id,
  116. merchantId: testMerchant.id,
  117. state: 1,
  118. stock: 50,
  119. lowestBuy: 1,
  120. createdBy: testAdmin.id
  121. });
  122. await goodsRepository.save(userGoods2);
  123. const response = await client.index.$get({
  124. query: {}
  125. }, {
  126. headers: {
  127. 'Authorization': `Bearer ${adminToken}`
  128. }
  129. });
  130. console.debug('管理员商品列表响应状态:', response.status);
  131. expect(response.status).toBe(200);
  132. if (response.status === 200) {
  133. const data = await response.json();
  134. expect(data).toHaveProperty('data');
  135. expect(Array.isArray(data.data)).toBe(true);
  136. // 验证返回所有用户的商品(管理员可以访问所有数据)
  137. const userGoodsCount = data.data.filter((goods: any) => goods.createdBy === testUser.id).length;
  138. const adminGoodsCount = data.data.filter((goods: any) => goods.createdBy === testAdmin.id).length;
  139. expect(userGoodsCount).toBeGreaterThan(0);
  140. expect(adminGoodsCount).toBeGreaterThan(0);
  141. }
  142. });
  143. it('应该拒绝未认证用户的访问', async () => {
  144. const response = await client.index.$get({
  145. query: {}
  146. });
  147. expect(response.status).toBe(401);
  148. });
  149. });
  150. describe('POST /goods', () => {
  151. it('应该成功创建商品并可以指定权限', async () => {
  152. const createData = {
  153. name: '管理员创建商品',
  154. price: 150.00,
  155. costPrice: 120.00,
  156. categoryId1: testCategory.id,
  157. categoryId2: testCategory.id,
  158. categoryId3: testCategory.id,
  159. goodsType: 1,
  160. supplierId: testSupplier.id,
  161. merchantId: testMerchant.id,
  162. state: 1,
  163. stock: 80,
  164. lowestBuy: 1,
  165. createdBy: testUser.id // 管理员可以指定创建人
  166. };
  167. const response = await client.index.$post({
  168. json: createData
  169. }, {
  170. headers: {
  171. 'Authorization': `Bearer ${adminToken}`
  172. }
  173. });
  174. console.debug('管理员创建商品响应状态:', response.status);
  175. if (response.status !== 201) {
  176. const errorData = await response.json();
  177. console.debug('管理员创建商品错误响应:', errorData);
  178. }
  179. expect(response.status).toBe(201);
  180. if (response.status === 201) {
  181. const data = await response.json();
  182. expect(data).toHaveProperty('id');
  183. expect(data.name).toBe(createData.name);
  184. expect(data.price).toBe(Number(createData.price));
  185. expect(data.createdBy).toBe(testUser.id); // 验证可以指定创建人
  186. }
  187. });
  188. it('应该验证创建商品的必填字段', async () => {
  189. const invalidData = {
  190. // 缺少必填字段
  191. name: '',
  192. price: -1,
  193. categoryId1: -1
  194. };
  195. const response = await client.index.$post({
  196. json: invalidData
  197. }, {
  198. headers: {
  199. 'Authorization': `Bearer ${adminToken}`
  200. }
  201. });
  202. expect(response.status).toBe(400);
  203. });
  204. });
  205. describe('GET /goods/:id', () => {
  206. it('应该返回指定商品的详情', async () => {
  207. // 先创建一个商品
  208. const dataSource = await IntegrationTestDatabase.getDataSource();
  209. const goodsRepository = dataSource.getRepository(Goods);
  210. const testGoods = goodsRepository.create({
  211. name: '测试商品详情',
  212. price: 100.00,
  213. costPrice: 80.00,
  214. categoryId1: testCategory.id,
  215. categoryId2: testCategory.id,
  216. categoryId3: testCategory.id,
  217. goodsType: 1,
  218. supplierId: testSupplier.id,
  219. merchantId: testMerchant.id,
  220. state: 1,
  221. stock: 100,
  222. lowestBuy: 1,
  223. createdBy: testUser.id
  224. });
  225. await goodsRepository.save(testGoods);
  226. const response = await client[':id'].$get({
  227. param: { id: testGoods.id }
  228. }, {
  229. headers: {
  230. 'Authorization': `Bearer ${adminToken}`
  231. }
  232. });
  233. console.debug('管理员商品详情响应状态:', response.status);
  234. expect(response.status).toBe(200);
  235. if (response.status === 200) {
  236. const data = await response.json();
  237. expect(data.id).toBe(testGoods.id);
  238. expect(data.name).toBe(testGoods.name);
  239. expect(data.createdBy).toBe(testUser.id);
  240. }
  241. });
  242. it('应该处理不存在的商品', async () => {
  243. const response = await client[':id'].$get({
  244. param: { id: 999999 }
  245. }, {
  246. headers: {
  247. 'Authorization': `Bearer ${adminToken}`
  248. }
  249. });
  250. expect(response.status).toBe(404);
  251. });
  252. });
  253. describe('PUT /goods/:id', () => {
  254. it('应该成功更新任何商品', async () => {
  255. // 先创建一个商品
  256. const dataSource = await IntegrationTestDatabase.getDataSource();
  257. const goodsRepository = dataSource.getRepository(Goods);
  258. const testGoods = goodsRepository.create({
  259. name: '测试更新商品',
  260. price: 100.00,
  261. costPrice: 80.00,
  262. categoryId1: testCategory.id,
  263. categoryId2: testCategory.id,
  264. categoryId3: testCategory.id,
  265. goodsType: 1,
  266. supplierId: testSupplier.id,
  267. merchantId: testMerchant.id,
  268. state: 1,
  269. stock: 100,
  270. lowestBuy: 1,
  271. createdBy: testUser.id
  272. });
  273. await goodsRepository.save(testGoods);
  274. const updateData = {
  275. name: '管理员更新后的商品名称',
  276. price: 120.00,
  277. state: 2,
  278. updatedBy: testAdmin.id // 管理员可以指定更新人
  279. };
  280. const response = await client[':id'].$put({
  281. param: { id: testGoods.id },
  282. json: updateData
  283. }, {
  284. headers: {
  285. 'Authorization': `Bearer ${adminToken}`
  286. }
  287. });
  288. console.debug('管理员更新商品响应状态:', response.status);
  289. expect(response.status).toBe(200);
  290. if (response.status === 200) {
  291. const data = await response.json();
  292. expect(data.name).toBe(updateData.name);
  293. expect(data.price).toBe(Number(updateData.price));
  294. expect(data.state).toBe(updateData.state);
  295. expect(data.updatedBy).toBe(testAdmin.id); // 验证可以指定更新人
  296. }
  297. });
  298. });
  299. describe('DELETE /goods/:id', () => {
  300. it('应该成功删除任何商品', async () => {
  301. // 先创建一个商品
  302. const dataSource = await IntegrationTestDatabase.getDataSource();
  303. const goodsRepository = dataSource.getRepository(Goods);
  304. const testGoods = goodsRepository.create({
  305. name: '测试删除商品',
  306. price: 100.00,
  307. costPrice: 80.00,
  308. categoryId1: testCategory.id,
  309. categoryId2: testCategory.id,
  310. categoryId3: testCategory.id,
  311. goodsType: 1,
  312. supplierId: testSupplier.id,
  313. merchantId: testMerchant.id,
  314. state: 1,
  315. stock: 100,
  316. lowestBuy: 1,
  317. createdBy: testUser.id
  318. });
  319. await goodsRepository.save(testGoods);
  320. const response = await client[':id'].$delete({
  321. param: { id: testGoods.id }
  322. }, {
  323. headers: {
  324. 'Authorization': `Bearer ${adminToken}`
  325. }
  326. });
  327. console.debug('管理员删除商品响应状态:', response.status);
  328. expect(response.status).toBe(204);
  329. });
  330. });
  331. describe('商品状态管理测试', () => {
  332. it('应该正确处理商品状态变更', async () => {
  333. const dataSource = await IntegrationTestDatabase.getDataSource();
  334. const goodsRepository = dataSource.getRepository(Goods);
  335. // 创建可用状态的商品
  336. const activeGoods = goodsRepository.create({
  337. name: '可用商品',
  338. price: 100.00,
  339. costPrice: 80.00,
  340. categoryId1: testCategory.id,
  341. categoryId2: testCategory.id,
  342. categoryId3: testCategory.id,
  343. goodsType: 1,
  344. supplierId: testSupplier.id,
  345. merchantId: testMerchant.id,
  346. state: 1,
  347. stock: 100,
  348. lowestBuy: 1,
  349. createdBy: testUser.id
  350. });
  351. await goodsRepository.save(activeGoods);
  352. // 创建不可用状态的商品
  353. const inactiveGoods = goodsRepository.create({
  354. name: '不可用商品',
  355. price: 200.00,
  356. costPrice: 160.00,
  357. categoryId1: testCategory.id,
  358. categoryId2: testCategory.id,
  359. categoryId3: testCategory.id,
  360. goodsType: 1,
  361. supplierId: testSupplier.id,
  362. merchantId: testMerchant.id,
  363. state: 2,
  364. stock: 50,
  365. lowestBuy: 1,
  366. createdBy: testUser.id
  367. });
  368. await goodsRepository.save(inactiveGoods);
  369. // 验证状态过滤
  370. const response = await client.index.$get({
  371. query: { filters: JSON.stringify({ state: 1 }) }
  372. }, {
  373. headers: {
  374. 'Authorization': `Bearer ${adminToken}`
  375. }
  376. });
  377. expect(response.status).toBe(200);
  378. const data = await response.json();
  379. // 类型检查确保data属性存在
  380. if ('data' in data && Array.isArray(data.data)) {
  381. // 应该只返回可用状态的商品
  382. const activeGoodsInResponse = data.data.filter((goods: any) => goods.state === 1);
  383. const inactiveGoodsInResponse = data.data.filter((goods: any) => goods.state === 2);
  384. expect(activeGoodsInResponse.length).toBeGreaterThan(0);
  385. expect(inactiveGoodsInResponse.length).toBe(0);
  386. } else {
  387. // 如果响应是错误格式,应该失败
  388. expect(data).toHaveProperty('data');
  389. }
  390. });
  391. });
  392. describe('商品库存管理测试', () => {
  393. it('应该正确处理商品库存更新', async () => {
  394. const dataSource = await IntegrationTestDatabase.getDataSource();
  395. const goodsRepository = dataSource.getRepository(Goods);
  396. // 创建一个商品
  397. const testGoods = goodsRepository.create({
  398. name: '库存测试商品',
  399. price: 100.00,
  400. costPrice: 80.00,
  401. categoryId1: testCategory.id,
  402. categoryId2: testCategory.id,
  403. categoryId3: testCategory.id,
  404. goodsType: 1,
  405. supplierId: testSupplier.id,
  406. merchantId: testMerchant.id,
  407. state: 1,
  408. stock: 100,
  409. lowestBuy: 1,
  410. createdBy: testUser.id
  411. });
  412. await goodsRepository.save(testGoods);
  413. // 更新库存
  414. const updateData = {
  415. stock: 50
  416. };
  417. const response = await client[':id'].$put({
  418. param: { id: testGoods.id },
  419. json: updateData
  420. }, {
  421. headers: {
  422. 'Authorization': `Bearer ${adminToken}`
  423. }
  424. });
  425. expect(response.status).toBe(200);
  426. if (response.status === 200) {
  427. const data = await response.json();
  428. expect(data.stock).toBe(Number(updateData.stock));
  429. }
  430. });
  431. });
  432. describe('管理员权限验证测试', () => {
  433. it('应该验证管理员可以访问所有数据', async () => {
  434. const dataSource = await IntegrationTestDatabase.getDataSource();
  435. const goodsRepository = dataSource.getRepository(Goods);
  436. // 创建多个用户的商品
  437. const userGoods = goodsRepository.create({
  438. name: '用户商品',
  439. price: 100.00,
  440. costPrice: 80.00,
  441. categoryId1: testCategory.id,
  442. categoryId2: testCategory.id,
  443. categoryId3: testCategory.id,
  444. goodsType: 1,
  445. supplierId: testSupplier.id,
  446. merchantId: testMerchant.id,
  447. state: 1,
  448. stock: 100,
  449. lowestBuy: 1,
  450. createdBy: testUser.id
  451. });
  452. await goodsRepository.save(userGoods);
  453. const adminGoods = goodsRepository.create({
  454. name: '管理员商品',
  455. price: 200.00,
  456. costPrice: 160.00,
  457. categoryId1: testCategory.id,
  458. categoryId2: testCategory.id,
  459. categoryId3: testCategory.id,
  460. goodsType: 1,
  461. supplierId: testSupplier.id,
  462. merchantId: testMerchant.id,
  463. state: 1,
  464. stock: 50,
  465. lowestBuy: 1,
  466. createdBy: testAdmin.id
  467. });
  468. await goodsRepository.save(adminGoods);
  469. // 使用管理员token获取列表
  470. const response = await client.index.$get({
  471. query: {}
  472. }, {
  473. headers: {
  474. 'Authorization': `Bearer ${adminToken}`
  475. }
  476. });
  477. expect(response.status).toBe(200);
  478. const data = await response.json();
  479. // 类型检查确保data属性存在
  480. if ('data' in data && Array.isArray(data.data)) {
  481. // 验证返回所有用户的商品
  482. const userGoodsInResponse = data.data.filter((goods: any) => goods.createdBy === testUser.id);
  483. const adminGoodsInResponse = data.data.filter((goods: any) => goods.createdBy === testAdmin.id);
  484. expect(userGoodsInResponse.length).toBeGreaterThan(0);
  485. expect(adminGoodsInResponse.length).toBeGreaterThan(0);
  486. } else {
  487. // 如果响应是错误格式,应该失败
  488. expect(data).toHaveProperty('data');
  489. }
  490. });
  491. });
  492. });