How to Write Tests Using the Node.js Test Runner and mongodb-memory-server

I recently migrated some tests from Jest to the Node.js test runner in two of my projects that use MongoDB. In one of those projects, test runtime was reduced from 107 seconds to 25 seconds (screenshot below). In the other project, test runtime was r...

Feb 13, 2025 - 21:04
 0
How to Write Tests Using the Node.js Test Runner and mongodb-memory-server

I recently migrated some tests from Jest to the Node.js test runner in two of my projects that use MongoDB. In one of those projects, test runtime was reduced from 107 seconds to 25 seconds (screenshot below). In the other project, test runtime was reduced by about 66%.

76% reduction in time taken to run tests in Jest vs Node.js test runner

I decided to share with you how I was able to implement this. I think you’ll find it helpful, as it’s more cost-effective (in terms of reducing money spent on running tests in CI/CD), and it also improves your developer experience.

Table of Contents

Prerequisites

To follow along with this guide, you should have experience working with Node.js, MongoDB, and Mongoose (or any other MongoDB object data mapper). You should also have Node.js (at least v20.18.2) and MongoDB installed on your computer.

The Node.js Test Runner

The Node.js test runner was introduced as an experimental feature in version 18 of Node.js. It became fully available in version 20. It gives you the ability to:

  1. Run tests

  2. Report test results

  3. Report test coverage (still experimental at version 23)

It’s a good idea to use the in-built test runner when writing tests in Node.js because it means that you have to use fewer external dependencies. You don’t need to install an external library (and its peer dependencies) to run tests.

The built-in best runner is also faster. Based on my experience using it on two projects (which formerly used Jest), I saw improvements of at least a 66% reduction in the time taken to run tests completely.

And unlike other testing frameworks or libraries, the Node.js test runner was built specifically for Node.js projects. It doesn’t try to accommodate the specifics of other programming environments like the browser. The specifics of Node.js are its main and only priority.

MongoDB In-Memory Server

For tests that involve making requests to a database, some developers prefer to mock the requests to avoid making requests to a real database. They do this because making a request to a real database requires a lot of setting up which can cost time and resources.

Writing and fetching data using a real database is slower compared to writing and fetching data from memory. When running automated tests, using a real MongoDB server will be slower than using an in-memory database server, and that is where mongodb-memory-server becomes useful.

Comparison between memory and database communication with CPU

According to its documentation, mongodb-memory-server creates and starts a real MongoDB server programmatically from within Node.js, but uses an in-memory database by default. It also allows you to connect to the database server it creates using your preferred object data mapper such as Mongoose, Prisma, or TypeORM. In this guide, we’ll use Mongoose (v8.9.6).

Since the data stored by mongodb-memory-server resides in memory by default, it’s faster to read from and write to than when using a real database. mongodb-memory-server is also easier to set up. These benefits make it a good choice for using it as a database server for writing tests.

Note: Make sure to install v9.1.6 of mongodb-memory-server to follow this guide. v10 currently has issues with cleaning up resources after tests are done. See this issue titled Node forking will include any --import from the original command.

The issue has been resolved at the time of writing this article, but the fix has not been merged for installs.

How to Write the Tests

Now I’ll take you through the following steps to get you started writing tests:

  1. Set up the project

  2. Set up mongoose schema

  3. Set up services

  4. Set up tests

  5. Write tests

  6. Pass tests

  7. Use TypeScript (Optional)

1. Set Up the Project

I created a GitHub repository to make it easier for you to follow this guide. Clone the repository at nodejs-test-runner-mongoose and checkout branch 01-setup.

In 01-setup, the dependencies for the project are in the package.json file. Install the dependencies using the npm install command to set up the project. To make sure that the setup is complete and correct, run the node . command in the terminal of your project. You should see your version of Node.js as an output on the terminal.

# install dependencies
npm install
...
# run the node command
node .
# the output
You are running Node.js v22.13.1

2. Set up Mongoose Schema

We’ll set up the schema for two collections (Task and User) in branch 02-setup-schema using Mongoose. The task/model.mjs and user/model.mjs files contain the schema for the Task and the User collection, respectively. We’ll also set up a database connection in index.mjs to ensure that the schema setup works correctly.

I won’t go into detail about Mongoose models and schema in this article because they are outside its scope.

When you run the node . command after implementing the changes in 02-setup-schema, you should see a similar result in the console as in the snippet below:

node .
You are running Node.js v22.13.1
Created user with id 679f1d7f73fbeaf23b2007df
Created task "Task title" for user with id "679f1d7f73fbeaf23b2007df"

You can see the differences between 01-setup and 02-setup-schema via the 01-setup <> 02-setup-schema diff on GitHub.

