Skip to content

Automated Testing with Node.js and Gitlab

Posted on:January 25, 2025 at 02:22 PM

Continuous Integration (CI) is an essential part of modern software development. In this guide, we’ll walk through setting up a CI pipeline to test a Node.js application using these tools:

test coverage visualization

Step 1: Write a Test

Let’s start with the basics. Suppose you already have a test, which might look something like this:

import { describe, it } from "node:test";
import assert from "node:assert";
import { fooOne } from "./foo.js";

describe("fooTest", () => {
  it("returns result", () => {
    const result = fooOne(12);
    assert.strictEqual(result, 14);
  });

  it("handles when x equals to 2", () => {
    const result = fooOne(2);
    assert.strictEqual(result, 3);
  });
});

Notice that we’re using the describe and it methods from the node:test module to define our test cases.

Step 2: Install Dependencies

To enable Test coverage visualization, we need to add the cobertura library as a development dependency:

npm i -D cobertura

Step 3: Define Your Test Script

To keep things clean and manageable, I recommend creating a separate script file for running your tests. This way, you avoid having a long and cluttered CLI command:

mkdir scripts
touch scripts/test.sh
chmod +x scripts/test.sh

Here’s an example test script:

// scripts/test.sh
#!/bin/sh
node \
  --test \
  --experimental-test-coverage  \
  --test-reporter=junit --test-reporter-destination=rspec.xml \
  --test-reporter=spec --test-reporter-destination=stdout \
  --test-reporter=cobertura --test-reporter-destination=cobertura.xml

Additional Flags

Depending on your setup, you might need these flags when running your tests:

Example for TypeScript Projects

If you’re using TypeScript, here’s an updated script:

// scripts/test.sh
#!/bin/sh

tsc --build . && \

node \
  --test \
  --enable-source-maps \
  --experimental-test-coverage  \
  --test-reporter=junit --test-reporter-destination=rspec.xml \
  --test-reporter=spec --test-reporter-destination=stdout \
  --test-reporter=cobertura --test-reporter-destination=cobertura.xml \
  './build/**/*.e2e.js'

Run and Verify

You can now test if the script works correctly:

./scripts/test.sh

After running the script, you should see the generated files rspec.xml and cobertura.xml in your project directory.

Add the Test Script to package.json

To stick with common practices, include the test script in your package.json file:

{
  "scripts": {
    "test": "./scripts/test.sh"
  }
}

This makes it easier to run your tests with a simple npm test command while keeping your workflow standardized.

Step 4: Add a New Job to .gitlab-ci.yml

The final step is to define a new job in your .gitlab-ci.yml file:

stages:
  - test

test:
  stage: test
  image: node:22-alpine
  artifacts:
    when: always
    paths:
      - rspec.xml
    reports:
      junit: rspec.xml
      coverage_report:
        coverage_format: cobertura
        path: ./cobertura.xml
  script:
    - npm ci
    - npm run test
  coverage: '/all files[^|]*\|[^|]*\s+([\d\.]+)/'

Summary

Overall, your Merge Request will look something like this example. You’ll see test reports, coverage visualization, and other key details directly in the GitLab UI, making it easier to track the health of your codebase.

test coverage visualization

test coverage report

test coverage number