From cd68f95860b64049d63336cd25094e85cb617b54 Mon Sep 17 00:00:00 2001 From: Andrew Scott Date: Fri, 20 Sep 2024 13:41:03 -0700 Subject: [PATCH] fixup! feat(jest-fake-timers): Add feature to enable automatically advancing timers --- docs/JestObjectAPI.md | 30 ++-- packages/jest-environment/src/index.ts | 32 ++-- .../src/__tests__/modernFakeTimers.test.ts | 140 +++++++++--------- .../jest-fake-timers/src/modernFakeTimers.ts | 19 ++- 4 files changed, 124 insertions(+), 97 deletions(-) diff --git a/docs/JestObjectAPI.md b/docs/JestObjectAPI.md index 810cf9772522..082916391e5e 100644 --- a/docs/JestObjectAPI.md +++ b/docs/JestObjectAPI.md @@ -1037,6 +1037,21 @@ Advances all timers by the needed milliseconds so that only the next timeouts/in Optionally, you can provide `steps`, so it will run `steps` amount of next timeouts/intervals. +### `jest.advanceTimersToNextTimerAsync(mode)` + +Configures whether timers advance automatically. When 'auto', jest will advance the clock to the next timer in the queue after a macrotask. With automatically advancing timers enabled, tests can be written in a way that is independent from whether fake timers are installed. Tests can always be written to wait for timers to resolve, even when using fake timers. + +This feature differs from the `advanceTimers` in two key ways: + +1. The microtask queue is allowed to empty between each timer execution, as would be the case without fake timers installed. +1. It advances as quickly and as far as necessary. If the next timer in the queue is at 1000ms, it will advance 1000ms immediately whereas `advanceTimers`, without manually advancing time in the test, would take `1000 / advanceTimersMs` real time to reach and execute the timer. + +:::info + +This function is not available when using legacy fake timers implementation. + +::: + ### `jest.advanceTimersToNextTimerAsync(steps)` Asynchronous equivalent of `jest.advanceTimersToNextTimer(steps)`. It allows any scheduled promise callbacks to execute _before_ running the timers. @@ -1067,25 +1082,10 @@ This means, if any timers have been scheduled (but have not yet executed), they Returns the number of fake timers still left to run. -### `jest.setAdvanceTimersAutomatically()` - -Configures whether timers advance automatically. When enabled, jest will advance the clock to the next timer in the queue after a macrotask. With automatically advancing timers enabled, tests can be written in a way that is independent from whether fake timers are installed. Tests can always be written to wait for timers to resolve, even when using fake timers. - -This feature differs from the `advanceTimers` in two key ways: - -1. The microtask queue is allowed to empty between each timer execution, as would be the case without fake timers installed. -1. It advances as quickly and as far as necessary. If the next timer in the queue is at 1000ms, it will advance 1000ms immediately whereas `advanceTimers`, without manually advancing time in the test, would take `1000 / advanceTimersMs` real time to reach and execute the timer. - ### `jest.now()` Returns the time in ms of the current clock. This is equivalent to `Date.now()` if real timers are in use, or if `Date` is mocked. In other cases (such as legacy timers) it may be useful for implementing custom mocks of `Date.now()`, `performance.now()`, etc. -:::info - -This function is not available when using legacy fake timers implementation. - -::: - ### `jest.setSystemTime(now?: number | Date)` Set the current system time used by fake timers. Simulates a user changing the system clock while your program is running. It affects the current time but it does not in itself cause e.g. timers to fire; they will fire exactly as they would have done without the call to `jest.setSystemTime()`. diff --git a/packages/jest-environment/src/index.ts b/packages/jest-environment/src/index.ts index d9bb2b844a3c..e2440613a877 100644 --- a/packages/jest-environment/src/index.ts +++ b/packages/jest-environment/src/index.ts @@ -6,7 +6,11 @@ */ import type {Context} from 'vm'; -import type {LegacyFakeTimers, ModernFakeTimers} from '@jest/fake-timers'; +import type { + LegacyFakeTimers, + ModernFakeTimers, + TimerTickMode, +} from '@jest/fake-timers'; import type {Circus, Config, Global} from '@jest/types'; import type {Mocked, ModuleMocker} from 'jest-mock'; @@ -82,24 +86,32 @@ export interface Jest { */ advanceTimersToNextTimer(steps?: number): void; /** - * Advances the clock to the the moment of the first scheduled timer, firing it. - * Optionally, you can provide steps, so it will run steps amount of - * next timeouts/intervals. + * Updates the behavior of the timer advancement without. + * + * When 'automatic', configures whether timers advance automatically. With automatically advancing + * timers enabled, tests can be written in a way that is independent from whether + * fake timers are installed. Tests can always be written to wait for timers to + * resolve, even when using fake timers. + * + * When 'manual' (the default), timers will not advance automatically. Instead, + * timers must be advanced using APIs such as `advanceTimersToNextTimer`, `advanceTimersByTime`, etc. * * @remarks * Not available when using legacy fake timers implementation. */ - advanceTimersToNextTimerAsync(steps?: number): Promise; + advanceTimersToNextTimerAsync(tickMode: TimerTickMode): void; /** - * Configures whether timers advance automatically. With automatically advancing - * timers enabled, tests can be written in a way that is independent from whether - * fake timers are installed. Tests can always be written to wait for timers to - * resolve, even when using fake timers. + * Advances the clock to the the moment of the first scheduled timer, firing it. + * Optionally, you can provide steps, so it will run steps amount of + * next timeouts/intervals. * * @remarks * Not available when using legacy fake timers implementation. */ - setAdvanceTimersAutomatically(autoAdvance: boolean): void; + advanceTimersToNextTimerAsync(steps?: number): Promise; + advanceTimersToNextTimerAsync( + stepsOrTickMode?: number | TimerTickMode, + ): Promise|void; /** * Disables automatic mocking in the module loader. */ diff --git a/packages/jest-fake-timers/src/__tests__/modernFakeTimers.test.ts b/packages/jest-fake-timers/src/__tests__/modernFakeTimers.test.ts index 12b677042c14..1c089dced2c1 100644 --- a/packages/jest-fake-timers/src/__tests__/modernFakeTimers.test.ts +++ b/packages/jest-fake-timers/src/__tests__/modernFakeTimers.test.ts @@ -1249,6 +1249,78 @@ describe('FakeTimers', () => { expect(timers.now()).toBe(200); expect(spy).toHaveBeenCalled(); }); + + describe('auto advance', () => { + let global: typeof globalThis; + let timers: FakeTimers; + beforeEach(() => { + global = { + Date, + Promise, + clearTimeout, + process, + setTimeout, + } as unknown as typeof globalThis; + + timers = new FakeTimers({config: makeProjectConfig(), global}); + + timers.useFakeTimers(); + timers.advanceTimersToNextTimerAsync('auto'); + }); + + afterEach(() => { + timers.dispose(); + }); + + it('can always wait for a timer to execute', async () => { + const p = new Promise(resolve => { + global.setTimeout(resolve, 100); + }); + await expect(p).resolves.toBeUndefined(); + }); + + it('can mix promises inside timers', async () => { + const p = new Promise(resolve => + global.setTimeout(async () => { + await Promise.resolve(); + global.setTimeout(resolve, 100); + }, 100), + ); + await expect(p).resolves.toBeUndefined(); + }); + + it('automatically advances all timers', async () => { + const p1 = new Promise(resolve => global.setTimeout(resolve, 50)); + const p2 = new Promise(resolve => global.setTimeout(resolve, 50)); + const p3 = new Promise(resolve => global.setTimeout(resolve, 100)); + await expect(Promise.all([p1, p2, p3])).resolves.toEqual([ + undefined, + undefined, + undefined, + ]); + }); + + it('can turn off and on auto advancing of time', async () => { + let p2Resolved = false; + const p1 = new Promise(resolve => global.setTimeout(resolve, 50)); + const p2 = new Promise(resolve => global.setTimeout(resolve, 51)).then( + () => (p2Resolved = true), + ); + const p3 = new Promise(resolve => global.setTimeout(resolve, 52)); + + await expect(p1).resolves.toBeUndefined(); + + timers.advanceTimersToNextTimerAsync('manual'); + await new Promise(resolve => setTimeout(resolve, 5)); + expect(p2Resolved).toBe(false); + + timers.advanceTimersToNextTimerAsync('auto'); + await new Promise(resolve => setTimeout(resolve, 5)); + await expect(p2).resolves.toBe(true); + await expect(p3).resolves.toBeUndefined(); + expect(p2Resolved).toBe(true); + }); + }); }); describe('runAllTimersAsync', () => { @@ -1330,74 +1402,6 @@ describe('FakeTimers', () => { }); }); - describe('setAdvanceTimersAutomatically', () => { - let global: typeof globalThis; - let timers: FakeTimers; - beforeEach(() => { - global = { - Date, - Promise, - clearTimeout, - process, - setTimeout, - } as unknown as typeof globalThis; - - timers = new FakeTimers({config: makeProjectConfig(), global}); - - timers.useFakeTimers(); - timers.setAdvanceTimersAutomatically(true); - }); - - it('can always wait for a timer to execute', async () => { - const p = new Promise(resolve => { - global.setTimeout(resolve, 100); - }); - await expect(p).resolves.toBeUndefined(); - }); - - it('can mix promises inside timers', async () => { - const p = new Promise(resolve => - global.setTimeout(async () => { - await Promise.resolve(); - global.setTimeout(resolve, 100); - }, 100), - ); - await expect(p).resolves.toBeUndefined(); - }); - - it('automatically advances all timers', async () => { - const p1 = new Promise(resolve => global.setTimeout(resolve, 50)); - const p2 = new Promise(resolve => global.setTimeout(resolve, 50)); - const p3 = new Promise(resolve => global.setTimeout(resolve, 100)); - await expect(Promise.all([p1, p2, p3])).resolves.toEqual([ - undefined, - undefined, - undefined, - ]); - }); - - it('can turn off and on auto advancing of time', async () => { - let p2Resolved = false; - const p1 = new Promise(resolve => global.setTimeout(resolve, 50)); - const p2 = new Promise(resolve => global.setTimeout(resolve, 51)).then( - () => (p2Resolved = true), - ); - const p3 = new Promise(resolve => global.setTimeout(resolve, 52)); - - await expect(p1).resolves.toBeUndefined(); - - timers.setAdvanceTimersAutomatically(false); - await new Promise(resolve => setTimeout(resolve, 5)); - expect(p2Resolved).toBe(false); - - timers.setAdvanceTimersAutomatically(true); - await new Promise(resolve => setTimeout(resolve, 5)); - await expect(p2).resolves.toBe(true); - await expect(p3).resolves.toBeUndefined(); - expect(p2Resolved).toBe(true); - }); - }); - describe('now', () => { let timers: FakeTimers; let fakedGlobal: typeof globalThis; diff --git a/packages/jest-fake-timers/src/modernFakeTimers.ts b/packages/jest-fake-timers/src/modernFakeTimers.ts index 2454ee90a7bc..3a9d183ff805 100644 --- a/packages/jest-fake-timers/src/modernFakeTimers.ts +++ b/packages/jest-fake-timers/src/modernFakeTimers.ts @@ -88,7 +88,19 @@ export default class FakeTimers { } } - async advanceTimersToNextTimerAsync(steps = 1): Promise { + advanceTimersToNextTimerAsync(mode: 'auto' | 'manual'): void; + advanceTimersToNextTimerAsync(steps?: number): Promise; + advanceTimersToNextTimerAsync( + stepsOrMode: number | 'auto' | 'manual' = 1, + ): Promise | void { + if (typeof stepsOrMode === 'number') { + return this._advanceTimersToNextTimerAsync(stepsOrMode); + } else { + this._setTickMode(stepsOrMode); + } + } + + private async _advanceTimersToNextTimerAsync(steps: number): Promise { if (this._checkFakeTimers()) { for (let i = steps; i > 0; i--) { await this._clock.nextAsync(); @@ -146,18 +158,17 @@ export default class FakeTimers { this._fakingTime = true; } - setAdvanceTimersAutomatically(autoAdvance: boolean): void { + private _setTickMode(newMode: 'manual' | 'auto'): void { if (!this._checkFakeTimers()) { return; } - const newMode = autoAdvance ? 'auto' : 'manual'; if (newMode === this.autoTickMode.mode) { return; } this.autoTickMode = {counter: this.autoTickMode.counter + 1, mode: newMode}; - if (autoAdvance) { + if (newMode === 'auto') { this._advanceUntilModeChanges(); } }