From f5e0b9877f23b46043fab9a0b2e7819e9c23f2ce Mon Sep 17 00:00:00 2001 From: Sung Ye In <66503450+syi0808@users.noreply.github.com> Date: Mon, 12 Aug 2024 23:14:12 +0900 Subject: [PATCH] feat(vitest): add return type and promisable mockFactory (#6139) Co-authored-by: Vladimir Sheremet --- docs/api/vi.md | 8 +++++--- packages/vitest/src/types/mocker.ts | 6 ++++-- test/config/test/workers-option.test.ts | 1 + test/core/test/mocking/factory.test.ts | 2 +- 4 files changed, 11 insertions(+), 6 deletions(-) diff --git a/docs/api/vi.md b/docs/api/vi.md index ea3780c1df9e..0c7e89530413 100644 --- a/docs/api/vi.md +++ b/docs/api/vi.md @@ -17,7 +17,7 @@ This section describes the API that you can use when [mocking a module](/guide/m ### vi.mock - **Type**: `(path: string, factory?: (importOriginal: () => unknown) => unknown) => void` -- **Type**: `(path: Promise, factory?: (importOriginal: () => T) => unknown) => void` +- **Type**: `(path: Promise, factory?: (importOriginal: () => T) => T | Promise) => void` Substitutes all imported modules from provided `path` with another module. You can use configured Vite aliases inside a path. The call to `vi.mock` is hoisted, so it doesn't matter where you call it. It will always be executed before all imports. If you need to reference some variables outside of its scope, you can define them inside [`vi.hoisted`](#vi-hoisted) and reference them inside `vi.mock`. @@ -31,7 +31,9 @@ Vitest will not mock modules that were imported inside a [setup file](/config/#s If `factory` is defined, all imports will return its result. Vitest calls factory only once and caches results for all subsequent imports until [`vi.unmock`](#vi-unmock) or [`vi.doUnmock`](#vi-dounmock) is called. -Unlike in `jest`, the factory can be asynchronous. You can use [`vi.importActual`](#vi-importactual) or a helper with the factory passed in as the first argument, and get the original module inside. Vitest also supports a module promise instead of a string in `vi.mock` method for better IDE support (when file is moved, path will be updated, `importOriginal` also inherits the type automatically). +Unlike in `jest`, the factory can be asynchronous. You can use [`vi.importActual`](#vi-importactual) or a helper with the factory passed in as the first argument, and get the original module inside. + +Vitest also supports a module promise instead of a string in the `vi.mock` and `vi.doMock` methods for better IDE support. When the file is moved, the path will be updated, and `importOriginal` also inherits the type automatically. Using this signature will also enforce factory return type to be compatible with the original module (but every export is optional). ```ts twoslash // @filename: ./path/to/module.js @@ -143,7 +145,7 @@ If there is no `__mocks__` folder or a factory provided, Vitest will import the ### vi.doMock - **Type**: `(path: string, factory?: (importOriginal: () => unknown) => unknown) => void` -- **Type**: `(path: Promise, factory?: (importOriginal: () => T) => unknown) => void` +- **Type**: `(path: Promise, factory?: (importOriginal: () => T) => T | Promise) => void` The same as [`vi.mock`](#vi-mock), but it's not hoisted to the top of the file, so you can reference variables in the global file scope. The next [dynamic import](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/import) of the module will be mocked. diff --git a/packages/vitest/src/types/mocker.ts b/packages/vitest/src/types/mocker.ts index fc64b140e85e..4937955f512b 100644 --- a/packages/vitest/src/types/mocker.ts +++ b/packages/vitest/src/types/mocker.ts @@ -1,6 +1,8 @@ +type Promisable = T | Promise + export type MockFactoryWithHelper = ( - importOriginal: () => Promise -) => any + importOriginal: () => Promise +) => Promisable> export type MockFactory = () => any export type MockMap = Map> diff --git a/test/config/test/workers-option.test.ts b/test/config/test/workers-option.test.ts index a31fa95f46bf..099cfca10064 100644 --- a/test/config/test/workers-option.test.ts +++ b/test/config/test/workers-option.test.ts @@ -4,6 +4,7 @@ import { getWorkersCountByPercentage } from 'vitest/src/utils/workers.js' import * as testUtils from '../../test-utils' vi.mock(import('node:os'), async importOriginal => ({ + ...(await importOriginal()), default: { ...(await importOriginal()).default, availableParallelism: () => 10, diff --git a/test/core/test/mocking/factory.test.ts b/test/core/test/mocking/factory.test.ts index 39183c9ae8f1..88c2113f0604 100644 --- a/test/core/test/mocking/factory.test.ts +++ b/test/core/test/mocking/factory.test.ts @@ -54,7 +54,7 @@ vi.mock('../../src/mocks/log.ts', async () => { }, } }) - +// @ts-expect-error null is not allowed to mocked implementation vi.mock('../../src/mocks/default.ts', () => null) describe('mocking with factory', () => {