Skip to content

Latest commit

 

History

History
103 lines (61 loc) · 6.74 KB

react-testing-library.md

File metadata and controls

103 lines (61 loc) · 6.74 KB

Intro to react-testing-library

What is it?

react-testing-library is a "Simple and complete React DOM testing utilities that encourage good testing practices."

Why do we recommend this library?

There is one design principal followed by this library:

The more your tests resemble the way your software is used, the more confidence they can give you.

It encourages developers to follow black box testing approach and avoid testing implementation details. The major benefit of the black box testing approach is it supports refactoring very well. In contrast, if you ever write tests that depend on implementation details of the module under test, your tests could fail when you change the implementation details.

For example, suppose you have a React component that is implemented as a Class so that you can use the local state. When you write a test case for this component, the test case should not have direct access to the state variable of the component instance (i.e. getting/setting the state directly), because the fact that the component has a local state is an implementation detail. Later on, you may refactor the component implementation to use global state (e.g. using React Context) or change the component to a functional component with React Hooks. If your tests ever have direct access to the local state, then your test cases would fail right away when you change the implementation to remove the local state (which is bad because you haven't change the behavior of the component.)

Here is a blog article with further discussion on this topic

How to install the library in your project

npm install --save-dev react-testing-library jest-dom

How to use it?

In order to test some behavior of a React component, you typically need to follow the steps below:

  • Render the target React component into a DOM tree
  • Search the DOM tree to find out the HTML element you are interested in
  • Verify the HTML element match your expectation (e.g. the texts appearing on the screen, the CSS classes, the attributes, etc), or
  • Interact with the element by simulating some DOM event (e.g click a button)
  • Verify the DOM tree is updated according to the DOM event (i.e. verifying the event handlers are executed properly to handle the DOM event)

With these steps in mind, read the documentation of this react-testing-library and answer the questions:

  • How to render a React component with the library? Which function should you use?
  • How to search an HTML element? What can you search with?
  • How to assert that some HTML element has expected attribute/CSS-class?
  • How to fire an event to simulate user interaction with the web application?
  • If some user interaction triggers asynchronous event, how do you wait for that event to finish (e.g. how do you wait for an element to appear on the screen before you go to the next step in tests?)

Here are some of the links you can keep as a reference:

Tutorials

Testing React Workshop @ Paypal Sep 2018 by Kent C. Dodds

Tips

How to configure react-testing-library in a central place?

Typically you need the following two lines in every test case that uses react-testing-library:

// add some helpful assertions
import 'jest-dom/extend-expect'

// this is basically: afterEach(cleanup)
import 'react-testing-library/cleanup-after-each'

So it makes sense to put the setup logic in a central place to avoid duplication.

If you create your React project with create-react-app, you needs to put these two lines in src/setupTests.js. Otherwise, refer to the documentation on how to do this.

How to search an DOM element in the DOM?

Besides using the APIs provided by the dom-testing-library like getByText, getByTitle etc, you can also use the standard querySelector API. This querySelector API allows you to select DOM elements with their class, ID, attribute values, etc.

How to check the class names associated with one element?

Often you need to write test cases to check if an HTML element has the correct CSS classes.

That's easy.

Once you obtain an element from the DOM tree, you can check its CSS class names by using the className attribute, e.g. element.className.

Comparing with Enzyme

Enzyme is another popular library for testing React components. However, it exposes some API that encourages you to depend on implementation details. Here is a blog article shows you why you should avoid using those APIs such as shallow rendering. If you have to use Enzyme in your project, be aware of those bad practices!!!

Here is a tutorial that shows how to migrate some tests from Enzyme to React-Testing-Library.

Lab

Lab 1

  • Extend the React Todo List and add unit tests for your components. After you finish adding unit tests to the components, you can start from scratch again by implementing the components following Test Driven Development style.

Lab 2

  • Here is a project. The author showed how you can write unit test with a library called Enzyme. You can try to convert the unit tests to use react-testing-library.