2
0

advertisements.integration.test.ts 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. import { describe, it, expect, beforeEach, vi, afterEach } 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 { UserEntityMt, RoleMt } from '@d8d/user-module-mt';
  6. import { FileMt } from '@d8d/file-module-mt';
  7. import { advertisementRoutes } from '../../src/routes';
  8. import { Advertisement } from '../../src/entities/advertisement.entity';
  9. import { AdvertisementType } from '../../src/entities/advertisement-type.entity';
  10. // 设置集成测试钩子
  11. setupIntegrationDatabaseHooksWithEntities([UserEntityMt, FileMt, RoleMt, Advertisement, AdvertisementType])
  12. describe('多租户广告管理API集成测试', () => {
  13. let client: ReturnType<typeof testClient<typeof advertisementRoutes>>;
  14. let testToken: string;
  15. let testUser: UserEntityMt;
  16. let testAdvertisementType: AdvertisementType;
  17. beforeEach(async () => {
  18. // 创建测试客户端
  19. client = testClient(advertisementRoutes);
  20. // 获取数据源
  21. const dataSource = await IntegrationTestDatabase.getDataSource();
  22. // 创建测试用户
  23. const userRepository = dataSource.getRepository(UserEntityMt);
  24. testUser = userRepository.create({
  25. username: `test_user_${Date.now()}`,
  26. password: 'test_password',
  27. nickname: '测试用户',
  28. registrationSource: 'web',
  29. tenantId: 1
  30. });
  31. await userRepository.save(testUser);
  32. // 创建测试广告类型
  33. const advertisementTypeRepository = dataSource.getRepository(AdvertisementType);
  34. testAdvertisementType = advertisementTypeRepository.create({
  35. name: '首页轮播',
  36. code: 'home_banner',
  37. remark: '用于首页轮播图展示',
  38. status: 1,
  39. tenantId: 1
  40. });
  41. await advertisementTypeRepository.save(testAdvertisementType);
  42. // 生成测试用户的token
  43. testToken = JWTUtil.generateToken({
  44. id: testUser.id,
  45. username: testUser.username,
  46. roles: [{name:'user'}]
  47. });
  48. });
  49. describe('GET /advertisements', () => {
  50. it('应该返回广告列表', async () => {
  51. const response = await client.index.$get({
  52. query: {}
  53. }, {
  54. headers: {
  55. 'Authorization': `Bearer ${testToken}`
  56. }
  57. });
  58. console.debug('广告列表响应状态:', response.status);
  59. expect(response.status).toBe(200);
  60. if (response.status === 200) {
  61. const data = await response.json();
  62. expect(data).toHaveProperty('data');
  63. expect(Array.isArray(data.data)).toBe(true);
  64. }
  65. });
  66. it('应该拒绝未认证用户的访问', async () => {
  67. const response = await client.index.$get({
  68. query: {}
  69. });
  70. expect(response.status).toBe(401);
  71. });
  72. });
  73. describe('POST /advertisements', () => {
  74. it('应该成功创建广告', async () => {
  75. const createData = {
  76. title: '测试广告',
  77. typeId: testAdvertisementType.id,
  78. code: 'test_ad',
  79. url: 'https://example.com',
  80. sort: 10,
  81. status: 1,
  82. actionType: 1
  83. };
  84. const response = await client.index.$post({
  85. json: createData
  86. }, {
  87. headers: {
  88. 'Authorization': `Bearer ${testToken}`
  89. }
  90. });
  91. console.debug('创建广告响应状态:', response.status);
  92. expect(response.status).toBe(201);
  93. if (response.status === 201) {
  94. const data = await response.json();
  95. expect(data).toHaveProperty('id');
  96. expect(data.title).toBe(createData.title);
  97. expect(data.code).toBe(createData.code);
  98. expect(data.status).toBe(createData.status);
  99. }
  100. });
  101. it('应该验证创建广告的必填字段', async () => {
  102. const invalidData = {
  103. // 缺少必填字段
  104. title: '',
  105. typeId: 0,
  106. code: '',
  107. url: 'https://example.com'
  108. };
  109. const response = await client.index.$post({
  110. json: invalidData
  111. }, {
  112. headers: {
  113. 'Authorization': `Bearer ${testToken}`
  114. }
  115. });
  116. expect(response.status).toBe(400);
  117. });
  118. });
  119. describe('GET /advertisements/:id', () => {
  120. it('应该返回指定广告的详情', async () => {
  121. // 先创建一个广告
  122. const dataSource = await IntegrationTestDatabase.getDataSource();
  123. const advertisementRepository = dataSource.getRepository(Advertisement);
  124. const testAdvertisement = advertisementRepository.create({
  125. title: '测试广告详情',
  126. typeId: testAdvertisementType.id,
  127. code: 'test_ad_detail',
  128. url: 'https://example.com',
  129. sort: 5,
  130. status: 1,
  131. actionType: 1,
  132. createdBy: testUser.id,
  133. tenantId: 1
  134. });
  135. await advertisementRepository.save(testAdvertisement);
  136. const response = await client[':id'].$get({
  137. param: { id: testAdvertisement.id }
  138. }, {
  139. headers: {
  140. 'Authorization': `Bearer ${testToken}`
  141. }
  142. });
  143. console.debug('广告详情响应状态:', response.status);
  144. expect(response.status).toBe(200);
  145. if (response.status === 200) {
  146. const data = await response.json();
  147. expect(data.id).toBe(testAdvertisement.id);
  148. expect(data.title).toBe(testAdvertisement.title);
  149. expect(data.code).toBe(testAdvertisement.code);
  150. }
  151. });
  152. it('应该处理不存在的广告', async () => {
  153. const response = await client[':id'].$get({
  154. param: { id: 999999 }
  155. }, {
  156. headers: {
  157. 'Authorization': `Bearer ${testToken}`
  158. }
  159. });
  160. expect(response.status).toBe(404);
  161. });
  162. });
  163. describe('PUT /advertisements/:id', () => {
  164. it('应该成功更新广告', async () => {
  165. // 先创建一个广告
  166. const dataSource = await IntegrationTestDatabase.getDataSource();
  167. const advertisementRepository = dataSource.getRepository(Advertisement);
  168. const testAdvertisement = advertisementRepository.create({
  169. title: '原始广告',
  170. typeId: testAdvertisementType.id,
  171. code: 'original_ad',
  172. url: 'https://example.com',
  173. sort: 5,
  174. status: 1,
  175. actionType: 1,
  176. createdBy: testUser.id,
  177. tenantId: 1
  178. });
  179. await advertisementRepository.save(testAdvertisement);
  180. const updateData = {
  181. title: '更新后的广告',
  182. code: 'updated_ad',
  183. sort: 15
  184. };
  185. const response = await client[':id'].$put({
  186. param: { id: testAdvertisement.id },
  187. json: updateData
  188. }, {
  189. headers: {
  190. 'Authorization': `Bearer ${testToken}`
  191. }
  192. });
  193. console.debug('更新广告响应状态:', response.status);
  194. expect(response.status).toBe(200);
  195. if (response.status === 200) {
  196. const data = await response.json();
  197. expect(data.title).toBe(updateData.title);
  198. expect(data.code).toBe(updateData.code);
  199. expect(data.sort).toBe(updateData.sort);
  200. }
  201. });
  202. });
  203. describe('DELETE /advertisements/:id', () => {
  204. it('应该成功删除广告', async () => {
  205. // 先创建一个广告
  206. const dataSource = await IntegrationTestDatabase.getDataSource();
  207. const advertisementRepository = dataSource.getRepository(Advertisement);
  208. const testAdvertisement = advertisementRepository.create({
  209. title: '待删除广告',
  210. typeId: testAdvertisementType.id,
  211. code: 'delete_ad',
  212. url: 'https://example.com',
  213. sort: 5,
  214. status: 1,
  215. actionType: 1,
  216. createdBy: testUser.id,
  217. tenantId: 1
  218. });
  219. await advertisementRepository.save(testAdvertisement);
  220. const response = await client[':id'].$delete({
  221. param: { id: testAdvertisement.id }
  222. }, {
  223. headers: {
  224. 'Authorization': `Bearer ${testToken}`
  225. }
  226. });
  227. console.debug('删除广告响应状态:', response.status);
  228. expect(response.status).toBe(204);
  229. // 验证广告确实被删除
  230. const deletedAdvertisement = await advertisementRepository.findOne({
  231. where: { id: testAdvertisement.id }
  232. });
  233. expect(deletedAdvertisement).toBeNull();
  234. });
  235. });
  236. describe('租户数据隔离测试', () => {
  237. it('应该确保租户只能访问自己的数据', async () => {
  238. const dataSource = await IntegrationTestDatabase.getDataSource();
  239. const advertisementRepository = dataSource.getRepository(Advertisement);
  240. // 创建租户1的广告
  241. const tenant1Advertisement = advertisementRepository.create({
  242. title: '租户1广告',
  243. typeId: testAdvertisementType.id,
  244. code: 'tenant1_ad',
  245. url: 'https://example.com',
  246. sort: 1,
  247. status: 1,
  248. actionType: 1,
  249. createdBy: testUser.id,
  250. tenantId: 1
  251. });
  252. await advertisementRepository.save(tenant1Advertisement);
  253. // 创建租户2的广告
  254. const tenant2Advertisement = advertisementRepository.create({
  255. title: '租户2广告',
  256. typeId: testAdvertisementType.id,
  257. code: 'tenant2_ad',
  258. url: 'https://example.com',
  259. sort: 1,
  260. status: 1,
  261. actionType: 1,
  262. createdBy: testUser.id,
  263. tenantId: 2
  264. });
  265. await advertisementRepository.save(tenant2Advertisement);
  266. // 测试租户1只能看到自己的广告
  267. const response = await client.index.$get({
  268. query: {}
  269. }, {
  270. headers: {
  271. 'Authorization': `Bearer ${testToken}`
  272. }
  273. });
  274. expect(response.status).toBe(200);
  275. const data = await response.json();
  276. // 验证返回的数据只包含租户1的广告
  277. if ('data' in data) {
  278. const tenant1Ads = data.data.filter((ad: any) => ad.tenantId === 1);
  279. const tenant2Ads = data.data.filter((ad: any) => ad.tenantId === 2);
  280. expect(tenant1Ads.length).toBeGreaterThan(0);
  281. expect(tenant2Ads.length).toBe(0); // 租户1不应该看到租户2的广告
  282. }
  283. });
  284. it('应该防止跨租户数据访问', async () => {
  285. const dataSource = await IntegrationTestDatabase.getDataSource();
  286. const advertisementRepository = dataSource.getRepository(Advertisement);
  287. // 创建租户2的广告
  288. const tenant2Advertisement = advertisementRepository.create({
  289. title: '租户2私有广告',
  290. typeId: testAdvertisementType.id,
  291. code: 'tenant2_private',
  292. url: 'https://example.com',
  293. sort: 1,
  294. status: 1,
  295. actionType: 1,
  296. createdBy: testUser.id,
  297. tenantId: 2
  298. });
  299. await advertisementRepository.save(tenant2Advertisement);
  300. // 租户1尝试访问租户2的广告
  301. const response = await client[':id'].$get({
  302. param: { id: tenant2Advertisement.id }
  303. }, {
  304. headers: {
  305. 'Authorization': `Bearer ${testToken}`
  306. }
  307. });
  308. // 应该返回404,因为租户1不能访问租户2的数据
  309. expect(response.status).toBe(404);
  310. });
  311. });
  312. });