talent-auth.integration.test.ts 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  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: '13800138001', // 测试用户的手机号
  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.phone).toBe('13800138001');
  92. expect(responseData.user.personInfo).toBeDefined();
  93. expect(responseData.user.personInfo.name).toBe('张三');
  94. expect(typeof responseData.token).toBe('string');
  95. expect(responseData.token.length).toBeGreaterThan(0);
  96. }
  97. });
  98. it('应该使用身份证号和正确密码成功登录', async () => {
  99. const loginData = {
  100. identifier: testIdCard,
  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.userType).toBe(UserType.TALENT);
  113. expect(responseData.user.personId).toBe(testPerson.id);
  114. expect(responseData.user.personInfo).toBeDefined();
  115. expect(responseData.user.personInfo.name).toBe('张三');
  116. expect(responseData.user.personInfo.idCard).toBe(testIdCard);
  117. expect(typeof responseData.token).toBe('string');
  118. expect(responseData.token.length).toBeGreaterThan(0);
  119. }
  120. });
  121. it('应该使用残疾证号和正确密码成功登录', async () => {
  122. const loginData = {
  123. identifier: testDisabilityId,
  124. password: 'TalentPass123!'
  125. };
  126. const response = await client.login.$post({
  127. json: loginData
  128. });
  129. expect(response.status).toBe(200);
  130. if (response.status === 200) {
  131. const responseData = await response.json();
  132. expect(responseData).toHaveProperty('token');
  133. expect(responseData).toHaveProperty('user');
  134. expect(responseData.user.username).toBe('talent_user');
  135. expect(responseData.user.personInfo).toBeDefined();
  136. expect(responseData.user.personInfo.disabilityId).toBe(testDisabilityId);
  137. }
  138. });
  139. it('应该拒绝错误密码的登录', async () => {
  140. const loginData = {
  141. identifier: testIdCard,
  142. password: 'WrongPassword123!'
  143. };
  144. const response = await client.login.$post({
  145. json: loginData
  146. });
  147. expect(response.status).toBe(401);
  148. if (response.status === 401) {
  149. const responseData = await response.json();
  150. expect(responseData.message).toContain('账号或密码错误');
  151. }
  152. });
  153. it('应该拒绝不存在的身份证号登录', async () => {
  154. const loginData = {
  155. identifier: '999999999999999999',
  156. password: 'TalentPass123!'
  157. };
  158. const response = await client.login.$post({
  159. json: loginData
  160. });
  161. expect(response.status).toBe(401);
  162. if (response.status === 401) {
  163. const responseData = await response.json();
  164. expect(responseData.message).toContain('账号或密码错误');
  165. }
  166. });
  167. it('应该拒绝不存在的手机号登录', async () => {
  168. const loginData = {
  169. identifier: '19999999999', // 不存在的手机号
  170. password: 'TalentPass123!'
  171. };
  172. const response = await client.login.$post({
  173. json: loginData
  174. });
  175. expect(response.status).toBe(401);
  176. if (response.status === 401) {
  177. const responseData = await response.json();
  178. expect(responseData.message).toContain('账号或密码错误');
  179. }
  180. });
  181. it('应该拒绝非人才用户登录(用户类型不是talent)', async () => {
  182. const dataSource = await IntegrationTestDatabase.getDataSource();
  183. // 创建非人才用户
  184. await TestDataFactory.createTestUser(dataSource, {
  185. username: 'non_talent_user',
  186. password: 'Password123!',
  187. email: 'non_talent@example.com',
  188. phone: '13800138002',
  189. userType: UserType.ADMIN,
  190. personId: testPerson.id // 关联残疾人记录但用户类型不是talent
  191. });
  192. const loginData = {
  193. identifier: testIdCard,
  194. password: 'Password123!'
  195. };
  196. const response = await client.login.$post({
  197. json: loginData
  198. });
  199. expect(response.status).toBe(401);
  200. if (response.status === 401) {
  201. const responseData = await response.json();
  202. // 由于getTalentUserByIdentifier只查询TALENT类型用户,非人才用户会被视为不存在
  203. // 这是更安全的做法,不透露用户存在性信息
  204. expect(responseData.message).toContain('账号或密码错误');
  205. }
  206. });
  207. it('应该拒绝禁用账户的登录', async () => {
  208. // 禁用测试用户
  209. const dataSource = await IntegrationTestDatabase.getDataSource();
  210. const userRepository = dataSource.getRepository(UserEntity);
  211. await userRepository.update(testUser.id, { isDisabled: DisabledStatus.DISABLED });
  212. const loginData = {
  213. identifier: testIdCard,
  214. password: 'TalentPass123!'
  215. };
  216. const response = await client.login.$post({
  217. json: loginData
  218. });
  219. expect(response.status).toBe(401);
  220. if (response.status === 401) {
  221. const responseData = await response.json();
  222. expect(responseData.message).toContain('账户已禁用');
  223. }
  224. });
  225. });
  226. describe('人才用户信息端点测试 (GET /auth/me)', () => {
  227. it('应该成功获取人才用户信息,包含残疾人详细信息', async () => {
  228. const response = await client.me.$get({},{
  229. headers: {
  230. Authorization: `Bearer ${testToken}`
  231. }
  232. });
  233. expect(response.status).toBe(200);
  234. if (response.status === 200) {
  235. const responseData = await response.json();
  236. expect(responseData.username).toBe('talent_user');
  237. expect(responseData.userType).toBe(UserType.TALENT);
  238. expect(responseData.personId).toBe(testPerson.id);
  239. expect(responseData.personInfo).toBeDefined();
  240. expect(responseData.personInfo.name).toBe('张三');
  241. expect(responseData.personInfo.idCard).toBe(testIdCard);
  242. expect(responseData.personInfo.disabilityId).toBe(testDisabilityId);
  243. expect(responseData.personInfo.phone).toBe('13800138001');
  244. }
  245. });
  246. it('应该拒绝无效令牌的访问', async () => {
  247. const response = await client.me.$get({},{
  248. headers: {
  249. Authorization: 'Bearer invalid_token'
  250. }
  251. });
  252. expect(response.status).toBe(401);
  253. });
  254. it('应该拒绝缺少令牌的访问', async () => {
  255. const response = await client.me.$get();
  256. expect(response.status).toBe(401);
  257. });
  258. });
  259. describe('人才用户退出登录端点测试 (POST /auth/logout)', () => {
  260. it('应该成功退出登录', async () => {
  261. const response = await client.logout.$post({},{
  262. headers: {
  263. Authorization: `Bearer ${testToken}`
  264. }
  265. });
  266. expect(response.status).toBe(200);
  267. if (response.status === 200) {
  268. const responseData = await response.json();
  269. expect(responseData.message).toBe('登出成功');
  270. }
  271. });
  272. it('应该拒绝无效令牌的退出登录', async () => {
  273. const response = await client.logout.$post({},{
  274. headers: {
  275. Authorization: 'Bearer invalid_token'
  276. }
  277. });
  278. expect(response.status).toBe(401);
  279. });
  280. });
  281. describe('人才用户权限验证', () => {
  282. it('verifyTalentUser方法应该正确识别人才用户', async () => {
  283. const dataSource = await IntegrationTestDatabase.getDataSource();
  284. const userService = new UserService(dataSource);
  285. const authService = new AuthService(userService);
  286. const isTalentUser = await authService.verifyTalentUser(testUser.id);
  287. expect(isTalentUser).toBe(true);
  288. // 创建非人才用户测试
  289. const nonTalentUserData = await TestDataFactory.createTestUser(dataSource, {
  290. username: 'non_talent_user2',
  291. password: 'Password123!',
  292. email: 'non_talent2@example.com',
  293. phone: '13800138003',
  294. userType: UserType.ADMIN
  295. });
  296. const isNonTalentUser = await authService.verifyTalentUser(nonTalentUserData.id);
  297. expect(isNonTalentUser).toBe(false);
  298. });
  299. });
  300. describe('数据查询方法测试', () => {
  301. it('getDisabledPersonByIdentifier应该能通过身份证号查询残疾人', async () => {
  302. const dataSource = await IntegrationTestDatabase.getDataSource();
  303. const userService = new UserService(dataSource);
  304. const person = await userService.getDisabledPersonByIdentifier(testIdCard);
  305. expect(person).toBeDefined();
  306. expect(person?.id).toBe(testPerson.id);
  307. expect(person?.name).toBe('张三');
  308. });
  309. it('getDisabledPersonByIdentifier应该能通过残疾证号查询残疾人', async () => {
  310. const dataSource = await IntegrationTestDatabase.getDataSource();
  311. const userService = new UserService(dataSource);
  312. const person = await userService.getDisabledPersonByIdentifier(testDisabilityId);
  313. expect(person).toBeDefined();
  314. expect(person?.id).toBe(testPerson.id);
  315. expect(person?.name).toBe('张三');
  316. });
  317. it('getUserByPersonId应该能通过person_id查询用户', async () => {
  318. const dataSource = await IntegrationTestDatabase.getDataSource();
  319. const userService = new UserService(dataSource);
  320. const user = await userService.getUserByPersonId(testPerson.id);
  321. expect(user).toBeDefined();
  322. expect(user?.id).toBe(testUser.id);
  323. expect(user?.username).toBe('talent_user');
  324. });
  325. it('getTalentUserByIdentifier应该能通过身份证号查询完整人才用户信息', async () => {
  326. const dataSource = await IntegrationTestDatabase.getDataSource();
  327. const userService = new UserService(dataSource);
  328. const user = await userService.getTalentUserByIdentifier(testIdCard);
  329. expect(user).toBeDefined();
  330. expect(user?.id).toBe(testUser.id);
  331. expect(user?.personId).toBe(testPerson.id);
  332. expect(user?.userType).toBe(UserType.TALENT);
  333. });
  334. });
  335. });