One area I’ve always found tricky when writing tests for React is where Higher Order Components are involved. I’ve found this complicates the test setup process. There are ways around this, which may or may not be possible depending on many factors. Sometimes you work on third party code that won’t accept ‘dramatic’ refactors just to scratch your own testing itches.
One example of where this problem might occur (particularly if you don’t read the docs!!) is with React Router, specifically when using withRouter
.
const MyComponent = ({ history }) => { ... });
export default withRouter(MyComponent);
In this example I have MyComponent
which wants to history.push('/some/location')
when the user completes some action.
I’d like to test that this process takes place as expected.
Here was my first attempt. This way works, but there are some drawbacks:
import { createMemoryHistory } from 'history';
import { Router } from 'react-router-dom';
// ...
it('should redirect when on the happy path', () => {
const history = createMemoryHistory({
initialEntries: ['/starting/point']
});
const { getByLabelText } = render(
<Router history={history}>
<Provider>
<MyComponent />
</Provider>
</Router>
);
expect(history.location.pathname).toEqual('/starting/point');
const myInput = getByLabelText('Some label text');
const value = { target: { value: 'a value here' } };
fireEvent.change(myInput, value);
fireEvent.keyPress(myInput, { key: 'Enter', code: 13, charCode: 13 });
expect(history.location.pathname).toEqual(
'/path/when/redirected'
);
});
This works.
There are a couple of drawbacks to this:
- It involves a couple of extra
import
s. - Behind the scenes,
withRouter
is still used (afaik) but by wrapping in another Router
, we can override the history
prop.
As this particular approach is so common, React Router gives us an alternative / preferable way to test this workflow – WrappedComponent
.
it('should redirect when on the happy path', () => {
const history = { push: jest.fn() };
const { getByLabelText } = render(
<Provider>
<MyComponent.WrappedComponent history={history} />
</Provider>
);
const myInput = getByLabelText('Some label text');
const value = { target: { value: 'a value here' } };
fireEvent.change(myInput, value);
fireEvent.keyPress(myInput, { key: 'Enter', code: 13, charCode: 13 });
expect(history.push).toHaveBeenCalledWith(
'/path/when/redirected'
);
expect(history.push).toHaveBeenCalledTimes(1);
});
The necessary actions to ‘run’ this test are unchanged. However, there are fewer lines as we can make use of the existing constructs provided by React Router to aid our testing workflow.
This may have been extremely obvious to you.
I can’t remember if WrappedComponent
has always been available and I have overlooked it, or it is something new since I last had to do testing with a project using React Router.
Either way, it’s time I refreshed my knowledge of the documentation. And hopefully this helps someone else when testing withRouter
at some point in the future.