tenant-isolation.integration.test.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  1. import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
  2. import { testClient } from 'hono/testing';
  3. import { IntegrationTestDatabase, setupIntegrationDatabaseHooksWithEntities, TestDataFactory } from '@d8d/shared-test-util';
  4. import { JWTUtil } from '@d8d/shared-utils';
  5. import { UserEntityMt, RoleMt } from '@d8d/user-module-mt';
  6. import { AreaEntityMt, AreaLevel } from '@d8d/geo-areas-mt';
  7. import { FileMt } from '@d8d/file-module-mt';
  8. import { userDeliveryAddressRoutesMt, adminDeliveryAddressRoutesMt } from '../../src/routes';
  9. import { DeliveryAddressMt } from '../../src/entities';
  10. // 设置集成测试钩子
  11. setupIntegrationDatabaseHooksWithEntities([UserEntityMt, RoleMt, AreaEntityMt, DeliveryAddressMt, FileMt])
  12. describe('多租户数据隔离测试', () => {
  13. let userClient: ReturnType<typeof testClient<typeof userDeliveryAddressRoutesMt>>;
  14. let adminClient: ReturnType<typeof testClient<typeof adminDeliveryAddressRoutesMt>>;
  15. let tenant1UserToken: string;
  16. let tenant2UserToken: string;
  17. let tenant1AdminToken: string;
  18. let tenant2AdminToken: string;
  19. let tenant1User: UserEntityMt;
  20. let tenant2User: UserEntityMt;
  21. let tenant1Admin: UserEntityMt;
  22. let tenant2Admin: UserEntityMt;
  23. let tenant1Province: AreaEntityMt;
  24. let tenant1City: AreaEntityMt;
  25. let tenant1District: AreaEntityMt;
  26. let tenant2Province: AreaEntityMt;
  27. let tenant2City: AreaEntityMt;
  28. let tenant2District: AreaEntityMt;
  29. beforeEach(async () => {
  30. // 创建测试客户端
  31. userClient = testClient(userDeliveryAddressRoutesMt);
  32. adminClient = testClient(adminDeliveryAddressRoutesMt);
  33. // 创建租户1的测试数据
  34. const tenant1Data = await TestDataFactory.createTestDataSet(1);
  35. tenant1User = tenant1Data.user;
  36. tenant1Province = tenant1Data.province;
  37. tenant1City = tenant1Data.city;
  38. tenant1District = tenant1Data.district;
  39. // 创建租户2的测试数据
  40. const tenant2Data = await TestDataFactory.createTestDataSet(2);
  41. tenant2User = tenant2Data.user;
  42. tenant2Province = tenant2Data.province;
  43. tenant2City = tenant2Data.city;
  44. tenant2District = tenant2Data.district;
  45. // 创建租户1的管理员
  46. const dataSource = await IntegrationTestDatabase.getDataSource();
  47. const userRepository = dataSource.getRepository(UserEntityMt);
  48. tenant1Admin = userRepository.create({
  49. username: `test_admin_tenant1_${Date.now()}`,
  50. password: 'admin_password',
  51. nickname: '租户1管理员',
  52. registrationSource: 'web',
  53. tenantId: 1
  54. });
  55. await userRepository.save(tenant1Admin);
  56. // 创建租户2的管理员
  57. tenant2Admin = userRepository.create({
  58. username: `test_admin_tenant2_${Date.now()}`,
  59. password: 'admin_password',
  60. nickname: '租户2管理员',
  61. registrationSource: 'web',
  62. tenantId: 2
  63. });
  64. await userRepository.save(tenant2Admin);
  65. // 生成各租户用户的token
  66. tenant1UserToken = JWTUtil.generateToken({
  67. id: tenant1User.id,
  68. username: tenant1User.username,
  69. roles: [{name:'user'}],
  70. tenantId: 1
  71. });
  72. tenant2UserToken = JWTUtil.generateToken({
  73. id: tenant2User.id,
  74. username: tenant2User.username,
  75. roles: [{name:'user'}],
  76. tenantId: 2
  77. });
  78. // 生成各租户管理员的token
  79. tenant1AdminToken = JWTUtil.generateToken({
  80. id: tenant1Admin.id,
  81. username: tenant1Admin.username,
  82. roles: [{name:'admin'}],
  83. tenantId: 1
  84. });
  85. tenant2AdminToken = JWTUtil.generateToken({
  86. id: tenant2Admin.id,
  87. username: tenant2Admin.username,
  88. roles: [{name:'admin'}],
  89. tenantId: 2
  90. });
  91. // 为两个租户都创建一些配送地址
  92. await TestDataFactory.createTestDeliveryAddress(
  93. tenant1User.id,
  94. tenant1Province.id,
  95. tenant1City.id,
  96. tenant1District.id,
  97. {
  98. name: '租户1地址1',
  99. phone: '13800138001',
  100. address: '租户1地址1'
  101. }
  102. );
  103. await TestDataFactory.createTestDeliveryAddress(
  104. tenant2User.id,
  105. tenant2Province.id,
  106. tenant2City.id,
  107. tenant2District.id,
  108. {
  109. name: '租户2地址1',
  110. phone: '13800138002',
  111. address: '租户2地址1'
  112. }
  113. );
  114. });
  115. describe('用户路由租户隔离', () => {
  116. it('用户应该只能看到自己租户的地址', async () => {
  117. // 租户1用户访问地址列表
  118. const tenant1Response = await userClient.index.$get({
  119. query: {}
  120. }, {
  121. headers: {
  122. 'Authorization': `Bearer ${tenant1UserToken}`
  123. }
  124. });
  125. expect(tenant1Response.status).toBe(200);
  126. const tenant1Data = await tenant1Response.json();
  127. if (tenant1Data && 'data' in tenant1Data) {
  128. // 租户1用户应该只能看到租户1的地址
  129. tenant1Data.data.forEach((address: any) => {
  130. expect(address.tenantId).toBe(1);
  131. expect(address.userId).toBe(tenant1User.id);
  132. });
  133. }
  134. // 租户2用户访问地址列表
  135. const tenant2Response = await userClient.index.$get({
  136. query: {}
  137. }, {
  138. headers: {
  139. 'Authorization': `Bearer ${tenant2UserToken}`
  140. }
  141. });
  142. expect(tenant2Response.status).toBe(200);
  143. const tenant2Data = await tenant2Response.json();
  144. if (tenant2Data && 'data' in tenant2Data) {
  145. // 租户2用户应该只能看到租户2的地址
  146. tenant2Data.data.forEach((address: any) => {
  147. expect(address.tenantId).toBe(2);
  148. expect(address.userId).toBe(tenant2User.id);
  149. });
  150. }
  151. });
  152. it('用户应该无法访问其他租户的地址详情', async () => {
  153. // 为租户1和租户2都创建地址
  154. const tenant1Address = await TestDataFactory.createTestDeliveryAddress(
  155. tenant1User.id,
  156. tenant1Province.id,
  157. tenant1City.id,
  158. tenant1District.id,
  159. {
  160. name: '租户1专用地址',
  161. phone: '13800138003',
  162. address: '租户1专用地址',
  163. tenantId: 1
  164. }
  165. );
  166. const tenant2Address = await TestDataFactory.createTestDeliveryAddress(
  167. tenant2User.id,
  168. tenant2Province.id,
  169. tenant2City.id,
  170. tenant2District.id,
  171. {
  172. name: '租户2专用地址',
  173. phone: '13800138004',
  174. address: '租户2专用地址',
  175. tenantId: 2
  176. }
  177. );
  178. // 租户1用户尝试访问租户2的地址
  179. const crossTenantResponse = await userClient[':id'].$get({
  180. param: { id: tenant2Address.id }
  181. }, {
  182. headers: {
  183. 'Authorization': `Bearer ${tenant1UserToken}`
  184. }
  185. });
  186. // 应该返回404,因为租户隔离机制应该阻止跨租户访问
  187. expect(crossTenantResponse.status).toBe(404);
  188. // 租户2用户尝试访问租户1的地址
  189. const crossTenantResponse2 = await userClient[':id'].$get({
  190. param: { id: tenant1Address.id }
  191. }, {
  192. headers: {
  193. 'Authorization': `Bearer ${tenant2UserToken}`
  194. }
  195. });
  196. // 应该返回404
  197. expect(crossTenantResponse2.status).toBe(404);
  198. });
  199. it('用户应该无法操作其他租户的地址', async () => {
  200. // 为租户2创建一个地址
  201. const tenant2Address = await TestDataFactory.createTestDeliveryAddress(
  202. tenant2User.id,
  203. tenant2Province.id,
  204. tenant2City.id,
  205. tenant2District.id,
  206. {
  207. name: '租户2地址',
  208. phone: '13800138005',
  209. address: '租户2地址',
  210. tenantId: 2
  211. }
  212. );
  213. // 租户1用户尝试更新租户2的地址
  214. const updateResponse = await userClient[':id'].$put({
  215. param: { id: tenant2Address.id },
  216. json: { name: '尝试更新' }
  217. }, {
  218. headers: {
  219. 'Authorization': `Bearer ${tenant1UserToken}`
  220. }
  221. });
  222. // 应该返回404
  223. expect(updateResponse.status).toBe(404);
  224. // 租户1用户尝试删除租户2的地址
  225. const deleteResponse = await userClient[':id'].$delete({
  226. param: { id: tenant2Address.id }
  227. }, {
  228. headers: {
  229. 'Authorization': `Bearer ${tenant1UserToken}`
  230. }
  231. });
  232. // 应该返回404
  233. expect(deleteResponse.status).toBe(404);
  234. });
  235. });
  236. describe('管理员路由租户隔离', () => {
  237. it('管理员应该只能管理自己租户的地址', async () => {
  238. // 租户1管理员访问地址列表
  239. const tenant1Response = await adminClient.index.$get({
  240. query: {}
  241. }, {
  242. headers: {
  243. 'Authorization': `Bearer ${tenant1AdminToken}`
  244. }
  245. });
  246. expect(tenant1Response.status).toBe(200);
  247. const tenant1Data = await tenant1Response.json();
  248. if (tenant1Data && 'data' in tenant1Data) {
  249. // 租户1管理员应该只能看到租户1的地址
  250. tenant1Data.data.forEach((address: any) => {
  251. expect(address.tenantId).toBe(1);
  252. });
  253. }
  254. // 租户2管理员访问地址列表
  255. const tenant2Response = await adminClient.index.$get({
  256. query: {}
  257. }, {
  258. headers: {
  259. 'Authorization': `Bearer ${tenant2AdminToken}`
  260. }
  261. });
  262. expect(tenant2Response.status).toBe(200);
  263. const tenant2Data = await tenant2Response.json();
  264. if (tenant2Data && 'data' in tenant2Data) {
  265. // 租户2管理员应该只能看到租户2的地址
  266. tenant2Data.data.forEach((address: any) => {
  267. expect(address.tenantId).toBe(2);
  268. });
  269. }
  270. });
  271. it('管理员应该无法访问其他租户的地址详情', async () => {
  272. // 为租户2创建一个地址
  273. const tenant2Address = await TestDataFactory.createTestDeliveryAddress(
  274. tenant2User.id,
  275. tenant2Province.id,
  276. tenant2City.id,
  277. tenant2District.id,
  278. {
  279. name: '租户2管理员测试地址',
  280. phone: '13800138006',
  281. address: '租户2管理员测试地址',
  282. tenantId: 2
  283. }
  284. );
  285. // 租户1管理员尝试访问租户2的地址
  286. const crossTenantResponse = await adminClient[':id'].$get({
  287. param: { id: tenant2Address.id }
  288. }, {
  289. headers: {
  290. 'Authorization': `Bearer ${tenant1AdminToken}`
  291. }
  292. });
  293. // 应该返回404
  294. expect(crossTenantResponse.status).toBe(404);
  295. });
  296. it('管理员应该无法操作其他租户的地址', async () => {
  297. // 为租户2创建一个地址
  298. const tenant2Address = await TestDataFactory.createTestDeliveryAddress(
  299. tenant2User.id,
  300. tenant2Province.id,
  301. tenant2City.id,
  302. tenant2District.id,
  303. {
  304. name: '租户2操作测试地址',
  305. phone: '13800138007',
  306. address: '租户2操作测试地址',
  307. tenantId: 2
  308. }
  309. );
  310. // 租户1管理员尝试更新租户2的地址
  311. const updateResponse = await adminClient[':id'].$put({
  312. param: { id: tenant2Address.id },
  313. json: { name: '租户1尝试更新' }
  314. }, {
  315. headers: {
  316. 'Authorization': `Bearer ${tenant1AdminToken}`
  317. }
  318. });
  319. // 应该返回404
  320. expect(updateResponse.status).toBe(404);
  321. // 租户1管理员尝试删除租户2的地址
  322. const deleteResponse = await adminClient[':id'].$delete({
  323. param: { id: tenant2Address.id }
  324. }, {
  325. headers: {
  326. 'Authorization': `Bearer ${tenant1AdminToken}`
  327. }
  328. });
  329. // 应该返回404
  330. expect(deleteResponse.status).toBe(404);
  331. });
  332. it('管理员应该只能在自己租户内创建地址', async () => {
  333. // 租户1管理员为租户1用户创建地址
  334. const createData = {
  335. userId: tenant1User.id,
  336. name: '租户1管理员创建',
  337. phone: '13800138008',
  338. address: '租户1管理员创建',
  339. receiverProvince: tenant1Province.id,
  340. receiverCity: tenant1City.id,
  341. receiverDistrict: tenant1District.id,
  342. receiverTown: 1,
  343. state: 1,
  344. isDefault: 0
  345. };
  346. const response = await adminClient.index.$post({
  347. json: createData
  348. }, {
  349. headers: {
  350. 'Authorization': `Bearer ${tenant1AdminToken}`
  351. }
  352. });
  353. expect(response.status).toBe(201);
  354. // 验证创建的地址属于租户1
  355. if (response.status === 201) {
  356. const data = await response.json();
  357. expect(data.tenantId).toBe(1);
  358. }
  359. });
  360. });
  361. describe('地区数据租户隔离', () => {
  362. it('应该使用正确的租户地区数据', async () => {
  363. // 租户1用户创建地址,应该使用租户1的地区数据
  364. const createData = {
  365. name: '租户1地区测试',
  366. phone: '13800138009',
  367. address: '租户1地区测试',
  368. receiverProvince: tenant1Province.id,
  369. receiverCity: tenant1City.id,
  370. receiverDistrict: tenant1District.id,
  371. receiverTown: 1,
  372. state: 1,
  373. isDefault: 0
  374. };
  375. const response = await userClient.index.$post({
  376. json: createData
  377. }, {
  378. headers: {
  379. 'Authorization': `Bearer ${tenant1UserToken}`
  380. }
  381. });
  382. expect(response.status).toBe(201);
  383. // 租户1用户尝试使用租户2的地区数据创建地址
  384. const crossTenantAreaData = {
  385. name: '跨租户地区测试',
  386. phone: '13800138010',
  387. address: '跨租户地区测试',
  388. receiverProvince: tenant2Province.id, // 租户2的省份
  389. receiverCity: tenant1City.id,
  390. receiverDistrict: tenant1District.id,
  391. receiverTown: 1,
  392. state: 1,
  393. isDefault: 0
  394. };
  395. const crossTenantResponse = await userClient.index.$post({
  396. json: crossTenantAreaData
  397. }, {
  398. headers: {
  399. 'Authorization': `Bearer ${tenant1UserToken}`
  400. }
  401. });
  402. // 应该返回400,因为地区数据不属于当前租户
  403. expect(crossTenantResponse.status).toBe(400);
  404. });
  405. });
  406. });