
在现代react应用开发中,自定义hook是逻辑复用的强大工具,而react-query(或tanstack query)则极大地简化了数据获取、缓存和同步。当一个自定义hook内部包含多个usequery调用,用于从不同api端点获取数据时,如何有效地进行单元测试成为了一个关键问题。本教程将深入探讨如何使用@testing-library/react-hooks和jest来测试此类复杂场景,并纠正常见的测试误区。
考虑一个自定义Hook useTest,它通过react-query同时获取测试详情和测试状态:
// TestHook.js
import { useQuery } from "react-query";
import { getTestByUid, getTestStatusesByUid } from "./api"; // 假设API函数在此模块
export const useTest = (uid) => {
const { data: test } = useQuery("test", () => getTestByUid(uid));
const { data: testStatuses } = useQuery("statuses", () => getTestStatusesByUid(uid));
return {
test,
testStatuses
};
};最初的测试尝试可能面临以下问题:
为了解决上述问题,我们推荐以下策略:
使用jest.mock()来模拟整个API模块是更简洁、更强大的方法。它允许我们替换模块中的所有导出函数,并在每个测试用例中灵活地设置它们的行为。
// test-hook.test.js
import { renderHook } from '@testing-library/react-hooks';
import { QueryClient, QueryClientProvider } from 'react-query';
import { useTest } from './test-hook';
import * as testApi from './api'; // 导入API模块
import React from 'react';
// 在文件顶部模拟整个API模块
jest.mock('./api'); // 这将使testApi.getTestByUid和testApi.getTestStatusesByUid成为jest mock函数通过jest.mock('./api'),testApi中的所有函数都变成了Jest的模拟函数,我们可以直接使用mockResolvedValue或mockRejectedValue来控制它们的行为,而无需使用spyOn。
当一个Hook的目的是组合多个数据源时,通常应该在一个测试用例中验证所有这些数据源是否被正确获取和返回。这不仅提高了测试效率,也更好地反映了Hook的整体功能。
// test-hook.test.js (续)
// ... (之前的导入和jest.mock)
const queryClient = new QueryClient({
defaultOptions: {
queries: {
retry: false, // 在测试中禁用重试以避免不必要的等待
},
},
});
const wrapper = ({ children }) => {
return <QueryClientProvider client={queryClient}>{children}</QueryClientProvider>;
};
describe('useTestHook', () => {
it('should return test details and statuses correctly', async () => {
// 为所有API调用设置模拟返回值
testApi.getTestByUid.mockResolvedValue({ name: 'secret test' });
testApi.getTestStatusesByUid.mockResolvedValue(['in_progress', 'ready_for_approval', 'rejected']);
const { result, waitForNextUpdate } = renderHook(() => useTest('bb450409-d778-4d57-a4b8-70fcfe2087bd'), {
wrapper,
});
// 等待React Query完成数据获取
await waitForNextUpdate();
// 断言Hook返回的所有数据
expect(result.current.test).toEqual({ name: 'secret test' });
expect(result.current.testStatuses).toEqual(['in_progress', 'ready_for_approval', 'rejected']);
});
});如前所述,useQuery的data属性直接包含API的解析值。因此,在模拟API函数时,直接返回期望的数据即可,无需额外包裹:
// 错误示例:
// testApi.getTestByUid.mockResolvedValue({ data: { name: 'secret test' } });
// 导致 result.current.test 为 { data: { name: 'secret test' } }
// 正确示例:
testApi.getTestByUid.mockResolvedValue({ name: 'secret test' });
// 导致 result.current.test 为 { name: 'secret test' }// api.js
export const getTestByUid = (uid) => {
// 实际应用中会是真实的API调用,这里仅为模拟提供函数签名
return Promise.resolve({ id: uid, name: "real test data" });
};
export const getTestStatusesByUid = (uid) => {
// 实际应用中会是真实的API调用
return Promise.resolve(["active", "completed"]);
};// test-hook.js
import { useQuery } from "react-query";
import { getTestByUid, getTestStatusesByUid } from "./api";
export const useTest = (uid) => {
const { data: test } = useQuery(["test", uid], () => getTestByUid(uid)); // 推荐使用数组作为queryKey
const { data: testStatuses } = useQuery(["statuses", uid], () => getTestStatusesByUid(uid)); // 推荐使用数组作为queryKey
return {
test,
testStatuses
};
};注意: 在useQuery的queryKey中使用数组["test", uid]是更好的实践,它能确保当uid变化时,react-query能够正确地识别并重新获取数据。
// test-hook.test.js
import { renderHook } from '@testing-library/react-hooks';
import { QueryClient, QueryClientProvider } from 'react-query';
import { useTest } from './test-hook';
import * as testApi from './api';
import React from 'react';
// 在文件顶部模拟整个API模块
jest.mock('./api');
// 初始化QueryClient,并禁用重试以简化测试
const queryClient = new QueryClient({
defaultOptions: {
queries: {
retry: false,
},
},
});
// 定义一个Wrapper组件,用于为Hook提供React Query上下文
const wrapper = ({ children }) => {
return <QueryClientProvider client={queryClient}>{children}</QueryClientProvider>;
};
describe('useTestHook', () => {
it('should return test details and statuses correctly for a given UID', async () => {
const mockUid = 'bb450409-d778-4d57-a4b8-70fcfe2087bd';
const mockTestDetails = { id: mockUid, name: 'secret test' };
const mockTestStatuses = ['in_progress', 'ready_for_approval', 'rejected'];
// 设置API函数的模拟返回值
testApi.getTestByUid.mockResolvedValue(mockTestDetails);
testApi.getTestStatusesByUid.mockResolvedValue(mockTestStatuses);
// 渲染自定义Hook
const { result, waitForNextUpdate } = renderHook(() => useTest(mockUid), {
wrapper,
});
// 等待React Query完成数据获取(即promise解析)
await waitForNextUpdate();
// 断言Hook的返回值是否符合预期
expect(result.current.test).toEqual(mockTestDetails);
expect(result.current.testStatuses).toEqual(mockTestStatuses);
// 也可以验证API函数是否被正确调用
expect(testApi.getTestByUid).toHaveBeenCalledTimes(1);
expect(testApi.getTestByUid).toHaveBeenCalledWith(mockUid);
expect(testApi.getTestStatusesByUid).toHaveBeenCalledTimes(1);
expect(testApi.getTestStatusesByUid).toHaveBeenCalledWith(mockUid);
});
it('should handle API errors gracefully (example)', async () => {
const mockUid = 'error-uid';
const errorMessage = 'Failed to fetch test details';
// 模拟一个API调用失败
testApi.getTestByUid.mockRejectedValue(new Error(errorMessage));
testApi.getTestStatusesByUid.mockResolvedValue(['available']); // 另一个API可以成功
const { result, waitForNextUpdate } = renderHook(() => useTest(mockUid), {
wrapper,
});
// 等待数据更新,这里可能需要等待错误状态
await waitForNextUpdate();
// 根据useQuery的错误处理逻辑,可能需要检查result.current.test是否为undefined或检查error对象
// 注意:useQuery的错误会存储在各自的error属性中,这里简化为检查data是否为undefined
expect(result.current.test).toBeUndefined();
expect(result.current.testStatuses).toEqual(['available']); // 另一个API数据依然存在
// 实际项目中,你可能还会断言 error 对象
// expect(result.current.error).toBeDefined();
});
});通过采用模块级模拟、整合测试用例以及确保正确的Mock数据结构,我们可以高效且准确地测试包含多个react-query调用的自定义React Hook。这种方法不仅提高了测试的可靠性,也使测试代码更易于维护和理解,从而为构建健壮的React应用程序奠定基础。
以上就是在React自定义Hook中高效测试多个React Query请求的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号