From a0c1d371edb35abfa0e6f53b05086a54d2ecc0a6 Mon Sep 17 00:00:00 2001 From: Hiroshi Ogawa Date: Mon, 24 Jun 2024 01:09:21 +0900 Subject: [PATCH] fix!(spy): simplify mock function generic types and align with jest (#4784) --- docs/guide/migration.md | 24 ++- packages/spy/src/index.ts | 201 ++++++++---------- .../src/integrations/chai/chai-subset.d.ts | 2 +- packages/vitest/src/types/index.ts | 1 - test/core/test/mock-internals.test.ts | 4 +- test/core/test/vi.spec.ts | 55 ++++- 6 files changed, 159 insertions(+), 128 deletions(-) diff --git a/docs/guide/migration.md b/docs/guide/migration.md index 5b3a6f11dfe7..3196c577a7f6 100644 --- a/docs/guide/migration.md +++ b/docs/guide/migration.md @@ -78,6 +78,24 @@ This makes `.suite` optional; if the task is defined at the top level, it will n This change also removes the file from `expect.getState().currentTestName` and makes `expect.getState().testPath` required. +### Simplified generic types of mock functions (e.g. `vi.fn`, `Mock`) + +Previously `vi.fn` accepted two generic types separately for arguemnts and return value. This is changed to directly accept a function type `vi.fn` to simplify the usage. + +```ts +import { type Mock, vi } from 'vitest' + +const add = (x: number, y: number): number => x + y + +// using vi.fn +const mockAdd = vi.fn, ReturnType>() // [!code --] +const mockAdd = vi.fn() // [!code ++] + +// using Mock +const mockAdd: Mock, ReturnType> = vi.fn() // [!code --] +const mockAdd: Mock = vi.fn() // [!code ++] +``` + ## Migrating to Vitest 1.0 @@ -324,13 +342,11 @@ export default defineConfig({ Vitest doesn't have an equivalent to `jest` namespace, so you will need to import types directly from `vitest`: ```ts -let fn: jest.Mock // [!code --] +let fn: jest.Mock<(name: string) => number> // [!code --] import type { Mock } from 'vitest' // [!code ++] -let fn: Mock<[string], string> // [!code ++] +let fn: Mock<(name: string) => number> // [!code ++] ``` -Also, Vitest has `Args` type as a first argument instead of `Returns`, as you can see in diff. - ### Timers Vitest doesn't support Jest's legacy timers. diff --git a/packages/spy/src/index.ts b/packages/spy/src/index.ts index 6cb34d980ad8..98e8c4acb800 100644 --- a/packages/spy/src/index.ts +++ b/packages/spy/src/index.ts @@ -38,7 +38,7 @@ export type MockSettledResult = | MockSettledResultFulfilled | MockSettledResultRejected -export interface MockContext { +export interface MockContext { /** * This is an array containing all arguments for each call. One item of the array is the arguments of that call. * @@ -53,11 +53,11 @@ export interface MockContext { * ['arg3'], // second call * ] */ - calls: TArgs[] + calls: Parameters[] /** * This is an array containing all instances that were instantiated when mock was called with a `new` keyword. Note that this is an actual context (`this`) of the function, not a return value. */ - instances: TReturns[] + instances: ReturnType[] /** * The order of mock's execution. This returns an array of numbers which are shared between all defined mocks. * @@ -101,7 +101,7 @@ export interface MockContext { * }, * ] */ - results: MockResult[] + results: MockResult>[] /** * An array containing all values that were `resolved` or `rejected` from the function. * @@ -129,11 +129,11 @@ export interface MockContext { * }, * ] */ - settledResults: MockSettledResult>[] + settledResults: MockSettledResult>>[] /** * This contains the arguments of the last call. If spy wasn't called, will return `undefined`. */ - lastCall: TArgs | undefined + lastCall: Parameters | undefined } type Procedure = (...args: any[]) => any @@ -150,43 +150,49 @@ type Classes = { }[keyof T] & (string | symbol) -/** - * @deprecated Use MockInstance instead - */ -export interface SpyInstance - extends MockInstance {} +/* +cf. https://typescript-eslint.io/rules/method-signature-style/ -export interface MockInstance { +Typescript assignability is different between + { foo: (f: T) => U } (this is "method-signature-style") +and + { foo(f: T): U } + +Jest uses the latter for `MockInstance.mockImplementation` etc... and it allows assignment such as: + const boolFn: Jest.Mock<() => boolean> = jest.fn<() => true>(() => true) +*/ +/* eslint-disable ts/method-signature-style */ +export interface MockInstance { /** * Use it to return the name given to mock with method `.mockName(name)`. */ - getMockName: () => string + getMockName(): string /** * Sets internal mock name. Useful to see the name of the mock if an assertion fails. */ - mockName: (n: string) => this + mockName(n: string): this /** * Current context of the mock. It stores information about all invocation calls, instances, and results. */ - mock: MockContext + mock: MockContext /** * Clears all information about every call. After calling it, all properties on `.mock` will return an empty state. This method does not reset implementations. * * It is useful if you need to clean up mock between different assertions. */ - mockClear: () => this + mockClear(): this /** * Does what `mockClear` does and makes inner implementation an empty function (returning `undefined` when invoked). This also resets all "once" implementations. * * This is useful when you want to completely reset a mock to the default state. */ - mockReset: () => this + mockReset(): this /** * Does what `mockReset` does and restores inner implementation to the original function. * * Note that restoring mock from `vi.fn()` will set implementation to an empty function that returns `undefined`. Restoring a `vi.fn(impl)` will restore implementation to `impl`. */ - mockRestore: () => void + mockRestore(): void /** * Returns current mock implementation if there is one. * @@ -194,14 +200,14 @@ export interface MockInstance { * * If mock was created with `vi.spyOn`, it will return `undefined` unless a custom implementation was provided. */ - getMockImplementation: () => ((...args: TArgs) => TReturns) | undefined + getMockImplementation(): T | undefined /** * Accepts a function that will be used as an implementation of the mock. * @example * const increment = vi.fn().mockImplementation(count => count + 1); * expect(increment(3)).toBe(4); */ - mockImplementation: (fn: (...args: TArgs) => TReturns) => this + mockImplementation(fn: T): this /** * Accepts a function that will be used as a mock implementation during the next call. Can be chained so that multiple function calls produce different results. * @example @@ -209,7 +215,7 @@ export interface MockInstance { * expect(fn(3)).toBe(4); * expect(fn(3)).toBe(3); */ - mockImplementationOnce: (fn: (...args: TArgs) => TReturns) => this + mockImplementationOnce(fn: T): this /** * Overrides the original mock implementation temporarily while the callback is being executed. * @example @@ -221,18 +227,16 @@ export interface MockInstance { * * myMockFn() // 'original' */ - withImplementation: ( - fn: (...args: TArgs) => TReturns, - cb: () => T - ) => T extends Promise ? Promise : this + withImplementation(fn: T, cb: () => T2): T2 extends Promise ? Promise : this + /** * Use this if you need to return `this` context from the method without invoking actual implementation. */ - mockReturnThis: () => this + mockReturnThis(): this /** * Accepts a value that will be returned whenever the mock function is called. */ - mockReturnValue: (obj: TReturns) => this + mockReturnValue(obj: ReturnType): this /** * Accepts a value that will be returned during the next function call. If chained, every consecutive call will return the specified value. * @@ -247,14 +251,14 @@ export interface MockInstance { * // 'first call', 'second call', 'default' * console.log(myMockFn(), myMockFn(), myMockFn()) */ - mockReturnValueOnce: (obj: TReturns) => this + mockReturnValueOnce(obj: ReturnType): this /** * Accepts a value that will be resolved when async function is called. * @example * const asyncMock = vi.fn().mockResolvedValue(42) * asyncMock() // Promise<42> */ - mockResolvedValue: (obj: Awaited) => this + mockResolvedValue(obj: Awaited>): this /** * Accepts a value that will be resolved during the next function call. If chained, every consecutive call will resolve specified value. * @example @@ -267,14 +271,14 @@ export interface MockInstance { * // Promise<'first call'>, Promise<'second call'>, Promise<'default'> * console.log(myMockFn(), myMockFn(), myMockFn()) */ - mockResolvedValueOnce: (obj: Awaited) => this + mockResolvedValueOnce(obj: Awaited>): this /** * Accepts an error that will be rejected when async function is called. * @example * const asyncMock = vi.fn().mockRejectedValue(new Error('Async error')) * await asyncMock() // throws 'Async error' */ - mockRejectedValue: (obj: any) => this + mockRejectedValue(obj: any): this /** * Accepts a value that will be rejected during the next function call. If chained, every consecutive call will reject specified value. * @example @@ -286,52 +290,43 @@ export interface MockInstance { * await asyncMock() // first call * await asyncMock() // throws "Async error" */ - mockRejectedValueOnce: (obj: any) => this + mockRejectedValueOnce(obj: any): this } +/* eslint-enable ts/method-signature-style */ -export interface Mock - extends MockInstance { - new (...args: TArgs): TReturns - (...args: TArgs): TReturns +export interface Mock + extends MockInstance { + new (...args: Parameters): ReturnType + (...args: Parameters): ReturnType } -export interface PartialMock + +type PartialMaybePromise = T extends Promise> + ? Promise>> + : Partial + +export interface PartialMock extends MockInstance< - TArgs, - TReturns extends Promise> - ? Promise>> - : Partial + (...args: Parameters) => PartialMaybePromise> > { - new (...args: TArgs): TReturns - (...args: TArgs): TReturns + new (...args: Parameters): ReturnType + (...args: Parameters): ReturnType } export type MaybeMockedConstructor = T extends new ( ...args: Array ) => infer R - ? Mock, R> + ? Mock<(...args: ConstructorParameters) => R> : T -export type MockedFunction = Mock< - Parameters, - ReturnType -> & { +export type MockedFunction = Mock & { [K in keyof T]: T[K]; } -export type PartiallyMockedFunction = PartialMock< - Parameters, - ReturnType -> & { +export type PartiallyMockedFunction = PartialMock & { [K in keyof T]: T[K]; } -export type MockedFunctionDeep = Mock< - Parameters, - ReturnType -> & -MockedObjectDeep -export type PartiallyMockedFunctionDeep = PartialMock< - Parameters, - ReturnType -> & -MockedObjectDeep +export type MockedFunctionDeep = Mock & + MockedObjectDeep +export type PartiallyMockedFunctionDeep = PartialMock & + MockedObjectDeep export type MockedObject = MaybeMockedConstructor & { [K in Methods]: T[K] extends Procedure ? MockedFunction : T[K]; } & { [K in Properties]: T[K] } @@ -368,15 +363,14 @@ interface Constructable { } export type MockedClass = MockInstance< - T extends new (...args: infer P) => any ? P : never, - InstanceType + (...args: ConstructorParameters) => InstanceType > & { prototype: T extends { prototype: any } ? Mocked : never } & T export type Mocked = { - [P in keyof T]: T[P] extends (...args: infer Args) => infer Returns - ? MockInstance + [P in keyof T]: T[P] extends Procedure + ? MockInstance : T[P] extends Constructable ? MockedClass : T[P]; @@ -394,19 +388,19 @@ export function spyOn>>( obj: T, methodName: S, accessType: 'get' -): MockInstance<[], T[S]> +): MockInstance<() => T[S]> export function spyOn>>( obj: T, methodName: G, accessType: 'set' -): MockInstance<[T[G]], void> +): MockInstance<(arg: T[G]) => void> export function spyOn> | Methods>>( obj: T, methodName: M ): Required[M] extends | { new (...args: infer A): infer R } | ((...args: infer A) => infer R) - ? MockInstance + ? MockInstance<(...args: A) => R> : never export function spyOn( obj: T, @@ -426,19 +420,22 @@ export function spyOn( let callOrder = 0 -function enhanceSpy( - spy: SpyInternalImpl, -): MockInstance { - const stub = spy as unknown as MockInstance +function enhanceSpy( + spy: SpyInternalImpl, ReturnType>, +): MockInstance { + type TArgs = Parameters + type TReturns = ReturnType - let implementation: ((...args: TArgs) => TReturns) | undefined + const stub = spy as unknown as MockInstance + + let implementation: T | undefined let instances: any[] = [] let invocations: number[] = [] const state = tinyspy.getInternalState(spy) - const mockContext: MockContext = { + const mockContext: MockContext = { get calls() { return state.calls }, @@ -499,7 +496,7 @@ function enhanceSpy( stub.mockReset = () => { stub.mockClear() - implementation = () => undefined as unknown as TReturns + implementation = (() => undefined) as T onceImplementations = [] return stub } @@ -512,29 +509,20 @@ function enhanceSpy( } stub.getMockImplementation = () => implementation - stub.mockImplementation = (fn: (...args: TArgs) => TReturns) => { + stub.mockImplementation = (fn: T) => { implementation = fn state.willCall(mockCall) return stub } - stub.mockImplementationOnce = (fn: (...args: TArgs) => TReturns) => { + stub.mockImplementationOnce = (fn: T) => { onceImplementations.push(fn) return stub } - function withImplementation( - fn: (...args: TArgs) => TReturns, - cb: () => void - ): MockInstance - function withImplementation( - fn: (...args: TArgs) => TReturns, - cb: () => Promise - ): Promise> - function withImplementation( - fn: (...args: TArgs) => TReturns, - cb: () => void | Promise, - ): MockInstance | Promise> { + function withImplementation(fn: T, cb: () => void): MockInstance + function withImplementation(fn: T, cb: () => Promise): Promise> + function withImplementation(fn: T, cb: () => void | Promise): MockInstance | Promise> { const originalImplementation = implementation implementation = fn @@ -563,25 +551,24 @@ function enhanceSpy( stub.withImplementation = withImplementation stub.mockReturnThis = () => - stub.mockImplementation(function (this: TReturns) { + stub.mockImplementation((function (this: TReturns) { return this - }) + }) as any) - stub.mockReturnValue = (val: TReturns) => stub.mockImplementation(() => val) - stub.mockReturnValueOnce = (val: TReturns) => - stub.mockImplementationOnce(() => val) + stub.mockReturnValue = (val: TReturns) => stub.mockImplementation((() => val) as any) + stub.mockReturnValueOnce = (val: TReturns) => stub.mockImplementationOnce((() => val) as any) stub.mockResolvedValue = (val: Awaited) => - stub.mockImplementation(() => Promise.resolve(val as TReturns) as any) + stub.mockImplementation((() => Promise.resolve(val as TReturns)) as any) stub.mockResolvedValueOnce = (val: Awaited) => - stub.mockImplementationOnce(() => Promise.resolve(val as TReturns) as any) + stub.mockImplementationOnce((() => Promise.resolve(val as TReturns)) as any) stub.mockRejectedValue = (val: unknown) => - stub.mockImplementation(() => Promise.reject(val) as any) + stub.mockImplementation((() => Promise.reject(val)) as any) stub.mockRejectedValueOnce = (val: unknown) => - stub.mockImplementationOnce(() => Promise.reject(val) as any) + stub.mockImplementationOnce((() => Promise.reject(val)) as any) Object.defineProperty(stub, 'mock', { get: () => mockContext, @@ -594,19 +581,13 @@ function enhanceSpy( return stub as any } -export function fn(): Mock -export function fn( - implementation: (...args: TArgs) => R -): Mock -export function fn( - implementation?: (...args: TArgs) => R, -): Mock { - const enhancedSpy = enhanceSpy( - tinyspy.internalSpyOn({ spy: implementation || (() => {}) }, 'spy'), - ) +export function fn( + implementation?: T, +): Mock { + const enhancedSpy = enhanceSpy(tinyspy.internalSpyOn({ spy: implementation || (() => {}) }, 'spy')) if (implementation) { enhancedSpy.mockImplementation(implementation) } - return enhancedSpy as Mock + return enhancedSpy as any } diff --git a/packages/vitest/src/integrations/chai/chai-subset.d.ts b/packages/vitest/src/integrations/chai/chai-subset.d.ts index 2f362a62d233..0894478dd816 100644 --- a/packages/vitest/src/integrations/chai/chai-subset.d.ts +++ b/packages/vitest/src/integrations/chai/chai-subset.d.ts @@ -1,5 +1,5 @@ declare module 'chai-subset' { const chaiSubset: Chai.ChaiPlugin - // eslint-disable-next-line no-restricted-syntax + export = chaiSubset } diff --git a/packages/vitest/src/types/index.ts b/packages/vitest/src/types/index.ts index f7d52bd5465b..eb9fe39b22e6 100644 --- a/packages/vitest/src/types/index.ts +++ b/packages/vitest/src/types/index.ts @@ -18,7 +18,6 @@ export type { DiffOptions } from '@vitest/utils/diff' export type { MockedFunction, MockedObject, - SpyInstance, MockInstance, Mock, MockContext, diff --git a/test/core/test/mock-internals.test.ts b/test/core/test/mock-internals.test.ts index ea3ac225d36f..246c691093c1 100644 --- a/test/core/test/mock-internals.test.ts +++ b/test/core/test/mock-internals.test.ts @@ -1,6 +1,6 @@ import childProcess, { exec } from 'node:child_process' import timers from 'node:timers' -import { type SpyInstance, afterEach, beforeEach, describe, expect, test, vi } from 'vitest' +import { type MockInstance, afterEach, beforeEach, describe, expect, test, vi } from 'vitest' import { execDefault, execHelloWorld, execImportAll } from '../src/exec' import { dynamicImport } from '../src/dynamic-import' @@ -31,7 +31,7 @@ test('mocked dynamically imported packages', async () => { describe('Math.random', () => { describe('mock is restored', () => { - let spy: SpyInstance + let spy: MockInstance beforeEach(() => { spy = vi.spyOn(Math, 'random').mockReturnValue(0.1) diff --git a/test/core/test/vi.spec.ts b/test/core/test/vi.spec.ts index 79ea9ad5b910..79f0164efc3d 100644 --- a/test/core/test/vi.spec.ts +++ b/test/core/test/vi.spec.ts @@ -2,8 +2,8 @@ * @vitest-environment jsdom */ -import type { MockedFunction, MockedObject } from 'vitest' -import { describe, expect, test, vi } from 'vitest' +import type { Mock, MockedFunction, MockedObject } from 'vitest' +import { describe, expect, expectTypeOf, test, vi } from 'vitest' import { getWorkerState } from '../../../packages/vitest/src/utils' function expectType(obj: T) { @@ -41,6 +41,12 @@ describe('testing vi utils', () => { }) expectType boolean>>(vi.fn(() => true)) expectType boolean>>(vi.fn()) + + expectType boolean>>(vi.fn<() => boolean>(() => true)) + expectType boolean>>(vi.fn<() => boolean>(() => true)) + expectType<() => boolean>(vi.fn(() => true)) + + expectType<(v: number) => boolean>(vi.fn()) }) test('vi partial mocked', () => { @@ -50,32 +56,28 @@ describe('testing vi utils', () => { baz: string } - type FooBarFactory = () => FooBar - - const mockFactory: FooBarFactory = vi.fn() + const mockFactory = vi.fn<() => FooBar>() vi.mocked(mockFactory, { partial: true }).mockReturnValue({ foo: vi.fn(), }) vi.mocked(mockFactory, { partial: true, deep: false }).mockReturnValue({ - bar: vi.fn(), + bar: vi.fn(), }) vi.mocked(mockFactory, { partial: true, deep: true }).mockReturnValue({ baz: 'baz', }) - type FooBarAsyncFactory = () => Promise - - const mockFactoryAsync: FooBarAsyncFactory = vi.fn() + const mockFactoryAsync = vi.fn<() => Promise>() vi.mocked(mockFactoryAsync, { partial: true }).mockResolvedValue({ foo: vi.fn(), }) vi.mocked(mockFactoryAsync, { partial: true, deep: false }).mockResolvedValue({ - bar: vi.fn(), + bar: vi.fn(), }) vi.mocked(mockFactoryAsync, { partial: true, deep: true }).mockResolvedValue({ @@ -83,6 +85,39 @@ describe('testing vi utils', () => { }) }) + test('vi.fn and Mock type', () => { + // use case from https://github.com/vitest-dev/vitest/issues/4723#issuecomment-1851034249 + + // hypotetical library to be tested + type SomeFn = (v: string) => number + function acceptSomeFn(f: SomeFn) { + f('hi') + } + + // SETUP + // no args are allowed even though it's not type safe + const someFn1: Mock = vi.fn() + + // argument types are infered + const someFn2: Mock = vi.fn((v) => { + expectTypeOf(v).toEqualTypeOf() + return 0 + }) + + // arguments are not necessary + const someFn3: Mock = vi.fn(() => 0) + + // @ts-expect-error wrong return type will be caught + const someFn4: Mock = vi.fn(() => '0') + + // TEST + acceptSomeFn(someFn1) + expect(someFn1).toBeCalledWith('hi') + expect(someFn2).not.toBeCalled() + expect(someFn3).not.toBeCalled() + expect(someFn4).not.toBeCalled() + }) + test('can change config', () => { const state = getWorkerState() expect(state.config.hookTimeout).toBe(10000)