Home Writing Reading

til / Migrating from Jest to Vitest

During a recent hack day at work, I wanted to explore if it would be possible for us to migrate our Jest tests to Vitest. We only have 77 tests as the project is new. It turned out to be very straight-forward and I was able to migrate all tests in a day.

Vitest is a new testing framework powered by Vite. It’s built by the Vue core team, but works with most of the big frameworks. Vitest provides Jest compatible APIs, which means migrations should be easy. It also has built in assertions for both Chai and Jasmine (which Jest uses).

The day before my experiment, the Vitest team removed the “in development and not stable” message from their GitHub. This means they’ve come to a point where they think the framework is stable and usable in production projects. All the functionality seems to be in place from my testing. One tiny issue is that I got Vue prop warnings in the console, even though the props seem to be fine.

Speed #

While doing some naive tests, our test suite in Vitest ran 52% faster than it would in Jest, 13.997 seconds compared to 6.76 seconds. This was only a simple test, but an indication that Vitest is much faster.

Coverage #

We have a script that automatically updates our coverage thresholds. If a PR falls below the thresholds, we can fail the tests in CI. Vitest uses c8 to create coverage statistics, but currently c8 doesn’t support Vue files, which means that our stats were way off. This is our only stopper in merging the new tests, but there’s a PR that fixes this which we hope will be merged soon.

Examples #

Here are some examples of code before and after migration.

// Before, using Jest
jest.mock('@/utils/campaign', () => ({
  ...jest.requireActual<any>('@/utils/campaign'),
  campaign: {
    loadClients: jest.fn(),
  },
}))

// After, using Vitest
vi.mock('@/utils/campaign', async () => ({
  ...(await vi.importActual<any>('@/utils/campaign')),
  campaign: {
    loadClients: vi.fn(),
  },
}))

Since Vitest includes Jest compatible APIs, we mostly had to exchange jest with vi. The biggest difference in this case is where we import the actual utilities. We need to import them asynchronously because Vitest is ESM first.

// Before, using Jest
;(campaignUtils.loadClients as jest.Mock).mockResolvedValue([
  { name: 'advertiser' },
])

// After, using Vitest
import { SpyInstanceFn } from 'vitest'

;(campaignUtils.loadClients as SpyInstanceFn).mockResolvedValue([
  { name: 'advertiser' },
])

Unlike Jest, Vitest doesn’t include global values by default. However, we turned them on to ease migration. Since we don’t have access to globals we need to import SpyInstanceFn when we are casting a mocked function to a spy.

To include the globals add the following in vite.config.ts and tsconfig.json (if you are using TypeScript)

// vite.config.ts
export default defineConfig({
	// ../
	test: {
		globals: true,
	},
});

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

  • Loading next post...
  • Loading previous post...