Edit this page
Learn how to set up and configure the jest-expo library to write unit and snapshot tests for a project with Jest.
Jest is the most widely used unit and snapshot JavaScript testing framework. In this guide, you will learn how to set up Jest in your project, write a unit test, write a snapshot test, and best practices for structuring your tests when using Jest with React Native.
You will also use the jest-expo
library, which is a Jest preset that mocks the native part of the Expo SDK and handles most of the configuration required for your Expo project.
If you have created your project using the default Expo template, you can skip this section. The jest-expo
and other required dev dependencies are already installed and configured.
jest-expo
If you have created your project using different template, follow the instructions below to install and configure jest-expo
in your project.
1
Install jest-expo
and other required dev dependencies in your project. Run the following command from your project's root directory:
-
npx expo install jest-expo jest @types/jest -- --save-dev
# For Bun and Yarn
-
npx expo install jest-expo jest @types/jest -- --dev
-
npx expo install jest-expo jest @types/jest "--" --save-dev
# For bun and yarn
-
npx expo install jest-expo jest @types/jest "--" --dev
Note: If your project is not using TypeScript, you can skip installing
@types/jest
.
2
Open package.json, add a script for running tests, and add the preset for using the base configuration from jest-expo
:
{
"scripts": {
"test": "jest --watchAll"
%%placeholder-start%%... %%placeholder-end%%},
"jest": {
"preset": "jest-expo"
}
}
transformIgnorePatterns
You can transpile node modules your project uses by configuring transformIgnorePatterns
in your package.json. This property takes a regex pattern as its value:
"jest": {
"preset": "jest-expo",
"transformIgnorePatterns": [
"node_modules/(?!((jest-)?react-native|@react-native(-community)?)|expo(nent)?|@expo(nent)?/.*|@expo-google-fonts/.*|react-navigation|@react-navigation/.*|@sentry/react-native|native-base|react-native-svg)"
]
}
"jest": {
"preset": "jest-expo",
"transformIgnorePatterns": [
"node_modules/(?!(?:.pnpm/)?((jest-)?react-native|@react-native(-community)?|expo(nent)?|@expo(nent)?/.*|@expo-google-fonts/.*|react-navigation|@react-navigation/.*|@sentry/react-native|native-base|react-native-svg))"
]
}
Jest has many configuration options, but the above configuration should cover most of your needs. However, you can always add to this pattern list. For more details, see Configuring Jest.
The React Native Testing Library (@testing-library/react-native
) is a lightweight solution for testing React Native components. It provides utility functions and works with Jest.
To install it, run the following command:
-
npx expo install @testing-library/react-native -- --save-dev
# For Bun and Yarn
-
npx expo install @testing-library/react-native -- --dev
-
npx expo install @testing-library/react-native "--" --save-dev
# For Bun and Yarn
-
npx expo install @testing-library/react-native "--" --dev
Deprecated: If you are using the default Expo template, after installing this library, you can uninstall thereact-test-renderer
and@types/react-test-renderer
from your project's dev dependencies. Thereact-test-renderer
has been deprecated and will no longer be maintained in the future. See React's documentation for more information.
A unit test checks the smallest unit of code, usually a function. To write your first unit test, take a look at the following example:
1
Inside the app directory of your project, create a new file called index.tsx, and the following code to render a simple component:
import { PropsWithChildren } from 'react';
import { StyleSheet, Text, View } from 'react-native';
export const CustomText = ({ children }: PropsWithChildren) => <Text>{children}</Text>;
export default function HomeScreen() {
return (
<View style={styles.container}>
<CustomText>Welcome!</CustomText>
</View>
);
}
const styles = StyleSheet.create({
container: {
flex: 1,
backgroundColor: '#fff',
alignItems: 'center',
justifyContent: 'center',
},
});
2
Create a __tests__ directory at the root of your project's directory. If this directory already exists in your project, use that. Then, create a new file called HomeScreen-test.tsx. The jest-expo
preset customizes the Jest configuration to also identify files with -test.ts|tsx extensions as tests.
Add the following example code in HomeScreen-test.tsx:
import { render } from '@testing-library/react-native';
import HomeScreen, { CustomText } from '@/app/index';
describe('<HomeScreen />', () => {
test('Text renders correctly on HomeScreen', () => {
const { getByText } = render(<HomeScreen />);
getByText('Welcome!');
});
});
In the above example, the getByText
query helps your tests find relevant element in your app's user interface and make assertion whether or not the certain element exists. The React Native Testing Library provides this query, and each query variant differs in its return type. For more examples and detailed API information, see the React Native Testing Library's Queries API reference.
3
Run the following command in a terminal window to execute the test:
-
npm run test
You will see one test being passed.
Organizing your test files is important to make them easier to maintain. A common pattern is creating a __tests__ directory and putting all your tests inside.
An example structure of tests next to the components directory is shown below:
__tests__
ThemedText-test.tsx
components
ThemedText.tsx
ThemedView.tsx
Alternatively, you can have multiple __tests__ sub-directories for different areas of your project. For example, create a separate test directory for components, and so on:
components
ThemedText.tsx
__tests__
ThemedText-test.tsx
utils
index.tsx
__tests__
index-test.tsx
It's all about preferences, and it is up to you to decide how you want to organize your project directory.
A snapshot test is used to make sure that UI stays consistent, especially when a project is working with global styles that are potentially shared across components.
To add a snapshot test for <HomeScreen />
, add the following code snippet in the describe()
in HomeScreen-test.tsx:
describe('<HomeScreen />', () => {
%%placeholder-start%%... %%placeholder-end%%
test('CustomText renders correctly', () => {
const tree = render(<CustomText>Some text</CustomText>).toJSON();
expect(tree).toMatchSnapshot();
});
});
Run npm run test
command, and you will see a snapshot created inside __tests__\__snapshots__ directory, and two tests passed.
Code coverage reports can help you understand how much of your code is tested. To see the code coverage report in your project using the HTML format, in package.json, under jest
, set the collectCoverage
to true and use collectCoverageFrom
to specify a list of files to ignore when collecting the coverage.
"jest": {
...
"collectCoverage": true,
"collectCoverageFrom": [
"**/*.{ts,tsx,js,jsx}",
"!**/coverage/**",
"!**/node_modules/**",
"!**/babel.config.js",
"!**/expo-env.d.ts",
"!**/.expo/**"
]
}
Run npm run test
. You will see a coverage directory created in your project. Find the lcov-report/index.html and open it in a browser to see the coverage report.
Usually, we don't recommend uploading index.html file to git. Add
coverage/**/*
in the .gitignore file to prevent it from being tracked.
You can also use different flows to run your tests. Below are a few example scripts that you can try:
"scripts": {
"test": "jest --watch --coverage=false --changedSince=origin/main",
"testDebug": "jest -o --watch --coverage=false",
"testFinal": "jest",
"updateSnapshots": "jest -u --coverage=false"
%%placeholder-start%%... %%placeholder-end%%
}
For more information, see CLI Options in Jest documentation.
See React Native Testing Library documentation, which provides testing utilities and encourages good testing practices and work with Jest.
Learn how to create integration tests for your app when using Expo Router.
Learn how to set up and run E2E tests on EAS Build with Maestro.