- Published on
How to test React application with Jest
- Authors
- Name
- Pavel Keyzik
- @pavelkeyzik
Image by Ferenc Almasi
Have you ever tested React application using Jest? I think it's awesome and saves you a lot of time and if you want to know why. Then continue to read this article.
Why do we have to write tests
The reason why we have to write tests is the confidence of updating this part of the code in the future. And another cool thing about it is that well-written tests are like documentation. I’m serious! I often read test cases to understand how code is working.
Do we always need them?
Probably not. If your application is just a pet project to try something or a small application without any complex logic. Then you can ignore them but even if you have a small project it’ll be better to use tests.
Have you ever tried Test-driven development (TDD)? You'll feel like it's too long to write tests first. But the reason why TDD is awesome it's making you think about API, not the actual implementation. So, you'll get what you need exactly instead of a calling function like thisIsDoingSomething()
and using it through the entire application because you don't have time to refactor.
How to write tests?
I'm going to show you the way I write unit tests but you can write however you want. Because JavaScript === Freedom.
The AAA (Arrange, Act, and Assert) pattern
The AAA pattern just tells you the structure of code inside your tests. A basic example is:
// Arange
const x = 2
const y = 4
// Act
const result = sum(x, y)
// Assert
expect(result).toBe(6)
Basic test of functionality
Let's say we have some function that takes coordinates of destinations and you want to test the shortest path to them.
describe('Get Shortest Path', () => {
it('should return a list of destination in order with shortest path', () => {
const destinations = [
{ x: 0, y: 0 },
{ x: 100, y: 100 },
{ x: 50, y: 50 },
]
const expectedResult = [
{ x: 0, y: 0 },
{ x: 50, y: 50 },
{ x: 100, y: 100 },
]
const result = getShortestPath(destinations)
expect(result).toBe(expectedResult)
})
})
Testing that React component has UI elements
When you build, for example, a<UsersList />
component, you'd expect to see a list of users, right? Then what about writing tests for this? It's so easy.
Usually, I start to think about edge cases. In our example it can be:
- We don't have anything
- We're fetching data and want to show Loading state
- We have everything that we need and can show
<UsersList />
component
Now, let's have a look at our tests and after that, you'll find all information about used functions down below.
import { render, screen } from '@testing-library/react'
import { UsersList } from 'components/UsersList'
describe('UsersList component', () => {
// Case 1. We don't have anything
it('should contain a message about empty list', () => {
render(<UsersList users={[]} status="loaded" />)
const result = screen.queryByText('No users')
expect(result).toBeInTheDocument()
})
// Case 2. Shows loading state when fetching something
it('should contain a message about loading data', () => {
render(<UsersList users={[]} status="loading" />)
const result = screen.queryByText('Loading...')
expect(result).toBeInTheDocument()
})
// Case 3. Shows data to the user
it('should contain a message about loading data', () => {
const users = [
{ id: 1, name: 'Mark' },
{ id: 2, name: 'Marie' },
]
render(<UsersList users={users} status="loaded" />)
const result = screen.queryAllByRole('listitem')
expect(result).toHaveLength(2)
expect(result[0]).toHaveTextContent('Mark')
expect(result[1]).toHaveTextContent('Marie')
})
})
- render() - takes our component and build a DOM elements
- screen - a helper to find elemens in our DOM
- screen.queryByText() - find element by text
- expect.toBeInTheDocument() - checks that element that we're searching for is in DOM
- expect.toHaveLength() - takes
.length
property of array and check this value - expect.toHaveTextContent() - takes DOM element and check
.textContent
property
You can read more about queries in @testing-library/react
here. It's definitely worth it to read this if you don't know what query to use.
And that's how our component may look like:
function UsersList({ users, status }) {
if (status === 'loading') {
return <div>Loading...</div>
}
if (!users || users.length === 0) {
return <div>No users</div>
}
return (
<ul>
{users.map((user) => (
<li key={user.id}>{user.name}</li>
))}
</ul>
)
}
Summary
Tests are great and in most cases, it's not so hard to write them. Just try to write a few of them, and you'll understand them better. I spent a lot of time understanding why and how to write them. And you know what? Practice makes perfect!