talent-auth.integration.test.ts 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  1. import { describe, it, expect, beforeEach } from 'vitest';
  2. import { testClient } from 'hono/testing';
  3. import {
  4. IntegrationTestDatabase,
  5. setupIntegrationDatabaseHooksWithEntities,
  6. } from '@d8d/shared-test-util';
  7. import { Role, UserEntity, UserService } from '@d8d/core-module/user-module';
  8. import { File } from '@d8d/core-module/file-module';
  9. import { Company } from '@d8d/allin-company-module/entities';
  10. import { Platform } from '@d8d/allin-platform-module/entities';
  11. import {
  12. DisabledPerson,
  13. DisabledBankCard,
  14. DisabledPhoto,
  15. DisabledRemark,
  16. DisabledVisit
  17. } from '@d8d/allin-disability-module/entities';
  18. import { BankName } from '@d8d/bank-names-module/entities';
  19. import { EmploymentOrder, OrderPerson, OrderPersonAsset } from '@d8d/allin-order-module/entities';
  20. import { talentAuthRoutes } from '../../src/routes/index';
  21. import { AuthService } from '../../src/services/index';
  22. import { DisabledStatus, UserType } from '@d8d/shared-types';
  23. import { TestDataFactory } from '../utils/test-data-factory';
  24. // 设置集成测试钩子,包含残疾人实体和公司实体(UserEntity依赖Company)
  25. setupIntegrationDatabaseHooksWithEntities([
  26. UserEntity, Role, File,
  27. DisabledPerson, DisabledBankCard, DisabledPhoto, DisabledRemark, DisabledVisit,
  28. Company, Platform, BankName, EmploymentOrder, OrderPerson, OrderPersonAsset
  29. ])
  30. describe('人才用户认证API集成测试', () => {
  31. let client: ReturnType<typeof testClient<typeof talentAuthRoutes>>;
  32. let authService: AuthService;
  33. let userService: UserService;
  34. let testToken: string;
  35. let testUser: any;
  36. let testPerson: any;
  37. let testIdCard: string;
  38. let testDisabilityId: string;
  39. beforeEach(async () => {
  40. // 创建测试客户端
  41. client = testClient(talentAuthRoutes);
  42. // 获取数据源
  43. const dataSource = await IntegrationTestDatabase.getDataSource();
  44. // 确保数据源已初始化
  45. if (!dataSource.isInitialized) {
  46. await dataSource.initialize();
  47. }
  48. // 初始化服务
  49. userService = new UserService(dataSource);
  50. authService = new AuthService(userService);
  51. // 创建测试人才用户(包含残疾人记录)
  52. const talentData = await TestDataFactory.createTestTalentUser(
  53. dataSource,
  54. {
  55. username: 'talent_user',
  56. password: 'TalentPass123!',
  57. email: 'talent@example.com',
  58. phone: '13800138001'
  59. },
  60. {
  61. name: '张三',
  62. idCard: '110101199001011234',
  63. disabilityId: 'D12345678',
  64. phone: '13800138001'
  65. }
  66. );
  67. testUser = talentData.user;
  68. testPerson = talentData.person;
  69. testIdCard = testPerson.idCard;
  70. testDisabilityId = testPerson.disabilityId;
  71. // 生成测试用户的token
  72. testToken = authService.generateToken(testUser);
  73. });
  74. describe('人才用户登录端点测试 (POST /auth/login)', () => {
  75. it('应该使用身份证号和正确密码成功登录', async () => {
  76. const loginData = {
  77. identifier: testIdCard,
  78. password: 'TalentPass123!'
  79. };
  80. const response = await client.login.$post({
  81. json: loginData
  82. });
  83. expect(response.status).toBe(200);
  84. if (response.status === 200) {
  85. const responseData = await response.json();
  86. expect(responseData).toHaveProperty('token');
  87. expect(responseData).toHaveProperty('user');
  88. expect(responseData.user.username).toBe('talent_user');
  89. expect(responseData.user.userType).toBe(UserType.TALENT);
  90. expect(responseData.user.personId).toBe(testPerson.id);
  91. expect(responseData.user.personInfo).toBeDefined();
  92. expect(responseData.user.personInfo.name).toBe('张三');
  93. expect(responseData.user.personInfo.idCard).toBe(testIdCard);
  94. expect(typeof responseData.token).toBe('string');
  95. expect(responseData.token.length).toBeGreaterThan(0);
  96. }
  97. });
  98. it('应该使用残疾证号和正确密码成功登录', async () => {
  99. const loginData = {
  100. identifier: testDisabilityId,
  101. password: 'TalentPass123!'
  102. };
  103. const response = await client.login.$post({
  104. json: loginData
  105. });
  106. expect(response.status).toBe(200);
  107. if (response.status === 200) {
  108. const responseData = await response.json();
  109. expect(responseData).toHaveProperty('token');
  110. expect(responseData).toHaveProperty('user');
  111. expect(responseData.user.username).toBe('talent_user');
  112. expect(responseData.user.personInfo).toBeDefined();
  113. expect(responseData.user.personInfo.disabilityId).toBe(testDisabilityId);
  114. }
  115. });
  116. it('应该拒绝错误密码的登录', async () => {
  117. const loginData = {
  118. identifier: testIdCard,
  119. password: 'WrongPassword123!'
  120. };
  121. const response = await client.login.$post({
  122. json: loginData
  123. });
  124. expect(response.status).toBe(401);
  125. if (response.status === 401) {
  126. const responseData = await response.json();
  127. expect(responseData.message).toContain('身份证号或密码错误');
  128. }
  129. });
  130. it('应该拒绝不存在的身份证号登录', async () => {
  131. const loginData = {
  132. identifier: '999999999999999999',
  133. password: 'TalentPass123!'
  134. };
  135. const response = await client.login.$post({
  136. json: loginData
  137. });
  138. expect(response.status).toBe(401);
  139. if (response.status === 401) {
  140. const responseData = await response.json();
  141. expect(responseData.message).toContain('身份证号或密码错误');
  142. }
  143. });
  144. it('应该拒绝非人才用户登录(用户类型不是talent)', async () => {
  145. const dataSource = await IntegrationTestDatabase.getDataSource();
  146. // 创建非人才用户
  147. await TestDataFactory.createTestUser(dataSource, {
  148. username: 'non_talent_user',
  149. password: 'Password123!',
  150. email: 'non_talent@example.com',
  151. phone: '13800138002',
  152. userType: UserType.ADMIN,
  153. personId: testPerson.id // 关联残疾人记录但用户类型不是talent
  154. });
  155. const loginData = {
  156. identifier: testIdCard,
  157. password: 'Password123!'
  158. };
  159. const response = await client.login.$post({
  160. json: loginData
  161. });
  162. expect(response.status).toBe(401);
  163. if (response.status === 401) {
  164. const responseData = await response.json();
  165. // 由于getTalentUserByIdentifier只查询TALENT类型用户,非人才用户会被视为不存在
  166. // 这是更安全的做法,不透露用户存在性信息
  167. expect(responseData.message).toContain('身份证号或密码错误');
  168. }
  169. });
  170. it('应该拒绝禁用账户的登录', async () => {
  171. // 禁用测试用户
  172. const dataSource = await IntegrationTestDatabase.getDataSource();
  173. const userRepository = dataSource.getRepository(UserEntity);
  174. await userRepository.update(testUser.id, { isDisabled: DisabledStatus.DISABLED });
  175. const loginData = {
  176. identifier: testIdCard,
  177. password: 'TalentPass123!'
  178. };
  179. const response = await client.login.$post({
  180. json: loginData
  181. });
  182. expect(response.status).toBe(401);
  183. if (response.status === 401) {
  184. const responseData = await response.json();
  185. expect(responseData.message).toContain('账户已禁用');
  186. }
  187. });
  188. });
  189. describe('人才用户信息端点测试 (GET /auth/me)', () => {
  190. it('应该成功获取人才用户信息,包含残疾人详细信息', async () => {
  191. const response = await client.me.$get({},{
  192. headers: {
  193. Authorization: `Bearer ${testToken}`
  194. }
  195. });
  196. expect(response.status).toBe(200);
  197. if (response.status === 200) {
  198. const responseData = await response.json();
  199. expect(responseData.username).toBe('talent_user');
  200. expect(responseData.userType).toBe(UserType.TALENT);
  201. expect(responseData.personId).toBe(testPerson.id);
  202. expect(responseData.personInfo).toBeDefined();
  203. expect(responseData.personInfo.name).toBe('张三');
  204. expect(responseData.personInfo.idCard).toBe(testIdCard);
  205. expect(responseData.personInfo.disabilityId).toBe(testDisabilityId);
  206. expect(responseData.personInfo.phone).toBe('13800138001');
  207. }
  208. });
  209. it('应该拒绝无效令牌的访问', async () => {
  210. const response = await client.me.$get({},{
  211. headers: {
  212. Authorization: 'Bearer invalid_token'
  213. }
  214. });
  215. expect(response.status).toBe(401);
  216. });
  217. it('应该拒绝缺少令牌的访问', async () => {
  218. const response = await client.me.$get();
  219. expect(response.status).toBe(401);
  220. });
  221. });
  222. describe('人才用户退出登录端点测试 (POST /auth/logout)', () => {
  223. it('应该成功退出登录', async () => {
  224. const response = await client.logout.$post({},{
  225. headers: {
  226. Authorization: `Bearer ${testToken}`
  227. }
  228. });
  229. expect(response.status).toBe(200);
  230. if (response.status === 200) {
  231. const responseData = await response.json();
  232. expect(responseData.message).toBe('登出成功');
  233. }
  234. });
  235. it('应该拒绝无效令牌的退出登录', async () => {
  236. const response = await client.logout.$post({},{
  237. headers: {
  238. Authorization: 'Bearer invalid_token'
  239. }
  240. });
  241. expect(response.status).toBe(401);
  242. });
  243. });
  244. describe('人才用户权限验证', () => {
  245. it('verifyTalentUser方法应该正确识别人才用户', async () => {
  246. const dataSource = await IntegrationTestDatabase.getDataSource();
  247. const userService = new UserService(dataSource);
  248. const authService = new AuthService(userService);
  249. const isTalentUser = await authService.verifyTalentUser(testUser.id);
  250. expect(isTalentUser).toBe(true);
  251. // 创建非人才用户测试
  252. const nonTalentUserData = await TestDataFactory.createTestUser(dataSource, {
  253. username: 'non_talent_user2',
  254. password: 'Password123!',
  255. email: 'non_talent2@example.com',
  256. phone: '13800138003',
  257. userType: UserType.ADMIN
  258. });
  259. const isNonTalentUser = await authService.verifyTalentUser(nonTalentUserData.id);
  260. expect(isNonTalentUser).toBe(false);
  261. });
  262. });
  263. describe('数据查询方法测试', () => {
  264. it('getDisabledPersonByIdentifier应该能通过身份证号查询残疾人', async () => {
  265. const dataSource = await IntegrationTestDatabase.getDataSource();
  266. const userService = new UserService(dataSource);
  267. const person = await userService.getDisabledPersonByIdentifier(testIdCard);
  268. expect(person).toBeDefined();
  269. expect(person?.id).toBe(testPerson.id);
  270. expect(person?.name).toBe('张三');
  271. });
  272. it('getDisabledPersonByIdentifier应该能通过残疾证号查询残疾人', async () => {
  273. const dataSource = await IntegrationTestDatabase.getDataSource();
  274. const userService = new UserService(dataSource);
  275. const person = await userService.getDisabledPersonByIdentifier(testDisabilityId);
  276. expect(person).toBeDefined();
  277. expect(person?.id).toBe(testPerson.id);
  278. expect(person?.name).toBe('张三');
  279. });
  280. it('getUserByPersonId应该能通过person_id查询用户', async () => {
  281. const dataSource = await IntegrationTestDatabase.getDataSource();
  282. const userService = new UserService(dataSource);
  283. const user = await userService.getUserByPersonId(testPerson.id);
  284. expect(user).toBeDefined();
  285. expect(user?.id).toBe(testUser.id);
  286. expect(user?.username).toBe('talent_user');
  287. });
  288. it('getTalentUserByIdentifier应该能通过身份证号查询完整人才用户信息', async () => {
  289. const dataSource = await IntegrationTestDatabase.getDataSource();
  290. const userService = new UserService(dataSource);
  291. const user = await userService.getTalentUserByIdentifier(testIdCard);
  292. expect(user).toBeDefined();
  293. expect(user?.id).toBe(testUser.id);
  294. expect(user?.personId).toBe(testPerson.id);
  295. expect(user?.userType).toBe(UserType.TALENT);
  296. });
  297. });
  298. });