3. Set Up Services

Next, we create service files (task/service.mjs and user/service.mjs) in branch 03-setup-services. Both files currently contain empty functions that we’ll write tests for later. These functions will contain business logic and also communicate with the database. We’re using JSDoc comments for typing parameters and return values.

Click 02-setup-schema <> 03-setup-services diff to see the code changes between 02-setup-schema and 03-setup-services.

4. Set Up Tests

In branch 04-set-up-tests, we set up the codebase to run tests. We create test.setup.mjs which contains code that will be run before each test file is executed.

In test.setup.mjs, the connect function creates a MongoDB In-Memory server and connects to it with Mongoose for running the tests. The closeDatabase function closes the database connection and cleans up all resources to free memory.

The connect and closeDatabase functions get executed in the t.before hook and the t.after hook respectively. This ensures that, before a test file is run, a database connection is established through t.before. Then after tests for the file have been completely run, the database connection is dropped and the resources used are cleared up through t.after.

In package.json, we’ll update the npm test script to node --test --import ./test.setup.mjs. This command ensures that the test.setup.mjs ES Module is preloaded and executed through the --import CLI command before each test file is run.

Then we’ll create the test files with empty tests in the __tests__ folders for user and task. After implementing the new changes in 04-set-up-tests, running the test script with npm run test should display output similar to the snippet below:

npm run test

> nodejs-test-runner-mongoose@1.0.0 test
> node --test --import ./test.setup.mjs

...

ℹ tests 8
ℹ suites 5
ℹ pass 8
ℹ fail 0
ℹ cancelled 0
ℹ skipped 0
ℹ todo 0
ℹ duration_ms 941.768873

All tests currently pass because there are no assertions that fail in them. We’ll write tests with assertions in the following section.

5. Write Tests

Now it’s time to write tests for the functions in the service files in the 05-write-tests branch. We’re using the Node.js assert library to ensure that values returned from the functions are what we expect. You can view the tests we’ve written when you compare the differences between 04-set-up-tests and 05-write-tests

When the tests script is run, all tests fail because we haven’t written the functions in the service files yet. You should see output similar to the snippet below when you run the test script:

npm run test

> nodejs-test-runner-mongoose@1.0.0 test
> node --test --import ./test.setup.mjs

...

ℹ tests 8
ℹ suites 5
ℹ pass 0
ℹ fail 8
ℹ cancelled 0
ℹ skipped 0
ℹ todo 0
ℹ duration_ms 1202.031961

6. Pass Tests

In 06-pass-tests, we write the functions in the service files to pass the tests. Only 6 out of 7 tests pass when the test script is run because we skipped the test for the getById function in user/service.mjs has with t.skip. We haven’t finished the getById function in user/service.mjs. I figured we could leave it as an exercise.

When you run the test script, you should get a similar output in the terminal as below:

ℹ tests 7
ℹ suites 4
ℹ pass 6
ℹ fail 0
ℹ cancelled 0
ℹ skipped 1
ℹ todo 0
ℹ duration_ms 1287.564918

You can see the code we wrote to pass tests in the code changes between 05-write-tests and 06-pass-tests.

7. Use TypeScript (Optional)

If you intend to run tests with TypeScript, you can checkout branch 07-with-typescript. You need to have Node.js >=v22.6.0 installed because we’re using the --experimental-strip-types option in the test. To set up tests to run with TypeScript, go through the following steps:

  1. Install TypeScript using the npm install typescript --save-dev command

  2. Install tsx using the npm install tsx command

  3. Create a default tsconfig.json file at the root of the project using the npx tsc --init command

In package.json, update the test script to this:

"test": "node --test --experimental-strip-types --import tsx --import ./test.setup.mjs"
  • --experimental-strip-types helps strip out types before each test file is executed.

  • Preloading tsx with the --import helps execute the TypeScript file. Without it, the test runner will not be able to find files imported without the .ts extension. For example, user/model.ts imported with the code snippet below will not be found.

      import { UserModel } from "./model";
    

The rest of the changes from 06-pass-tests to 07-with-typescript involve updating types, changing file extensions from .mjs to .ts and updating import statements.

Conclusion

In this guide, you have learned how to use the built-in Node.js test runner and why it’s often a better choice over other testing libraries and frameworks. You have also learned how to use mongodb-memory-server as a replacement for a real MongoDB server, as well as why it’s a good idea to use this instead of a real MongoDB server for tests.

Most importantly, you have learned how to set up and run tests in Node.js using the Node.js test runner and mongodb-memory-server. You should now know how to set up your projects to run the tests if you use TypeScript.

If you find the nodejs-test-runner-mongoose repository useful, kindly give it a star. It encourages me. Thank you.