userManagement.integration.test.tsx 8.8 KB

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