Jest is an open source test runner created by Facebook. It has a lot of great features:
- Immersive watch mode for providing near instant feedback when developing tests.
- Snapshot testing for validating features.
- Great built-in reporter for printing out test results.
Setting Up @nx/jest
Installation
Make sure to install the @nx/jest
version that matches the version of nx
in your repository. If the version numbers get out of sync, you can encounter some difficult to debug errors. You can fix Nx version mismatches with this recipe.
In any Nx workspace, you can install @nx/jest
by running the following command:
โฏ
nx add @nx/jest
This will install the correct version of @nx/jest
.
Configuring @nx/jest/plugin for both E2E and Unit Tests
While Jest is most often used for unit tests, there are cases where it can be used for e2e tests as well as unit tests within the same workspace. In this case, you can configure the @nx/jest/plugin
twice for the different cases.
1{
2 "plugins": [
3 {
4 "plugin": "@nx/jest/plugin",
5 "exclude": ["e2e/**/*"],
6 "options": {
7 "targetName": "test"
8 }
9 },
10 {
11 "plugin": "@nx/jest/plugin",
12 "include": ["e2e/**/*"],
13 "options": {
14 "targetName": "e2e-local",
15 "ciTargetName": "e2e-ci",
16 "disableJestRuntime": false
17 }
18 }
19 ]
20}
21
If you experience slowness from @nx/jest/plugin
, then set disableJestRuntime
to true
to skip creating the Jest runtime. By disabling the Jest runtime, Nx will use its own utilities to find inputs
, outputs
, and test files for Atomized targets. This can reduce computation time by as much as 80%.
Splitting E2E Tests
If Jest is used to run E2E tests, you can enable splitting the tasks by file to get improved caching, distribution, and retrying flaky tests. Enable this Atomizer feature by providing a ciTargetName
. This will create a target with that name which can be used in CI to run the tests for each file in a distributed fashion.
1{
2 "plugins": [
3 {
4 "plugin": "@nx/jest/plugin",
5 "include": ["e2e/**/*"],
6 "options": {
7 "targetName": "e2e-local",
8 "ciTargetName": "e2e-ci"
9 }
10 }
11 ]
12}
13
Customizing atomized unit/e2e tasks group name
By default, the atomized tasks group name is derived from the ciTargetName
. For example, atomized tasks for the e2e-ci
target will be grouped under the name "E2E (CI)" when displayed in Nx Cloud or nx show project <project> --web
UI. You can customize that name by explicitly providing the optional ciGroupName
plugin option as such:
1{
2 "plugins": [
3 {
4 "plugin": "@nx/jest/plugin",
5 "include": ["e2e/**/*"],
6 "options": {
7 "targetName": "e2e-local",
8 "ciTargetName": "e2e-ci",
9 "ciGroupname": "My E2E tests (CI)"
10 }
11 }
12 ]
13}
14
How @nx/jest Infers Tasks
Since Nx 18, Nx plugins can infer tasks for your projects based on the configuration of different tools. You can read more about it at the Inferred Tasks concept page.
The @nx/jest
plugin will create a task for any project that has an Jest configuration file present. Any of the following files will be recognized as an Jest configuration file:
jest.config.js
jest.config.ts
jest.config.mjs
jest.config.mts
jest.config.cjs
jest.config.cts
View Inferred Tasks
To view inferred tasks for a project, open the project details view in Nx Console or run nx show project my-project --web
in the command line.
@nx/jest Configuration
The @nx/jest/plugin
is configured in the plugins
array in nx.json
.
1{
2 "plugins": [
3 {
4 "plugin": "@nx/jest/plugin",
5 "options": {
6 "targetName": "test"
7 }
8 }
9 ]
10}
11
- The
targetName
option controls the name of the inferred Jest tasks. The default name istest
.
Using Jest
Generate a new project set up with Jest
By default, Nx will use Jest when creating applications and libraries.
โฏ
nx g @nx/web:app apps/frontend
Add Jest to a project
Run the configuration
generator
โฏ
nx g @nx/jest:configuration --project=<project-name>
Hint: You can use the
--dry-run
flag to see what will be generated.
Replacing <project-name>
with the name of the project you're wanting to add Jest too.
Testing Applications
The recommended way to run/debug Jest tests via an editor
To run Jest tests via nx use
โฏ
nx test frontend
Testing Specific Files
Using a single positional argument or the --testFile
flag will run all test files matching the regex. For more info check out the Jest documentation.
โฏ
nx test frontend HomePage.tsx
โฏ
# or
โฏ
nx test frontend --testFile HomePage.tsx
Watching for Changes
Using the --watch
flag will run the tests whenever a file changes.
โฏ
nx test frontend --watch
Snapshot Testing
Jest has support for Snapshot Testing, a tool which simplifies validating data. Check out the official Jest Documentation on Snapshot Testing.
Example of using snapshots:
1describe('SuperAwesomeFunction', () => {
2 it('should return the correct data shape', () => {
3 const actual = superAwesomeFunction();
4 expect(actual).toMatchSnapshot();
5 });
6});
7
When using snapshots, you can update them with the --updateSnapshot
flag, -u
for short.
By default, snapshots will be generated when there are not existing snapshots for the associated test.
โฏ
nx test frontend -u
Snapshot files should be checked in with your code.
Performance in CI
Typically, in CI it's recommended to use nx affected -t test --parallel=[# CPUs] -- --runInBand
for the best performance.
This is because each jest process creates a workers based on system resources, running multiple projects via nx and using jest workers will create too many process overall causing the system to run slower than desired. Using the --runInBand
flag tells jest to run in a single process.
Configurations
Jest
Primary configurations for Jest will be via the jest.config.ts
file that generated for your project. This file will extend the root jest.preset.js
file. Learn more about Jest configurations.
The root level jest.config.ts
file configures Jest multi project support. This configuration allows editor/IDE integrations to pick up individual project's configurations rather than the one at the root.
The set of Jest projects within Nx workspaces tends to change. Instead of statically defining a list in jest.config.ts
, Nx provides a utility function called getJestProjectsAsync
which retrieves a list of paths to all the Jest config files from projects using the @nx/jest:jest
executor or with tasks running the jest
command.
You can manually add Jest projects not identified by the getJestProjectsAsync
function by doing something like the following:
1import { getJestProjectsAsync } from '@nx/jest';
2
3export default async () => ({
4 projects: [
5 ...(await getJestProjectsAsync()),
6 '<rootDir>/path/to/jest.config.ts',
7 ],
8});
9
Nx
The Nx task options can be configured via the project config file or via the command line flags.
If you're using inferred tasks, or running Jest directly with the nx:run-commands
executor, you can provide the Jest args for the command you're running.
If you're using the @nx/jest:jest
executor, you can provide the options the executor accepts.
Code Coverage
Enable code coverage with the --coverage
flag or by adding it to the executor options in the project configuration file.
By default, coverage reports will be generated in the coverage/
directory under projects name. i.e. coverage/apps/frontend
. Modify this directory with the --coverageDirectory
flag. Coverage reporters can also be customized with the --coverageReporters
flag.
coverageDirectory
andcoverageReporters
are configurable via the project configuration file as well.
Global setup/teardown with nx libraries
In order to use Jest's global setup/teardown functions that reference nx libraries, you'll need to register the TS path for jest to resolve the libraries. Nx provides a helper function that you can import within your setup/teardown file.
1import { registerTsProject } from '@nx/js/src/internal';
2const cleanupRegisteredPaths = registerTsProject('./tsconfig.base.json');
3
4import { yourFancyFunction } from '@some-org/my-util-library';
5export default async function () {
6 yourFancyFunction();
7
8 // make sure to run the clean up!
9 cleanupRegisteredPaths();
10}
11
If you're using @swc/jest
and a global setup/teardown file, you have to set the noInterop: false
and use dynamic imports within the setup function:
1/* eslint-disable */
2import { readFileSync } from 'fs';
3
4// Reading the SWC compilation config and remove the "exclude"
5// for the test files to be compiled by SWC
6const { exclude: _, ...swcJestConfig } = JSON.parse(
7 readFileSync(`${__dirname}/.swcrc`, 'utf-8')
8);
9
10// disable .swcrc look-up by SWC core because we're passing in swcJestConfig ourselves.
11// If we do not disable this, SWC Core will read .swcrc and won't transform our test files due to "exclude"
12if (swcJestConfig.swcrc === undefined) {
13 swcJestConfig.swcrc = false;
14}
15
16// jest needs EsModule Interop to find the default exported function
17swcJestConfig.module.noInterop = false;
18
19export default {
20 globalSetup: '<rootDir>/src/global-setup-swc.ts',
21 transform: {
22 '^.+\\.[tj]s$': ['@swc/jest', swcJestConfig],
23 },
24 // other settings
25};
26
1import { registerTsProject } from '@nx/js/src/internal';
2const cleanupRegisteredPaths = registerTsProject('./tsconfig.base.json');
3
4export default async function () {
5 // swc will hoist all imports, and we need to make sure the register happens first
6 // so we import all nx project alias within the setup function first.
7 const { yourFancyFunction } = await import('@some-org/my-util-library');
8
9 yourFancyFunction();
10
11 // make sure to run the clean up!
12 cleanupRegisteredPaths();
13}
14