An open API service indexing awesome lists of open source software.

https://github.com/flexdinesh/testing-hooks

Code examples to test custom React hooks with Enzyme
https://github.com/flexdinesh/testing-hooks

Last synced: 10 months ago
JSON representation

Code examples to test custom React hooks with Enzyme

Awesome Lists containing this project

README

          

# Testing-Hooks

This repo lays down different strategies to test React custom hooks and components that use hooks.

## Test Strategies

There are broadly two strategies to test our React codebase.

1. Testing user observable behaviour
2. Testing implementation details

### Testing user observable behaviour

Testing user observable behaviour means writing tests against components that test

- how the component is rendered
- how the component is re-rendered when user interacts with the DOM
- how props/state control what is rendered

Consider the following component - `Greet`

```jsx
function Greet({ user = 'User' }) {
const [name, setName] = React.useState(user);

return

setName('Pinocchio')}>Hello, {name}!
;
}
```

Testing the user observable behaviour in `Greet` would mean

- test if `Greet` is rendered without crashing
- test if `Hello, User!` is rendered when user prop is not passed
- test if `Hello, Bruce!` is rendered when `Bruce` is passed as value to `user` prop
- test if the text changes to `Hello, Pinocchio!` when the user clicks on the element

### Testing implementation details

Testing implementation details means writing tests against state logic that test

- how the state is initialized with default/prop values
- how the state changes when handlers are invoked

Consider the same component - `Greet`

```jsx
function Greet({ user = 'User' }) {
const [name, setName] = React.useState(user);

return

setName('Pinocchio')}>Hello, {name}!
;
}
```

Testing implementation details in `Greet` would mean

- test if `name` is set to default value `User` when user prop is not passed to `Greet`
- test if `name` is set to prop value when user prop is passed to `Greet`
- test if `name` is updated when `setName` is invoked

## Test custom hooks with Enzyme

_Note: Please make sure your React version is `^16.8.5`. Hooks won't re-render components with enzyme shallow render in previous versions and the React team fixed it in this release. If your React version is below that, you might have to use enzyme mount and `.update()` your wrapper after each change to test the re-render._

Testing implementation details might seem unnecessary and might even be considered as a bad practice when you are writing tests against components that contains presentational (UI) logic and render elements to the DOM. But **custom hooks** contain only **state logic** and it is imperative that we test the implementation details thoroughly so we know exactly how our custom hook will behave within a component.

Let's write a custom hook to update and validate a form field.

```js
/* useFormField.js */

import React from 'react';

function useFormField(initialVal = '') {
const [val, setVal] = React.useState(initialVal);
const [isValid, setValid] = React.useState(true);

function onChange(e) {
setVal(e.target.value);

if (!e.target.value) {
setValid(false);
} else if (!isValid) setValid(true);
}

return [val, onChange, isValid];
}

export default useFormField;
```

**As great as custom hooks are in abstracting away re-usable logic in our code, they do have one limitation. Even though they are just JavaScript functions they will work only inside React components. You cannot just invoke them and write tests against what a hook returns. You have to wrap them inside a React component and test the values that it returns.**

- custom hooks cannot be tested like JavaScript functions
- custom hooks should be wrapped inside a React component to test its behaviour

Thanks to the composibility of hooks, we could pass a hook as a prop to a component and everything will work exactly as how it's supposed to work. We can write a wrapper component to render and test our hook.

```jsx
/* useFormField.test.js */

function HookWrapper(props) {
const hook = props.hook ? props.hook() : undefined;
return

;
}
```

Now we can access the hook like a JavaScript object and test its behaviour.

```jsx
/* useFormField.test.js */

import React from 'react';
import { shallow } from 'enzyme';
import useFormField from './useFormField';

function HookWrapper(props) {
const hook = props.hook ? props.hook() : undefined;
return

;
}

it('should set init value', () => {
let wrapper = shallow( useFormField('')} />);

let { hook } = wrapper.find('div').props();
let [val, onChange, isValid] = hook;
expect(val).toEqual('');

wrapper = shallow( useFormField('marco')} />);

// destructuring objects - {} should be inside brackets - () to avoid syntax error
({ hook } = wrapper.find('div').props());
[val, onChange, isValid] = hook;
expect(val).toEqual('marco');
});
```

The full test suite for `useFormField` custom hook will look like this.

```jsx
/* useFormField.test.js */

import React from 'react';
import { shallow } from 'enzyme';
import useFormField from './useFormField';

function HookWrapper(props) {
const hook = props.hook ? props.hook() : undefined;
return

;
}

describe('useFormField', () => {
it('should render', () => {
let wrapper = shallow();

expect(wrapper.exists()).toBeTruthy();
});

it('should set init value', () => {
let wrapper = shallow( useFormField('')} />);

let { hook } = wrapper.find('div').props();
let [val, onChange, isValid] = hook;
expect(val).toEqual('');

wrapper = shallow( useFormField('marco')} />);

// destructuring objects - {} should be inside brackets - () to avoid syntax error
({ hook } = wrapper.find('div').props());
[val, onChange, isValid] = hook;
expect(val).toEqual('marco');
});

it('should set the right val value', () => {
let wrapper = shallow( useFormField('marco')} />);

let { hook } = wrapper.find('div').props();
let [val, onChange, isValid] = hook;
expect(val).toEqual('marco');

onChange({ target: { value: 'polo' } });

({ hook } = wrapper.find('div').props());
[val, onChange, isValid] = hook;
expect(val).toEqual('polo');
});

it('should set the right isValid value', () => {
let wrapper = shallow( useFormField('marco')} />);

let { hook } = wrapper.find('div').props();
let [val, onChange, isValid] = hook;
expect(val).toEqual('marco');
expect(isValid).toEqual(true);

onChange({ target: { value: 'polo' } });

({ hook } = wrapper.find('div').props());
[val, onChange, isValid] = hook;
expect(val).toEqual('polo');
expect(isValid).toEqual(true);

onChange({ target: { value: '' } });

({ hook } = wrapper.find('div').props());
[val, onChange, isValid] = hook;
expect(val).toEqual('');
expect(isValid).toEqual(false);
});
});
```

Rendering the custom hook and accessing it as a prop should give us full access to its return values.

Happy testing!

## License

MIT © Dinesh Pandiyan