Two shades of mocking a function in Vitest

Feb 20, 2024 · 7 min read
Share on
Two shades of mocking a function in Vitest

Mocking is the approach of replacing a real object with a fake object that simulates the original behavior, hence allowing us to test our code in isolation (unit test), without having to worry about instability may arised by the behavior of the external dependencies. In this article, we will discuss two different ways of mocking a external function call in Vitest, using vi.fn() and vi.spyOn() APIs, and when to use which.

Table of contents

Prerequisites

You should have a Vite project set up and running. The recommended way is to initialize your project with Vite as its bundle management tool, using the command npm create vite@latest.

You need to install Vitest as the unit testing framework for our tests, by running the following command:

yarn add -D vitest

Additionally, if you are testing your front-end application, you can install the following tools to help you with any framework-specific unit tests:

In vitest.config.js (or vite.config.js for Vite projects), we add the following test object to enable Vitest APIs as globally available in our tests:

//...
export default defineConfig({
  test: {
    global: true,
    environment: 'jsdom', //this is optional, only when you are testing DOM related functionalities
  },
})

If you are writing your tests in TypeScript, you can also add the following tsconfig.json file to provide the types for Vitest APIs to TypeScript type checker:

{/** tsconfig.json */
  "compilerOptions": {
    "types": [["vitest/globals"]]
  }
}

Lastly, we add a new test:unit command to the package.json file. This command will find and run the unit tests from the src/ as its root folder, as show below:

"scripts": {
    "test:unit": "vitest --root src/",
}

With this setup, we are ready to explore different ways of mocking external function call with Vitest, starting with our testing scenario.

Our test scenario

We have a function fetchData that makes a network request to an API and returns the data, as follows:

// fetchData.js
export async function fetchData() {
  const response = await fetch(
    'http://exploringvue.com/.netlify/functions/pizzas'
  );
  const data = await response.json();
  return data;
}

We want to test this function, and we need to mock the fetch call to avoid making a real network request, potentially harming the quality and the scope of the test scenarios.

There are many ways of mocking a fetch call in Vitest. One way is to use vi.fn(), which we will discuss next.

Mocking a function using vi.fn()

fetch is a global method that helps to retrieve resources asynchronously from the server. It is available in both server side (Nodejs API) and client side (Web API) environments. Our testing environment is Nodejs, hence the fetch method is available as part of global object, as global.fetch and not window.fetch.

vi.fn() is the most straightforward approach to mock a function call in Vitest. It returns a new mock instance (or spy instance) that we can use to replace the original function, as shown below:

import { vi } from 'vitest';
import { fetchData } from './fetchData';

const mockedImplementation = () => Promise.resolve({ 
  json() { 
    return { data: 'mocked data'}
  } 
})

describe ('fetchData', () => {
  global.fetch = vi.fn(mockedImplementation);

  it('should return the mocked data', async () => {
    const data = await fetchData();
    expect(data).toEqual({ data: 'mocked data' });
  });
});

In the above code, we are replacing the original global.fetch method with a new mock function that returns a resolved promise with the data { data: 'mocked data' }. This way, we can test the fetchData function without making a real network request.

vi.fn() also accepts no arguments, which means it returns a new mock function that does nothing, such as global.fetch = vi.fn(), and uses mockImplementation() to provide a new mock implementation:

describe ('fetchData', () => {
  global.fetch = vi.fn();
  global.fetch.mockImplementation(mockedImplementation);
  //...
});

This approach can be handy when we only want to fast-mock a function call's return value without any side effects to the test results.

However, there is a problem with the mock approach with vi.fn(). In our current code, we are losing the original implementation of the global.fetch method, and we won't be able to restore it after the test. This can potentially cause test-contamination, where the mock function from one test may affect the result of another test, which uses the original fetch. To solve this issue, we should save the original implementation of the global.fetch method before all the tests run within the same test suite:

import { vi } from 'vitest';

describe ('fetchData', () => {
  let originalFetch;

  beforeAll(() => {
    originalFetch = global.fetch;
    global.fetch = vi.fn(mockedImplementation);
  });

  // ...tests
});

And after running all the test cases, we restore the global.fetch method to its original implementation:

describe ('fetchData', () => {
  // ...

  afterAll(() => {
    global.fetch = originalFetch;
  });
});

The above code will ensure the global.fetch function will be restored after we finish with the fetchData test suite and won't affect other relevant tests. Altenatively, there is a better way to mock and spy a function while maintains the original implementation and reduces the chance for test-contamination, with lesser code. We will discuss this in the next section.

Mocking a global function using vi.spyOn()

Similar to vi.fn(), vi.spyOn() also return a Vitest spy instance but we don't need to replace the original global.fetch with it. vi.spyOn() accepts two arguments:

  • The object containing the method (global), and
  • The method name to spy on (fetch).
describe ('fetchData', () => {
  const fetchSpy = vi.spyOn(global, 'fetch');

  // ...tests
});

And then we can mock the implementation of the fetch method using the mockImplementation method of the spy instance:

describe ('fetchData', () => {
  const fetchSpy = vi.spyOn(global, 'fetch');

  fetchSpy.mockImplementation(mockedImplementation);

  // the tests are the same
});

With this approach, we don't need to save the original implementation of the global.fetch method manually. Instead, we can use the mockRestore method of the spy instance within afterAll hook to restore the original implementation afterwards:

describe ('fetchData', () => {
  //...

  afterAll(() => {
    fetchSpy.mockRestore();
  });
});

This is my preferred way of mocking a external function call in Vitest, as it is more readable and maintainable, in comparison to the vi.fn() approach. In fact, vi.spyOn() returns a spy instance whose implementation defaults to the original method, and we can change it per test case. This approach is really handy for both integrating tests and unit tests, as it allows us to spy on a specific dependency of the target code, and to assert the interaction between them.

So, when should we use vi.fn() and vi.spyOn()?

vi.fn vs vi.spyOn - When to use which?

While both vi.fn() and vi.spyOn() can be used to mock a function call in Vitest, and behind the scenes, vi.spyOn is a wrapper around vi.fn, they are designated for different use cases, as shown in the table below:

vi.fn()vi.spyOn()
When you want to replace the original function with a new mock functionWhen you want to spy on the original function and mock its implementation per scenario
You need to save the original implementation manuallyThere is a built-in method for restore the original function
More suitable for fast-mocking a function callMore suitable for mocking a function call with a side effect
For unit testing the function's implementation, not its interaction with other functionsFor both unit and integration tests

Summary

We have learned how to mock a function call, such as the global fetch in Vitest using vi.fn() and vi.spyOn(). We have also discussed the differences between these two approaches and when to use which. Mocking is a powerful tool for isolating the dependencies of the code we are testing, or for asserting the interaction between them. In the next article, we will discuss further on different mocking scenarios, both for unit and integration tests, and the best practices for asserting mocks and cleaning up after mocking.

👉 Learn about Vue 3 and TypeScript with my new book Learning Vue!

👉 If you'd like to catch up with me sometimes, follow me on X | LinkedIn.

Like this post or find it helpful? Share it 👇🏼 😉

Share on

Learning Vue

Learn the core concepts of Vue.js, the modern JavaScript framework for building frontend applications and interfaces from scratch

Get a copy
Learning Vue