如何测试在 redux 操作中抛出的错误
How to test errors throwing in redux actions
在一般的函数中我可以这样做:
expect(() => foo()).toThrow();
但是当我尝试使用 redux 操作执行此操作时,我得到
UnhandledPromiseRejectionWarning and test result: Expected the function to throw an error matching: "Unauthorized" But it didn't throw anything.
动作示例:
const someAction = id => async dispatch => {
try {
const { data } = await apiCall(id);
dispatch({ type: ACTION_TYPE, payload: data });
} catch (err) {
throw new Error(err);
}
};
测试示例:
test('should handle errors', () => {
const errMsg = 'Unauthorized';
apiSpy.mockImplementationOnce(() => Promise.reject(errMsg));
expect(() => store.dispatch(someAction(id))).toThrow(errMsg);
});
我觉得我在这里错过了什么。
解决方法如下:
actionCreator.ts
:
import { apiCall } from './apiCall';
export const ACTION_TYPE = 'ACTION_TYPE';
export const someAction = id => async dispatch => {
try {
const { data } = await apiCall(id);
dispatch({ type: ACTION_TYPE, payload: data });
} catch (err) {
throw new Error(err);
}
};
apiCall.ts
:
export const apiCall = async (id: string) => ({ data: id });
actionCreator.spec.ts
:
import createMockStore from 'redux-mock-store';
import thunk, { ThunkDispatch } from 'redux-thunk';
import { someAction } from './actionCreator';
import { AnyAction } from 'redux';
import * as API from './apiCall';
const mws = [thunk];
const mockStore = createMockStore<{}, ThunkDispatch<{}, any, AnyAction>>(mws);
describe('someAction', () => {
afterEach(() => {
jest.restoreAllMocks();
});
test('should handle errors', async () => {
const errMsg = 'Unauthorized';
const apiSpy = jest.spyOn(API, 'apiCall').mockRejectedValueOnce(errMsg);
const id = '1';
const store = mockStore({});
await expect(store.dispatch(someAction(id))).rejects.toThrow(errMsg);
expect(apiSpy).toBeCalledWith(id);
});
test('should dispatch action correctly', () => {
expect.assertions(2);
const apiSpy = jest.spyOn(API, 'apiCall').mockResolvedValueOnce({ data: 'mocked data' });
const id = '1';
const store = mockStore({});
return store.dispatch(someAction(id)).then(() => {
expect(store.getActions()).toEqual([{ type: 'ACTION_TYPE', payload: 'mocked data' }]);
expect(apiSpy).toBeCalledWith(id);
});
});
});
包含覆盖率报告的单元测试结果:
PASS src/Whosebug/58396438/actionCreator.spec.ts (6.631s)
someAction
✓ should handle errors (8ms)
✓ should dispatch action correctly (2ms)
------------------|----------|----------|----------|----------|-------------------|
File | % Stmts | % Branch | % Funcs | % Lines | Uncovered Line #s |
------------------|----------|----------|----------|----------|-------------------|
All files | 84.62 | 100 | 60 | 100 | |
actionCreator.ts | 100 | 100 | 100 | 100 | |
apiCall.ts | 50 | 100 | 0 | 100 | |
------------------|----------|----------|----------|----------|-------------------|
Test Suites: 1 passed, 1 total
Tests: 2 passed, 2 total
Snapshots: 0 total
Time: 7.747s
源代码:https://github.com/mrdulin/jest-codelab/tree/master/src/Whosebug/58396438
在一般的函数中我可以这样做:
expect(() => foo()).toThrow();
但是当我尝试使用 redux 操作执行此操作时,我得到
UnhandledPromiseRejectionWarning and test result: Expected the function to throw an error matching: "Unauthorized" But it didn't throw anything.
动作示例:
const someAction = id => async dispatch => {
try {
const { data } = await apiCall(id);
dispatch({ type: ACTION_TYPE, payload: data });
} catch (err) {
throw new Error(err);
}
};
测试示例:
test('should handle errors', () => {
const errMsg = 'Unauthorized';
apiSpy.mockImplementationOnce(() => Promise.reject(errMsg));
expect(() => store.dispatch(someAction(id))).toThrow(errMsg);
});
我觉得我在这里错过了什么。
解决方法如下:
actionCreator.ts
:
import { apiCall } from './apiCall';
export const ACTION_TYPE = 'ACTION_TYPE';
export const someAction = id => async dispatch => {
try {
const { data } = await apiCall(id);
dispatch({ type: ACTION_TYPE, payload: data });
} catch (err) {
throw new Error(err);
}
};
apiCall.ts
:
export const apiCall = async (id: string) => ({ data: id });
actionCreator.spec.ts
:
import createMockStore from 'redux-mock-store';
import thunk, { ThunkDispatch } from 'redux-thunk';
import { someAction } from './actionCreator';
import { AnyAction } from 'redux';
import * as API from './apiCall';
const mws = [thunk];
const mockStore = createMockStore<{}, ThunkDispatch<{}, any, AnyAction>>(mws);
describe('someAction', () => {
afterEach(() => {
jest.restoreAllMocks();
});
test('should handle errors', async () => {
const errMsg = 'Unauthorized';
const apiSpy = jest.spyOn(API, 'apiCall').mockRejectedValueOnce(errMsg);
const id = '1';
const store = mockStore({});
await expect(store.dispatch(someAction(id))).rejects.toThrow(errMsg);
expect(apiSpy).toBeCalledWith(id);
});
test('should dispatch action correctly', () => {
expect.assertions(2);
const apiSpy = jest.spyOn(API, 'apiCall').mockResolvedValueOnce({ data: 'mocked data' });
const id = '1';
const store = mockStore({});
return store.dispatch(someAction(id)).then(() => {
expect(store.getActions()).toEqual([{ type: 'ACTION_TYPE', payload: 'mocked data' }]);
expect(apiSpy).toBeCalledWith(id);
});
});
});
包含覆盖率报告的单元测试结果:
PASS src/Whosebug/58396438/actionCreator.spec.ts (6.631s)
someAction
✓ should handle errors (8ms)
✓ should dispatch action correctly (2ms)
------------------|----------|----------|----------|----------|-------------------|
File | % Stmts | % Branch | % Funcs | % Lines | Uncovered Line #s |
------------------|----------|----------|----------|----------|-------------------|
All files | 84.62 | 100 | 60 | 100 | |
actionCreator.ts | 100 | 100 | 100 | 100 | |
apiCall.ts | 50 | 100 | 0 | 100 | |
------------------|----------|----------|----------|----------|-------------------|
Test Suites: 1 passed, 1 total
Tests: 2 passed, 2 total
Snapshots: 0 total
Time: 7.747s
源代码:https://github.com/mrdulin/jest-codelab/tree/master/src/Whosebug/58396438