userManagement.integration.test.tsx 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. import { describe, it, expect, vi, beforeEach } from 'vitest';
  2. import { render, screen, fireEvent, waitFor } from '@testing-library/react';
  3. import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
  4. import { UserManagement } from '../../src/components/UserManagement';
  5. import { userClient, userClientManager } from '../../src/api/userClient';
  6. // 完整的mock响应对象
  7. const createMockResponse = (status: number, data?: any) => ({
  8. status,
  9. ok: status >= 200 && status < 300,
  10. body: null,
  11. bodyUsed: false,
  12. statusText: status === 200 ? 'OK' : status === 201 ? 'Created' : status === 204 ? 'No Content' : 'Error',
  13. headers: new Headers(),
  14. url: '',
  15. redirected: false,
  16. type: 'basic' as ResponseType,
  17. json: async () => data || {},
  18. text: async () => '',
  19. blob: async () => new Blob(),
  20. arrayBuffer: async () => new ArrayBuffer(0),
  21. formData: async () => new FormData(),
  22. clone: function() { return this; }
  23. });
  24. // Mock API client
  25. vi.mock('../../src/api/userClient', () => {
  26. const mockUserClient = {
  27. index: {
  28. $get: vi.fn(() => Promise.resolve({ status: 200, body: null })),
  29. $post: vi.fn(() => Promise.resolve({ status: 201, body: null })),
  30. },
  31. ':id': {
  32. $put: vi.fn(() => Promise.resolve({ status: 200, body: null })),
  33. $delete: vi.fn(() => Promise.resolve({ status: 204, body: null })),
  34. },
  35. };
  36. const mockUserClientManager = {
  37. get: vi.fn(() => mockUserClient),
  38. };
  39. return {
  40. userClientManager: mockUserClientManager,
  41. userClient: mockUserClient,
  42. };
  43. });
  44. // Mock toast
  45. vi.mock('sonner', () => ({
  46. toast: {
  47. success: vi.fn(() => {}),
  48. error: vi.fn(() => {}),
  49. },
  50. }));
  51. const createTestQueryClient = () =>
  52. new QueryClient({
  53. defaultOptions: {
  54. queries: {
  55. retry: false,
  56. },
  57. },
  58. });
  59. const renderWithProviders = (component: React.ReactElement) => {
  60. const queryClient = createTestQueryClient();
  61. return render(
  62. <QueryClientProvider client={queryClient}>
  63. {component as any}
  64. </QueryClientProvider>
  65. );
  66. };
  67. describe('用户管理集成测试', () => {
  68. beforeEach(() => {
  69. vi.clearAllMocks();
  70. });
  71. it('应该完成完整的用户CRUD流程', async () => {
  72. const mockUsers = {
  73. data: [
  74. {
  75. id: 1,
  76. username: 'existinguser',
  77. nickname: 'Existing User',
  78. email: 'existing@example.com',
  79. phone: '1234567890',
  80. name: 'Existing Name',
  81. isDisabled: 0,
  82. createdAt: '2024-01-01T00:00:00Z',
  83. roles: [{ id: 1, name: 'admin' }],
  84. avatarFile: null,
  85. },
  86. ],
  87. pagination: {
  88. total: 1,
  89. page: 1,
  90. pageSize: 10,
  91. },
  92. };
  93. const { toast } = await import('sonner');
  94. // Mock initial user list
  95. (userClientManager.get().index.$get as any).mockResolvedValue(createMockResponse(200, mockUsers));
  96. renderWithProviders(<UserManagement />);
  97. // Wait for initial data to load
  98. await waitFor(() => {
  99. expect(screen.getByText('existinguser')).toBeInTheDocument();
  100. });
  101. // Test create user
  102. const createButton = screen.getByTestId('create-user-button');
  103. fireEvent.click(createButton);
  104. // Fill create form
  105. const usernameInput = screen.getByPlaceholderText('请输入用户名');
  106. const passwordInput = screen.getByPlaceholderText('请输入密码');
  107. const emailInput = screen.getByPlaceholderText('请输入邮箱');
  108. fireEvent.change(usernameInput, { target: { value: 'newuser' } });
  109. fireEvent.change(passwordInput, { target: { value: 'password123' } });
  110. fireEvent.change(emailInput, { target: { value: 'new@example.com' } });
  111. // Mock successful creation
  112. (userClient.index.$post as any).mockResolvedValue(createMockResponse(201, { id: 2, username: 'newuser' }));
  113. const submitButton = screen.getByTestId('create-user-submit-button');
  114. fireEvent.click(submitButton);
  115. await waitFor(() => {
  116. expect(userClient.index.$post).toHaveBeenCalledWith({
  117. json: {
  118. username: 'newuser',
  119. password: 'password123',
  120. email: 'new@example.com',
  121. nickname: null,
  122. phone: null,
  123. name: null,
  124. isDisabled: 0,
  125. },
  126. });
  127. expect(toast.success).toHaveBeenCalledWith('用户创建成功');
  128. });
  129. // Test edit user
  130. const editButtons = screen.getAllByRole('button', { name: '编辑用户' });
  131. fireEvent.click(editButtons[0]);
  132. // Verify edit form is populated
  133. await waitFor(() => {
  134. expect(screen.getByDisplayValue('existinguser')).toBeInTheDocument();
  135. });
  136. // Update user
  137. const updateUsernameInput = screen.getByDisplayValue('existinguser');
  138. fireEvent.change(updateUsernameInput, { target: { value: 'updateduser' } });
  139. // Mock successful update
  140. (userClient[':id']['$put'] as any).mockResolvedValue(createMockResponse(200));
  141. const updateButton = screen.getByText('更新用户');
  142. fireEvent.click(updateButton);
  143. await waitFor(() => {
  144. expect(userClient[':id']['$put']).toHaveBeenCalledWith({
  145. param: { id: 1 },
  146. json: {
  147. username: 'updateduser',
  148. nickname: 'Existing User',
  149. email: 'existing@example.com',
  150. phone: '1234567890',
  151. name: 'Existing Name',
  152. password: undefined,
  153. avatarFileId: undefined,
  154. isDisabled: 0,
  155. },
  156. });
  157. expect(toast.success).toHaveBeenCalledWith('用户更新成功');
  158. });
  159. // Test delete user
  160. const deleteButtons = screen.getAllByRole('button', { name: '删除用户' });
  161. fireEvent.click(deleteButtons[0]);
  162. // Confirm deletion
  163. expect(screen.getByText('确认删除')).toBeInTheDocument();
  164. // Mock successful deletion
  165. (userClient[':id']['$delete'] as any).mockResolvedValue({
  166. status: 204,
  167. });
  168. const confirmDeleteButton = screen.getByText('删除');
  169. fireEvent.click(confirmDeleteButton);
  170. await waitFor(() => {
  171. expect(userClient[':id']['$delete']).toHaveBeenCalledWith({
  172. param: { id: 1 },
  173. });
  174. expect(toast.success).toHaveBeenCalledWith('用户删除成功');
  175. });
  176. });
  177. it('应该优雅处理API错误', async () => {
  178. const { userClient } = await import('../../src/api/userClient');
  179. const { toast } = await import('sonner');
  180. // Mock API error
  181. (userClientManager.get().index.$get as any).mockRejectedValue(new Error('API Error'));
  182. renderWithProviders(<UserManagement />);
  183. // Should handle error without crashing
  184. await waitFor(() => {
  185. expect(screen.getByText('用户管理')).toBeInTheDocument();
  186. });
  187. // Test create user error
  188. const createButton = screen.getByText('创建用户');
  189. fireEvent.click(createButton);
  190. const usernameInput = screen.getByPlaceholderText('请输入用户名');
  191. const passwordInput = screen.getByPlaceholderText('请输入密码');
  192. fireEvent.change(usernameInput, { target: { value: 'testuser' } });
  193. fireEvent.change(passwordInput, { target: { value: 'password' } });
  194. // Mock creation error
  195. (userClient.index.$post as any).mockRejectedValue(new Error('Creation failed'));
  196. const submitButton = screen.getByTestId('create-user-submit-button');
  197. fireEvent.click(submitButton);
  198. await waitFor(() => {
  199. expect(toast.error).toHaveBeenCalledWith('创建失败,请重试');
  200. });
  201. });
  202. it('应该处理搜索和过滤器集成', async () => {
  203. const { userClient } = await import('../../src/api/userClient');
  204. const mockUsers = {
  205. data: [],
  206. pagination: { total: 0, page: 1, pageSize: 10 },
  207. };
  208. (userClientManager.get().index.$get as any).mockResolvedValue(createMockResponse(200, mockUsers));
  209. renderWithProviders(<UserManagement />);
  210. // Test search
  211. const searchInput = screen.getByPlaceholderText('搜索用户名、昵称或邮箱...');
  212. fireEvent.change(searchInput, { target: { value: 'searchterm' } });
  213. await waitFor(() => {
  214. expect(userClientManager.get().index.$get).toHaveBeenCalledWith({
  215. query: {
  216. page: 1,
  217. pageSize: 10,
  218. keyword: 'searchterm',
  219. filters: undefined,
  220. },
  221. });
  222. });
  223. // Test filter
  224. const filterButton = screen.getByTestId('advanced-filter-button');
  225. fireEvent.click(filterButton);
  226. // Wait for filter panel to appear
  227. await waitFor(() => {
  228. expect(screen.getByTestId('status-filter-trigger')).toBeInTheDocument();
  229. }, { timeout: 2000 });
  230. // Apply status filter
  231. const statusSelect = screen.getByTestId('status-filter-trigger');
  232. fireEvent.click(statusSelect);
  233. const enabledOption = screen.getByTestId('status-enabled-option');
  234. fireEvent.click(enabledOption);
  235. await waitFor(() => {
  236. expect(userClientManager.get().index.$get).toHaveBeenCalledWith({
  237. query: {
  238. page: 1,
  239. pageSize: 10,
  240. keyword: 'searchterm',
  241. filters: expect.stringContaining('isDisabled'),
  242. },
  243. });
  244. });
  245. });
  246. });