-
Notifications
You must be signed in to change notification settings - Fork 634
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(async/unstable): add
waitFor
function to wait for condition to…
… be true (#6230)
- Loading branch information
Showing
4 changed files
with
94 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,63 @@ | ||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. | ||
// This module is browser compatible. | ||
|
||
import { deadline } from "./deadline.ts"; | ||
|
||
/** Options for {@linkcode waitFor}. */ | ||
export interface WaitForOptions { | ||
/** Signal used to abort the waitFor. */ | ||
signal?: AbortSignal; | ||
/** Indicates the step jump in time to wait for the predicate to be true. | ||
* | ||
* @default {100} | ||
*/ | ||
step?: number; | ||
} | ||
|
||
/** | ||
* Resolve a {@linkcode Promise} after a given predicate becomes true or the | ||
* timeout amount of milliseconds has been reached. | ||
* | ||
* @throws {DOMException} If signal is aborted before either the waitFor | ||
* predicate is true or the timeout duration was reached, and `signal.reason` | ||
* is undefined. | ||
* @param predicate a Nullary (no arguments) function returning a boolean | ||
* @param ms Duration in milliseconds for how long the waitFor should last. | ||
* @param options Additional options. | ||
* | ||
* @example Basic usage | ||
* ```ts ignore | ||
* import { waitFor } from "@std/async/unstable-wait-for"; | ||
* | ||
* // Deno server to acknowledge reception of request/webhook | ||
* let requestReceived = false; | ||
* Deno.serve((_req) => { | ||
* requestReceived = true; | ||
* return new Response("Hello, world"); | ||
* }); | ||
* | ||
* // ... | ||
* waitFor(() => requestReceived, 10000); | ||
* // If less than 10 seconds pass, the requestReceived flag will be true | ||
* // assert(requestReceived); | ||
* // ... | ||
* ``` | ||
*/ | ||
export function waitFor( | ||
predicate: () => boolean | Promise<boolean>, | ||
ms: number, | ||
options: WaitForOptions = {}, | ||
): Promise<void> { | ||
const { step = 100 } = options; | ||
|
||
// Create a new promise that resolves when the predicate is true | ||
let interval: number; | ||
const p: Promise<void> = new Promise(function (resolve) { | ||
interval = setInterval(() => { | ||
if (predicate()) resolve(); | ||
}, step); | ||
}); | ||
|
||
// Return a deadline promise | ||
return deadline(p, ms, options).finally(() => clearInterval(interval)); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. | ||
import { assertAlmostEquals, assertEquals, assertRejects } from "@std/assert"; | ||
import { waitFor } from "./unstable_wait_for.ts"; | ||
|
||
// NOT detecting leaks means that the internal interval was correctly cleared | ||
|
||
Deno.test("waitFor() returns fulfilled promise", async () => { | ||
let flag = false; | ||
setTimeout(() => flag = true, 100); | ||
const start = Date.now(); | ||
await waitFor(() => flag === true, 1000); | ||
// Expects the promise to be resolved after 100ms | ||
assertAlmostEquals(Date.now() - start, 100, 10); | ||
}); | ||
|
||
Deno.test("waitFor() throws DOMException on timeout", async () => { | ||
let flag = false; | ||
const id = setTimeout(() => flag = true, 1000); | ||
const start = Date.now(); | ||
const error = await assertRejects( | ||
() => waitFor(() => flag === true, 100), | ||
DOMException, | ||
"Signal timed out.", | ||
); | ||
assertAlmostEquals(Date.now() - start, 100, 10); | ||
assertEquals(error.name, "TimeoutError"); | ||
clearTimeout(id); | ||
}); |