mirror of
https://github.com/vitest-dev/vitest.git
synced 2025-12-08 18:26:03 +00:00
791 lines
21 KiB
Markdown
791 lines
21 KiB
Markdown
---
|
|
title: Mocking | Guide
|
|
---
|
|
|
|
# Mocking
|
|
|
|
When writing tests it's only a matter of time before you need to create a "fake" version of an internal — or external — service. This is commonly referred to as **mocking**. Vitest provides utility functions to help you out through its `vi` helper. You can import it from `vitest` or access it globally if [`global` configuration](/config/#globals) is enabled.
|
|
|
|
::: warning
|
|
Always remember to clear or restore mocks before or after each test run to undo mock state changes between runs! See [`mockReset`](/api/mock#mockreset) docs for more info.
|
|
:::
|
|
|
|
If you are not familiar with `vi.fn`, `vi.mock` or `vi.spyOn` methods, check the [API section](/api/vi) first.
|
|
|
|
## Dates
|
|
|
|
Sometimes you need to be in control of the date to ensure consistency when testing. Vitest uses [`@sinonjs/fake-timers`](https://github.com/sinonjs/fake-timers) package for manipulating timers, as well as system date. You can find more about the specific API in detail [here](/api/vi#vi-setsystemtime).
|
|
|
|
### Example
|
|
|
|
```js
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest'
|
|
|
|
const businessHours = [9, 17]
|
|
|
|
function purchase() {
|
|
const currentHour = new Date().getHours()
|
|
const [open, close] = businessHours
|
|
|
|
if (currentHour > open && currentHour < close) {
|
|
return { message: 'Success' }
|
|
}
|
|
|
|
return { message: 'Error' }
|
|
}
|
|
|
|
describe('purchasing flow', () => {
|
|
beforeEach(() => {
|
|
// tell vitest we use mocked time
|
|
vi.useFakeTimers()
|
|
})
|
|
|
|
afterEach(() => {
|
|
// restoring date after each test run
|
|
vi.useRealTimers()
|
|
})
|
|
|
|
it('allows purchases within business hours', () => {
|
|
// set hour within business hours
|
|
const date = new Date(2000, 1, 1, 13)
|
|
vi.setSystemTime(date)
|
|
|
|
// access Date.now() will result in the date set above
|
|
expect(purchase()).toEqual({ message: 'Success' })
|
|
})
|
|
|
|
it('disallows purchases outside of business hours', () => {
|
|
// set hour outside business hours
|
|
const date = new Date(2000, 1, 1, 19)
|
|
vi.setSystemTime(date)
|
|
|
|
// access Date.now() will result in the date set above
|
|
expect(purchase()).toEqual({ message: 'Error' })
|
|
})
|
|
})
|
|
```
|
|
|
|
## Functions
|
|
|
|
Mocking functions can be split up into two different categories; *spying & mocking*.
|
|
|
|
Sometimes all you need is to validate whether or not a specific function has been called (and possibly which arguments were passed). In these cases a spy would be all we need which you can use directly with `vi.spyOn()` ([read more here](/api/vi#vi-spyon)).
|
|
|
|
However spies can only help you **spy** on functions, they are not able to alter the implementation of those functions. In the case where we do need to create a fake (or mocked) version of a function we can use `vi.fn()` ([read more here](/api/vi#vi-fn)).
|
|
|
|
We use [Tinyspy](https://github.com/tinylibs/tinyspy) as a base for mocking functions, but we have our own wrapper to make it `jest` compatible. Both `vi.fn()` and `vi.spyOn()` share the same methods, however only the return result of `vi.fn()` is callable.
|
|
|
|
### Example
|
|
|
|
```js
|
|
import { afterEach, describe, expect, it, vi } from 'vitest'
|
|
|
|
const messages = {
|
|
items: [
|
|
{ message: 'Simple test message', from: 'Testman' },
|
|
// ...
|
|
],
|
|
getLatest, // can also be a `getter or setter if supported`
|
|
}
|
|
|
|
function getLatest(index = messages.items.length - 1) {
|
|
return messages.items[index]
|
|
}
|
|
|
|
describe('reading messages', () => {
|
|
afterEach(() => {
|
|
vi.restoreAllMocks()
|
|
})
|
|
|
|
it('should get the latest message with a spy', () => {
|
|
const spy = vi.spyOn(messages, 'getLatest')
|
|
expect(spy.getMockName()).toEqual('getLatest')
|
|
|
|
expect(messages.getLatest()).toEqual(
|
|
messages.items[messages.items.length - 1],
|
|
)
|
|
|
|
expect(spy).toHaveBeenCalledTimes(1)
|
|
|
|
spy.mockImplementationOnce(() => 'access-restricted')
|
|
expect(messages.getLatest()).toEqual('access-restricted')
|
|
|
|
expect(spy).toHaveBeenCalledTimes(2)
|
|
})
|
|
|
|
it('should get with a mock', () => {
|
|
const mock = vi.fn().mockImplementation(getLatest)
|
|
|
|
expect(mock()).toEqual(messages.items[messages.items.length - 1])
|
|
expect(mock).toHaveBeenCalledTimes(1)
|
|
|
|
mock.mockImplementationOnce(() => 'access-restricted')
|
|
expect(mock()).toEqual('access-restricted')
|
|
|
|
expect(mock).toHaveBeenCalledTimes(2)
|
|
|
|
expect(mock()).toEqual(messages.items[messages.items.length - 1])
|
|
expect(mock).toHaveBeenCalledTimes(3)
|
|
})
|
|
})
|
|
```
|
|
|
|
### More
|
|
|
|
- [Jest's Mock Functions](https://jestjs.io/docs/mock-function-api)
|
|
|
|
## Globals
|
|
|
|
You can mock global variables that are not present with `jsdom` or `node` by using [`vi.stubGlobal`](/api/vi#vi-stubglobal) helper. It will put the value of the global variable into a `globalThis` object.
|
|
|
|
```ts
|
|
import { vi } from 'vitest'
|
|
|
|
const IntersectionObserverMock = vi.fn(() => ({
|
|
disconnect: vi.fn(),
|
|
observe: vi.fn(),
|
|
takeRecords: vi.fn(),
|
|
unobserve: vi.fn(),
|
|
}))
|
|
|
|
vi.stubGlobal('IntersectionObserver', IntersectionObserverMock)
|
|
|
|
// now you can access it as `IntersectionObserver` or `window.IntersectionObserver`
|
|
```
|
|
|
|
## Modules
|
|
|
|
See ["Mocking Modules" guide](/guide/mocking-modules).
|
|
|
|
## File System
|
|
|
|
Mocking the file system ensures that the tests do not depend on the actual file system, making the tests more reliable and predictable. This isolation helps in avoiding side effects from previous tests. It allows for testing error conditions and edge cases that might be difficult or impossible to replicate with an actual file system, such as permission issues, disk full scenarios, or read/write errors.
|
|
|
|
Vitest doesn't provide any file system mocking API out of the box. You can use `vi.mock` to mock the `fs` module manually, but it's hard to maintain. Instead, we recommend using [`memfs`](https://www.npmjs.com/package/memfs) to do that for you. `memfs` creates an in-memory file system, which simulates file system operations without touching the actual disk. This approach is fast and safe, avoiding any potential side effects on the real file system.
|
|
|
|
### Example
|
|
|
|
To automatically redirect every `fs` call to `memfs`, you can create `__mocks__/fs.cjs` and `__mocks__/fs/promises.cjs` files at the root of your project:
|
|
|
|
::: code-group
|
|
```ts [__mocks__/fs.cjs]
|
|
// we can also use `import`, but then
|
|
// every export should be explicitly defined
|
|
|
|
const { fs } = require('memfs')
|
|
module.exports = fs
|
|
```
|
|
|
|
```ts [__mocks__/fs/promises.cjs]
|
|
// we can also use `import`, but then
|
|
// every export should be explicitly defined
|
|
|
|
const { fs } = require('memfs')
|
|
module.exports = fs.promises
|
|
```
|
|
:::
|
|
|
|
```ts [read-hello-world.js]
|
|
import { readFileSync } from 'node:fs'
|
|
|
|
export function readHelloWorld(path) {
|
|
return readFileSync(path, 'utf-8')
|
|
}
|
|
```
|
|
|
|
```ts [hello-world.test.js]
|
|
import { beforeEach, expect, it, vi } from 'vitest'
|
|
import { fs, vol } from 'memfs'
|
|
import { readHelloWorld } from './read-hello-world.js'
|
|
|
|
// tell vitest to use fs mock from __mocks__ folder
|
|
// this can be done in a setup file if fs should always be mocked
|
|
vi.mock('node:fs')
|
|
vi.mock('node:fs/promises')
|
|
|
|
beforeEach(() => {
|
|
// reset the state of in-memory fs
|
|
vol.reset()
|
|
})
|
|
|
|
it('should return correct text', () => {
|
|
const path = '/hello-world.txt'
|
|
fs.writeFileSync(path, 'hello world')
|
|
|
|
const text = readHelloWorld(path)
|
|
expect(text).toBe('hello world')
|
|
})
|
|
|
|
it('can return a value multiple times', () => {
|
|
// you can use vol.fromJSON to define several files
|
|
vol.fromJSON(
|
|
{
|
|
'./dir1/hw.txt': 'hello dir1',
|
|
'./dir2/hw.txt': 'hello dir2',
|
|
},
|
|
// default cwd
|
|
'/tmp',
|
|
)
|
|
|
|
expect(readHelloWorld('/tmp/dir1/hw.txt')).toBe('hello dir1')
|
|
expect(readHelloWorld('/tmp/dir2/hw.txt')).toBe('hello dir2')
|
|
})
|
|
```
|
|
|
|
## Requests
|
|
|
|
Because Vitest runs in Node, mocking network requests is tricky; web APIs are not available, so we need something that will mimic network behavior for us. We recommend [Mock Service Worker](https://mswjs.io/) to accomplish this. It allows you to mock `http`, `WebSocket` and `GraphQL` network requests, and is framework agnostic.
|
|
|
|
Mock Service Worker (MSW) works by intercepting the requests your tests make, allowing you to use it without changing any of your application code. In-browser, this uses the [Service Worker API](https://developer.mozilla.org/en-US/docs/Web/API/Service_Worker_API). In Node.js, and for Vitest, it uses the [`@mswjs/interceptors`](https://github.com/mswjs/interceptors) library. To learn more about MSW, read their [introduction](https://mswjs.io/docs/)
|
|
|
|
### Configuration
|
|
|
|
You can use it like below in your [setup file](/config/#setupfiles)
|
|
|
|
::: code-group
|
|
|
|
```js [HTTP Setup]
|
|
import { afterAll, afterEach, beforeAll } from 'vitest'
|
|
import { setupServer } from 'msw/node'
|
|
import { http, HttpResponse } from 'msw'
|
|
|
|
const posts = [
|
|
{
|
|
userId: 1,
|
|
id: 1,
|
|
title: 'first post title',
|
|
body: 'first post body',
|
|
},
|
|
// ...
|
|
]
|
|
|
|
export const restHandlers = [
|
|
http.get('https://rest-endpoint.example/path/to/posts', () => {
|
|
return HttpResponse.json(posts)
|
|
}),
|
|
]
|
|
|
|
const server = setupServer(...restHandlers)
|
|
|
|
// Start server before all tests
|
|
beforeAll(() => server.listen({ onUnhandledRequest: 'error' }))
|
|
|
|
// Close server after all tests
|
|
afterAll(() => server.close())
|
|
|
|
// Reset handlers after each test for test isolation
|
|
afterEach(() => server.resetHandlers())
|
|
```
|
|
|
|
```js [GraphQL Setup]
|
|
import { afterAll, afterEach, beforeAll } from 'vitest'
|
|
import { setupServer } from 'msw/node'
|
|
import { graphql, HttpResponse } from 'msw'
|
|
|
|
const posts = [
|
|
{
|
|
userId: 1,
|
|
id: 1,
|
|
title: 'first post title',
|
|
body: 'first post body',
|
|
},
|
|
// ...
|
|
]
|
|
|
|
const graphqlHandlers = [
|
|
graphql.query('ListPosts', () => {
|
|
return HttpResponse.json({
|
|
data: { posts },
|
|
})
|
|
}),
|
|
]
|
|
|
|
const server = setupServer(...graphqlHandlers)
|
|
|
|
// Start server before all tests
|
|
beforeAll(() => server.listen({ onUnhandledRequest: 'error' }))
|
|
|
|
// Close server after all tests
|
|
afterAll(() => server.close())
|
|
|
|
// Reset handlers after each test for test isolation
|
|
afterEach(() => server.resetHandlers())
|
|
```
|
|
|
|
```js [WebSocket Setup]
|
|
import { afterAll, afterEach, beforeAll } from 'vitest'
|
|
import { setupServer } from 'msw/node'
|
|
import { ws } from 'msw'
|
|
|
|
const chat = ws.link('wss://chat.example.com')
|
|
|
|
const wsHandlers = [
|
|
chat.addEventListener('connection', ({ client }) => {
|
|
client.addEventListener('message', (event) => {
|
|
console.log('Received message from client:', event.data)
|
|
// Echo the received message back to the client
|
|
client.send(`Server received: ${event.data}`)
|
|
})
|
|
}),
|
|
]
|
|
|
|
const server = setupServer(...wsHandlers)
|
|
|
|
// Start server before all tests
|
|
beforeAll(() => server.listen({ onUnhandledRequest: 'error' }))
|
|
|
|
// Close server after all tests
|
|
afterAll(() => server.close())
|
|
|
|
// Reset handlers after each test for test isolation
|
|
afterEach(() => server.resetHandlers())
|
|
```
|
|
:::
|
|
|
|
> Configuring the server with `onUnhandledRequest: 'error'` ensures that an error is thrown whenever there is a request that does not have a corresponding request handler.
|
|
|
|
### More
|
|
There is much more to MSW. You can access cookies and query parameters, define mock error responses, and much more! To see all you can do with MSW, read [their documentation](https://mswjs.io/docs).
|
|
|
|
## Timers
|
|
|
|
When we test code that involves timeouts or intervals, instead of having our tests wait it out or timeout, we can speed up our tests by using "fake" timers that mock calls to `setTimeout` and `setInterval`.
|
|
|
|
See the [`vi.useFakeTimers` API section](/api/vi#vi-usefaketimers) for a more in depth detailed API description.
|
|
|
|
### Example
|
|
|
|
```js
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest'
|
|
|
|
function executeAfterTwoHours(func) {
|
|
setTimeout(func, 1000 * 60 * 60 * 2) // 2 hours
|
|
}
|
|
|
|
function executeEveryMinute(func) {
|
|
setInterval(func, 1000 * 60) // 1 minute
|
|
}
|
|
|
|
const mock = vi.fn(() => console.log('executed'))
|
|
|
|
describe('delayed execution', () => {
|
|
beforeEach(() => {
|
|
vi.useFakeTimers()
|
|
})
|
|
afterEach(() => {
|
|
vi.restoreAllMocks()
|
|
})
|
|
it('should execute the function', () => {
|
|
executeAfterTwoHours(mock)
|
|
vi.runAllTimers()
|
|
expect(mock).toHaveBeenCalledTimes(1)
|
|
})
|
|
it('should not execute the function', () => {
|
|
executeAfterTwoHours(mock)
|
|
// advancing by 2ms won't trigger the func
|
|
vi.advanceTimersByTime(2)
|
|
expect(mock).not.toHaveBeenCalled()
|
|
})
|
|
it('should execute every minute', () => {
|
|
executeEveryMinute(mock)
|
|
vi.advanceTimersToNextTimer()
|
|
expect(mock).toHaveBeenCalledTimes(1)
|
|
vi.advanceTimersToNextTimer()
|
|
expect(mock).toHaveBeenCalledTimes(2)
|
|
})
|
|
})
|
|
```
|
|
|
|
## Classes
|
|
|
|
You can mock an entire class with a single `vi.fn` call.
|
|
|
|
```ts
|
|
class Dog {
|
|
name: string
|
|
|
|
constructor(name: string) {
|
|
this.name = name
|
|
}
|
|
|
|
static getType(): string {
|
|
return 'animal'
|
|
}
|
|
|
|
greet = (): string => {
|
|
return `Hi! My name is ${this.name}!`
|
|
}
|
|
|
|
speak(): string {
|
|
return 'bark!'
|
|
}
|
|
|
|
isHungry() {}
|
|
feed() {}
|
|
}
|
|
```
|
|
|
|
We can re-create this class with `vi.fn` (or `vi.spyOn().mockImplementation()`):
|
|
|
|
```ts
|
|
const Dog = vi.fn(class {
|
|
static getType = vi.fn(() => 'mocked animal')
|
|
|
|
constructor(name) {
|
|
this.name = name
|
|
}
|
|
|
|
greet = vi.fn(() => `Hi! My name is ${this.name}!`)
|
|
speak = vi.fn(() => 'loud bark!')
|
|
feed = vi.fn()
|
|
})
|
|
```
|
|
|
|
::: warning
|
|
If a non-primitive is returned from the constructor function, that value will become the result of the new expression. In this case the `[[Prototype]]` may not be correctly bound:
|
|
|
|
```ts
|
|
const CorrectDogClass = vi.fn(function (name) {
|
|
this.name = name
|
|
})
|
|
|
|
const IncorrectDogClass = vi.fn(name => ({
|
|
name
|
|
}))
|
|
|
|
const Marti = new CorrectDogClass('Marti')
|
|
const Newt = new IncorrectDogClass('Newt')
|
|
|
|
Marti instanceof CorrectDogClass // ✅ true
|
|
Newt instanceof IncorrectDogClass // ❌ false!
|
|
```
|
|
|
|
If you are mocking classes, prefer the class syntax over the function.
|
|
:::
|
|
|
|
::: tip WHEN TO USE?
|
|
Generally speaking, you would re-create a class like this inside the module factory if the class is re-exported from another module:
|
|
|
|
```ts
|
|
import { Dog } from './dog.js'
|
|
|
|
vi.mock(import('./dog.js'), () => {
|
|
const Dog = vi.fn(class {
|
|
feed = vi.fn()
|
|
// ... other mocks
|
|
})
|
|
return { Dog }
|
|
})
|
|
```
|
|
|
|
This method can also be used to pass an instance of a class to a function that accepts the same interface:
|
|
|
|
```ts [src/feed.ts]
|
|
function feed(dog: Dog) {
|
|
// ...
|
|
}
|
|
```
|
|
```ts [tests/dog.test.ts]
|
|
import { expect, test, vi } from 'vitest'
|
|
import { feed } from '../src/feed.js'
|
|
|
|
const Dog = vi.fn(class {
|
|
feed = vi.fn()
|
|
})
|
|
|
|
test('can feed dogs', () => {
|
|
const dogMax = new Dog('Max')
|
|
|
|
feed(dogMax)
|
|
|
|
expect(dogMax.feed).toHaveBeenCalled()
|
|
expect(dogMax.isHungry()).toBe(false)
|
|
})
|
|
```
|
|
:::
|
|
|
|
Now, when we create a new instance of the `Dog` class its `speak` method (alongside `feed` and `greet`) is already mocked:
|
|
|
|
```ts
|
|
const Cooper = new Dog('Cooper')
|
|
Cooper.speak() // loud bark!
|
|
Cooper.greet() // Hi! My name is Cooper!
|
|
|
|
// you can use built-in assertions to check the validity of the call
|
|
expect(Cooper.speak).toHaveBeenCalled()
|
|
expect(Cooper.greet).toHaveBeenCalled()
|
|
|
|
const Max = new Dog('Max')
|
|
|
|
// methods are not shared between instances if you assigned them directly
|
|
expect(Max.speak).not.toHaveBeenCalled()
|
|
expect(Max.greet).not.toHaveBeenCalled()
|
|
```
|
|
|
|
We can reassign the return value for a specific instance:
|
|
|
|
```ts
|
|
const dog = new Dog('Cooper')
|
|
|
|
// "vi.mocked" is a type helper, since
|
|
// TypeScript doesn't know that Dog is a mocked class,
|
|
// it wraps any function in a Mock<T> type
|
|
// without validating if the function is a mock
|
|
vi.mocked(dog.speak).mockReturnValue('woof woof')
|
|
|
|
dog.speak() // woof woof
|
|
```
|
|
|
|
To mock the property, we can use the `vi.spyOn(dog, 'name', 'get')` method. This makes it possible to use spy assertions on the mocked property:
|
|
|
|
```ts
|
|
const dog = new Dog('Cooper')
|
|
|
|
const nameSpy = vi.spyOn(dog, 'name', 'get').mockReturnValue('Max')
|
|
|
|
expect(dog.name).toBe('Max')
|
|
expect(nameSpy).toHaveBeenCalledTimes(1)
|
|
```
|
|
|
|
::: tip
|
|
You can also spy on getters and setters using the same method.
|
|
:::
|
|
|
|
::: danger
|
|
Using classes with `vi.fn()` was introduced in Vitest 4. Previously, you had to use `function` and `prototype` inheritence directly. See [v3 guide](https://v3.vitest.dev/guide/mocking.html#classes).
|
|
:::
|
|
|
|
## Cheat Sheet
|
|
|
|
:::info
|
|
`vi` in the examples below is imported directly from `vitest`. You can also use it globally, if you set `globals` to `true` in your [config](/config/).
|
|
:::
|
|
|
|
I want to…
|
|
|
|
### Mock exported variables
|
|
```js [example.js]
|
|
export const getter = 'variable'
|
|
```
|
|
```ts [example.test.ts]
|
|
import * as exports from './example.js'
|
|
|
|
vi.spyOn(exports, 'getter', 'get').mockReturnValue('mocked')
|
|
```
|
|
|
|
::: warning
|
|
This will not work in the Browser Mode. For a workaround, see [Limitations](/guide/browser/#spying-on-module-exports).
|
|
:::
|
|
|
|
### Mock an exported function
|
|
|
|
1. Example with `vi.mock`:
|
|
|
|
::: warning
|
|
Don't forget that a `vi.mock` call is hoisted to top of the file. It will always be executed before all imports.
|
|
:::
|
|
|
|
```ts [example.js]
|
|
export function method() {}
|
|
```
|
|
```ts
|
|
import { method } from './example.js'
|
|
|
|
vi.mock('./example.js', () => ({
|
|
method: vi.fn()
|
|
}))
|
|
```
|
|
|
|
2. Example with `vi.spyOn`:
|
|
```ts
|
|
import * as exports from './example.js'
|
|
|
|
vi.spyOn(exports, 'method').mockImplementation(() => {})
|
|
```
|
|
|
|
::: warning
|
|
`vi.spyOn` example will not work in the Browser Mode. For a workaround, see [Limitations](/guide/browser/#spying-on-module-exports).
|
|
:::
|
|
|
|
### Mock an exported class implementation
|
|
|
|
1. Example with a fake `class`:
|
|
```ts [example.js]
|
|
export class SomeClass {}
|
|
```
|
|
```ts
|
|
import { SomeClass } from './example.js'
|
|
|
|
vi.mock(import('./example.js'), () => {
|
|
const SomeClass = vi.fn(class FakeClass {
|
|
someMethod = vi.fn()
|
|
})
|
|
return { SomeClass }
|
|
})
|
|
// SomeClass.mock.instances will have SomeClass
|
|
```
|
|
|
|
2. Example with `vi.mock` and `.prototype`:
|
|
```ts [example.js]
|
|
export class SomeClass {}
|
|
```
|
|
```ts
|
|
import { SomeClass } from './example.js'
|
|
|
|
vi.mock(import('./example.js'), () => {
|
|
const SomeClass = vi.fn()
|
|
SomeClass.prototype.someMethod = vi.fn()
|
|
return { SomeClass }
|
|
})
|
|
// SomeClass.mock.instances will have SomeClass
|
|
```
|
|
|
|
3. Example with `vi.spyOn`:
|
|
|
|
```ts
|
|
import * as mod from './example.js'
|
|
|
|
const SomeClass = vi.fn()
|
|
SomeClass.prototype.someMethod = vi.fn()
|
|
|
|
vi.spyOn(mod, 'SomeClass').mockImplementation(SomeClass)
|
|
```
|
|
|
|
::: warning
|
|
`vi.spyOn` example will not work in the Browser Mode. For a workaround, see [Limitations](/guide/browser/#spying-on-module-exports).
|
|
:::
|
|
|
|
### Spy on an object returned from a function
|
|
|
|
1. Example using cache:
|
|
|
|
```ts [example.js]
|
|
export function useObject() {
|
|
return { method: () => true }
|
|
}
|
|
```
|
|
|
|
```ts [useObject.js]
|
|
import { useObject } from './example.js'
|
|
|
|
const obj = useObject()
|
|
obj.method()
|
|
```
|
|
|
|
```ts [useObject.test.js]
|
|
import { useObject } from './example.js'
|
|
|
|
vi.mock(import('./example.js'), () => {
|
|
let _cache
|
|
const useObject = () => {
|
|
if (!_cache) {
|
|
_cache = {
|
|
method: vi.fn(),
|
|
}
|
|
}
|
|
// now every time that useObject() is called it will
|
|
// return the same object reference
|
|
return _cache
|
|
}
|
|
return { useObject }
|
|
})
|
|
|
|
const obj = useObject()
|
|
// obj.method was called inside some-path
|
|
expect(obj.method).toHaveBeenCalled()
|
|
```
|
|
|
|
### Mock part of a module
|
|
|
|
```ts
|
|
import { mocked, original } from './some-path.js'
|
|
|
|
vi.mock(import('./some-path.js'), async (importOriginal) => {
|
|
const mod = await importOriginal()
|
|
return {
|
|
...mod,
|
|
mocked: vi.fn()
|
|
}
|
|
})
|
|
original() // has original behaviour
|
|
mocked() // is a spy function
|
|
```
|
|
|
|
::: warning
|
|
Don't forget that this only [mocks _external_ access](#mocking-pitfalls). In this example, if `original` calls `mocked` internally, it will always call the function defined in the module, not in the mock factory.
|
|
:::
|
|
|
|
### Mock the current date
|
|
|
|
To mock `Date`'s time, you can use `vi.setSystemTime` helper function. This value will **not** automatically reset between different tests.
|
|
|
|
Beware that using `vi.useFakeTimers` also changes the `Date`'s time.
|
|
|
|
```ts
|
|
const mockDate = new Date(2022, 0, 1)
|
|
vi.setSystemTime(mockDate)
|
|
const now = new Date()
|
|
expect(now.valueOf()).toBe(mockDate.valueOf())
|
|
// reset mocked time
|
|
vi.useRealTimers()
|
|
```
|
|
|
|
### Mock a global variable
|
|
|
|
You can set global variable by assigning a value to `globalThis` or using [`vi.stubGlobal`](/api/vi#vi-stubglobal) helper. When using `vi.stubGlobal`, it will **not** automatically reset between different tests, unless you enable [`unstubGlobals`](/config/#unstubglobals) config option or call [`vi.unstubAllGlobals`](/api/vi#vi-unstuballglobals).
|
|
|
|
```ts
|
|
vi.stubGlobal('__VERSION__', '1.0.0')
|
|
expect(__VERSION__).toBe('1.0.0')
|
|
```
|
|
|
|
### Mock `import.meta.env`
|
|
|
|
1. To change environmental variable, you can just assign a new value to it.
|
|
|
|
::: warning
|
|
The environmental variable value will **_not_** automatically reset between different tests.
|
|
:::
|
|
|
|
```ts
|
|
import { beforeEach, expect, it } from 'vitest'
|
|
|
|
// you can reset it in beforeEach hook manually
|
|
const originalViteEnv = import.meta.env.VITE_ENV
|
|
|
|
beforeEach(() => {
|
|
import.meta.env.VITE_ENV = originalViteEnv
|
|
})
|
|
|
|
it('changes value', () => {
|
|
import.meta.env.VITE_ENV = 'staging'
|
|
expect(import.meta.env.VITE_ENV).toBe('staging')
|
|
})
|
|
```
|
|
|
|
2. If you want to automatically reset the value(s), you can use the `vi.stubEnv` helper with the [`unstubEnvs`](/config/#unstubenvs) config option enabled (or call [`vi.unstubAllEnvs`](/api/vi#vi-unstuballenvs) manually in a `beforeEach` hook):
|
|
|
|
```ts
|
|
import { expect, it, vi } from 'vitest'
|
|
|
|
// before running tests "VITE_ENV" is "test"
|
|
import.meta.env.VITE_ENV === 'test'
|
|
|
|
it('changes value', () => {
|
|
vi.stubEnv('VITE_ENV', 'staging')
|
|
expect(import.meta.env.VITE_ENV).toBe('staging')
|
|
})
|
|
|
|
it('the value is restored before running an other test', () => {
|
|
expect(import.meta.env.VITE_ENV).toBe('test')
|
|
})
|
|
```
|
|
|
|
```ts [vitest.config.ts]
|
|
export default defineConfig({
|
|
test: {
|
|
unstubEnvs: true,
|
|
},
|
|
})
|
|
```
|