DEV Community

Cover image for Data-driven Unit Tests with Jest
Dani Meier
Dani Meier

Posted on

Data-driven Unit Tests with Jest

Photo by Markus Spiske on Unsplash

Data-driven Unit Tests are an easy and efficient way to run a test with a range of input data. That can be useful in situations like

  • testing against a broader range of values
  • testing for null and undefined values
  • testing against edge cases, e.g., for dates (beginning of the month, end of the month, …)
  • or to tweak your unit test stats - because each input row counts as a separate test. Your manager will be impressed. 😉

Simple sample

Let’s define a simple test to check if a number is even:

it('should be even', () => {
    expect(2 % 2).toBe(0);
});
 ✓ is even

To check a series of numbers, we can use Jest’s it.each (or test.each, since it is just an alias of test). We pass the input data as an array, each item is then tested individually (n).

it.each([2,4,6])('should be even', (n) => {
    expect(n % 2).toBe(0);
});
✓ should be even
✓ should be even
✓ should be even

We can use several predefined tokens to improve the name of the test, which makes it easier to locate a failing test:

it.each([2,4,5,6])('%i should be even', (n) => {
    expect(n % 2).toBe(0);
});
✓ 2 should be even
✓ 4 should be even
✕ 5 should be even
✓ 6 should be even

Multiple parameters

You can pass multiple parameters to your test as well. In the next sample, we pass the expected value to the test as well:

it.each([[2,4], [6,36], [8,64]])('can square %i to %s', (n, expected) => {
    expect(n*n).toBe(expected);
});
✓ can square 2 to 4
✓ can square 6 to 36
✓ can square 8 to 64

The input table is defined as an array of arrays. Each inner array is one row of the input table.

I like tests to be well readable, understandable and clear in their intention. The focus should be on the actual test and not on the implementation. Jest provides a syntax that supports exactly that:

it.each`
      n     | expected
      ${1}  | ${1}
      ${2}  | ${4}
      ${4}  | ${16}
      ${12} | ${144}
    `('can square $n to $expected', ({ n, expected }) => {
            expect(n*n).toBe(expected);
});
✓ can square 1 to 1
✓ can square 2 to 4
✓ can square 4 to 16
✓ can square 12 to 144

The input table now looks like an actual table. It is defined as a template string, in which the first row contains the variables names separated by a pipe |. Each subsequent row is one combination of input data to run the test against, defined as a template literal expression.
You can reference the variable names inside the test name with $variable.

Run multiple tests

.each is also available for describe-Blocks, which makes it easy to run multiple tests or even a whole test suite against a set of input data.

describe.each`
    x     | y
    ${10} | ${2}
    ${9}  | ${3}
    ${8}  | ${4}
`('With x=$x and y=$y', ({x, y}) => {
    it('x should be larger then y', () => {
        expect(x).toBeGreaterThan(y);
    });

    it('should x should by dividable by y without rest', () => {
        expect(x % y).toBe(0);
    });
});
With x=10 and y=2
    ✓ x should be larger then y
    ✓ should x should by dividable by y without rest
With x=9 and y=3
    ✓ x should be larger then y (1ms)
    ✓ should x should by dividable by y without rest
With x=8 and y=4
    ✓ x should be larger then y
    ✓ should x should by dividable by y without rest (1ms)

You can even combine describe.each and it.each, but keep an eye on comprehensibility.

Top comments (3)

Collapse
 
leogalani profile image
Leonardo Barba Galani

might sound silly but ... thanks.
this opens so many possibilities!

Collapse
 
chrisachard profile image
Chris Achard

Tests are something I know I should do more of, but keep putting off 🤦‍♂️ Do you have any advice for how to make sure I keep doing them?

Collapse
 
wolverineks profile image
Kevin Sullivan

Testing is a skill and a habit. You have to practice it. It's not a light switch you can turn on when you want to. TDD small stuff in your spare time, then grow it bigger and more often. Then repeat, but at work. It'll take a while, but it's worth it.