2022-01-07 22:09:52 -05:00
// Copyright 2018-2022 the Deno authors. All rights reserved. MIT license.
2021-02-05 03:48:32 +05:30
"use strict" ;
2020-07-19 19:49:44 +02:00
( ( window ) => {
2020-09-17 18:09:50 +02:00
const core = window . Deno . core ;
2021-05-19 00:24:01 +09:00
const { setExitHandler } = window . _ _bootstrap . os ;
2020-07-19 19:49:44 +02:00
const { Console , inspectArgs } = window . _ _bootstrap . console ;
2021-10-10 17:20:30 +02:00
const { metrics } = core ;
2021-10-13 18:04:44 +01:00
const { serializePermissions } = window . _ _bootstrap . permissions ;
2020-07-19 19:49:44 +02:00
const { assert } = window . _ _bootstrap . util ;
2021-07-04 00:17:52 +02:00
const {
2021-11-20 15:43:40 +01:00
AggregateError ,
2021-07-04 00:17:52 +02:00
ArrayPrototypeFilter ,
ArrayPrototypePush ,
2021-12-11 00:30:46 +01:00
ArrayPrototypeShift ,
2021-10-11 09:45:02 -04:00
ArrayPrototypeSome ,
2021-07-04 00:17:52 +02:00
DateNow ,
2021-10-11 09:45:02 -04:00
Error ,
Function ,
2022-01-25 17:03:38 +01:00
Number ,
ObjectKeys ,
2021-07-04 00:17:52 +02:00
Promise ,
2022-01-25 17:03:38 +01:00
RegExp ,
RegExpPrototypeTest ,
Set ,
2021-07-04 00:17:52 +02:00
StringPrototypeEndsWith ,
StringPrototypeIncludes ,
StringPrototypeSlice ,
2022-01-25 17:03:38 +01:00
StringPrototypeStartsWith ,
2021-10-11 09:45:02 -04:00
SymbolToStringTag ,
2022-01-25 17:03:38 +01:00
TypeError ,
2021-07-04 00:17:52 +02:00
} = window . _ _bootstrap . primordials ;
2020-07-19 19:49:44 +02:00
2021-12-11 00:30:46 +01:00
const opSanitizerDelayResolveQueue = [ ] ;
2021-11-30 01:27:30 +01:00
// Even if every resource is closed by the end of a test, there can be a delay
// until the pending ops have all finished. This function returns a promise
// that resolves when it's (probably) fine to run the op sanitizer.
//
// This is implemented by adding a macrotask callback that runs after the
// timer macrotasks, so we can guarantee that a currently running interval
// will have an associated op. An additional `setTimeout` of 0 is needed
// before that, though, in order to give time for worker message ops to finish
// (since timeouts of 0 don't queue tasks in the timer queue immediately).
function opSanitizerDelay ( ) {
2021-12-11 00:30:46 +01:00
return new Promise ( ( resolve ) => {
2021-11-30 01:27:30 +01:00
setTimeout ( ( ) => {
2021-12-11 00:30:46 +01:00
ArrayPrototypePush ( opSanitizerDelayResolveQueue , resolve ) ;
2021-11-30 01:27:30 +01:00
} , 0 ) ;
} ) ;
}
function handleOpSanitizerDelayMacrotask ( ) {
2021-12-11 00:30:46 +01:00
ArrayPrototypeShift ( opSanitizerDelayResolveQueue ) ? . ( ) ;
return opSanitizerDelayResolveQueue . length === 0 ;
2021-11-30 01:27:30 +01:00
}
2020-07-19 19:49:44 +02:00
// Wrap test function in additional assertion that makes sure
// the test case does not leak async "ops" - ie. number of async
// completed ops after the test is the same as number of dispatched
// ops. Note that "unref" ops are ignored since in nature that are
// optional.
function assertOps ( fn ) {
2021-10-12 09:58:04 -04:00
/** @param step {TestStep} */
return async function asyncOpSanitizer ( step ) {
2020-07-19 19:49:44 +02:00
const pre = metrics ( ) ;
2021-02-22 00:21:25 +08:00
try {
2021-10-12 09:58:04 -04:00
await fn ( step ) ;
2021-02-22 00:21:25 +08:00
} finally {
// Defer until next event loop turn - that way timeouts and intervals
// cleared can actually be removed from resource table, otherwise
// false positives may occur (https://github.com/denoland/deno/issues/4591)
2021-11-30 01:27:30 +01:00
await opSanitizerDelay ( ) ;
2021-02-22 00:21:25 +08:00
}
2021-04-29 02:17:04 +08:00
2021-10-13 08:55:12 -04:00
if ( step . shouldSkipSanitizers ) {
return ;
2021-10-12 09:58:04 -04:00
}
2020-07-19 19:49:44 +02:00
const post = metrics ( ) ;
2021-10-11 23:00:33 +08:00
2020-07-19 19:49:44 +02:00
// We're checking diff because one might spawn HTTP server in the background
// that will be a pending async op before test starts.
const dispatchedDiff = post . opsDispatchedAsync - pre . opsDispatchedAsync ;
const completedDiff = post . opsCompletedAsync - pre . opsCompletedAsync ;
2021-10-11 23:00:33 +08:00
const details = [ ] ;
for ( const key in post . ops ) {
const dispatchedDiff = Number (
post . ops [ key ] ? . opsDispatchedAsync -
( pre . ops [ key ] ? . opsDispatchedAsync ? ? 0 ) ,
) ;
const completedDiff = Number (
post . ops [ key ] ? . opsCompletedAsync -
( pre . ops [ key ] ? . opsCompletedAsync ? ? 0 ) ,
) ;
if ( dispatchedDiff !== completedDiff ) {
details . push ( `
$ { key } :
Before :
- dispatched : $ { pre . ops [ key ] ? . opsDispatchedAsync ? ? 0 }
- completed : $ { pre . ops [ key ] ? . opsCompletedAsync ? ? 0 }
After :
- dispatched : $ { post . ops [ key ] . opsDispatchedAsync }
- completed : $ { post . ops [ key ] . opsCompletedAsync } ` );
}
}
const message = ` Test case is leaking async ops.
2020-07-19 19:49:44 +02:00
Before :
- dispatched : $ { pre . opsDispatchedAsync }
- completed : $ { pre . opsCompletedAsync }
After :
- dispatched : $ { post . opsDispatchedAsync }
- completed : $ { post . opsCompletedAsync }
2021-10-11 23:00:33 +08:00
$ { details . length > 0 ? "Ops:" + details . join ( "" ) : "" }
2020-07-19 19:49:44 +02:00
Make sure to await all promises returned from Deno APIs before
2021-10-11 23:00:33 +08:00
finishing test case . ` ;
assert (
dispatchedDiff === completedDiff ,
message ,
2020-07-19 19:49:44 +02:00
) ;
} ;
}
2022-01-25 17:03:38 +01:00
function prettyResourceNames ( name ) {
switch ( name ) {
case "fsFile" :
return [ "A file" , "opened" , "closed" ] ;
case "fetchRequest" :
return [ "A fetch request" , "started" , "finished" ] ;
case "fetchRequestBody" :
return [ "A fetch request body" , "created" , "closed" ] ;
case "fetchResponseBody" :
return [ "A fetch response body" , "created" , "consumed" ] ;
case "httpClient" :
return [ "An HTTP client" , "created" , "closed" ] ;
case "dynamicLibrary" :
return [ "A dynamic library" , "loaded" , "unloaded" ] ;
case "httpConn" :
return [ "An inbound HTTP connection" , "accepted" , "closed" ] ;
case "httpStream" :
return [ "An inbound HTTP request" , "accepted" , "closed" ] ;
case "tcpStream" :
return [ "A TCP connection" , "opened/accepted" , "closed" ] ;
case "unixStream" :
return [ "A Unix connection" , "opened/accepted" , "closed" ] ;
case "tlsStream" :
return [ "A TLS connection" , "opened/accepted" , "closed" ] ;
case "tlsListener" :
return [ "A TLS listener" , "opened" , "closed" ] ;
case "unixListener" :
return [ "A Unix listener" , "opened" , "closed" ] ;
case "unixDatagram" :
return [ "A Unix datagram" , "opened" , "closed" ] ;
case "tcpListener" :
return [ "A TCP listener" , "opened" , "closed" ] ;
case "udpSocket" :
return [ "A UDP socket" , "opened" , "closed" ] ;
case "timer" :
return [ "A timer" , "started" , "fired/cleared" ] ;
case "textDecoder" :
return [ "A text decoder" , "created" , "finsihed" ] ;
case "messagePort" :
return [ "A message port" , "created" , "closed" ] ;
case "webSocketStream" :
return [ "A WebSocket" , "opened" , "closed" ] ;
case "fsEvents" :
return [ "A file system watcher" , "created" , "closed" ] ;
case "childStdin" :
return [ "A child process stdin" , "opened" , "closed" ] ;
case "childStdout" :
return [ "A child process stdout" , "opened" , "closed" ] ;
case "childStderr" :
return [ "A child process stderr" , "opened" , "closed" ] ;
case "child" :
return [ "A child process" , "started" , "closed" ] ;
case "signal" :
return [ "A signal listener" , "created" , "fired/cleared" ] ;
case "stdin" :
return [ "The stdin pipe" , "opened" , "closed" ] ;
case "stdout" :
return [ "The stdout pipe" , "opened" , "closed" ] ;
case "stderr" :
return [ "The stderr pipe" , "opened" , "closed" ] ;
case "compression" :
return [ "A CompressionStream" , "created" , "closed" ] ;
default :
return [ ` A " ${ name } " resource ` , "created" , "cleaned up" ] ;
}
}
function resourceCloseHint ( name ) {
switch ( name ) {
case "fsFile" :
return "Close the file handle by calling `file.close()`." ;
case "fetchRequest" :
return "Await the promise returned from `fetch()` or abort the fetch with an abort signal." ;
case "fetchRequestBody" :
return "Terminate the request body `ReadableStream` by closing or erroring it." ;
case "fetchResponseBody" :
return "Consume or close the response body `ReadableStream`, e.g `await resp.text()` or `await resp.body.cancel()`." ;
case "httpClient" :
return "Close the HTTP client by calling `httpClient.close()`." ;
case "dynamicLibrary" :
return "Unload the dynamic library by calling `dynamicLibrary.close()`." ;
case "httpConn" :
return "Close the inbound HTTP connection by calling `httpConn.close()`." ;
case "httpStream" :
return "Close the inbound HTTP request by responding with `e.respondWith().` or closing the HTTP connection." ;
case "tcpStream" :
return "Close the TCP connection by calling `tcpConn.close()`." ;
case "unixStream" :
return "Close the Unix socket connection by calling `unixConn.close()`." ;
case "tlsStream" :
return "Close the TLS connection by calling `tlsConn.close()`." ;
case "tlsListener" :
return "Close the TLS listener by calling `tlsListener.close()`." ;
case "unixListener" :
return "Close the Unix socket listener by calling `unixListener.close()`." ;
case "unixDatagram" :
return "Close the Unix datagram socket by calling `unixDatagram.close()`." ;
case "tcpListener" :
return "Close the TCP listener by calling `tcpListener.close()`." ;
case "udpSocket" :
return "Close the UDP socket by calling `udpSocket.close()`." ;
case "timer" :
return "Clear the timer by calling `clearInterval` or `clearTimeout`." ;
case "textDecoder" :
return "Close the text decoder by calling `textDecoder.decode('')` or `await textDecoderStream.readable.cancel()`." ;
case "messagePort" :
return "Close the message port by calling `messagePort.close()`." ;
case "webSocketStream" :
return "Close the WebSocket by calling `webSocket.close()`." ;
case "fsEvents" :
return "Close the file system watcher by calling `watcher.close()`." ;
case "childStdin" :
return "Close the child process stdin by calling `proc.stdin.close()`." ;
case "childStdout" :
return "Close the child process stdout by calling `proc.stdout.close()`." ;
case "childStderr" :
return "Close the child process stderr by calling `proc.stderr.close()`." ;
case "child" :
return "Close the child process by calling `proc.kill()` or `proc.close()`." ;
case "signal" :
return "Clear the signal listener by calling `Deno.removeSignalListener`." ;
case "stdin" :
return "Close the stdin pipe by calling `Deno.stdin.close()`." ;
case "stdout" :
return "Close the stdout pipe by calling `Deno.stdout.close()`." ;
case "stderr" :
return "Close the stderr pipe by calling `Deno.stderr.close()`." ;
case "compression" :
return "Close the compression stream by calling `await stream.writable.close()`." ;
default :
return "Close the resource before the end of the test." ;
}
}
2020-07-19 19:49:44 +02:00
// Wrap test function in additional assertion that makes sure
// the test case does not "leak" resources - ie. resource table after
// the test has exactly the same contents as before the test.
function assertResources (
fn ,
) {
2021-10-12 09:58:04 -04:00
/** @param step {TestStep} */
return async function resourceSanitizer ( step ) {
2020-09-17 18:09:50 +02:00
const pre = core . resources ( ) ;
2021-10-12 09:58:04 -04:00
await fn ( step ) ;
2021-10-13 08:55:12 -04:00
if ( step . shouldSkipSanitizers ) {
return ;
2021-10-12 09:58:04 -04:00
}
2020-09-17 18:09:50 +02:00
const post = core . resources ( ) ;
2020-07-19 19:49:44 +02:00
2022-01-25 17:03:38 +01:00
const allResources = new Set ( [ ... ObjectKeys ( pre ) , ... ObjectKeys ( post ) ] ) ;
2020-07-19 19:49:44 +02:00
2022-01-25 17:03:38 +01:00
const details = [ ] ;
for ( const resource of allResources ) {
const preResource = pre [ resource ] ;
const postResource = post [ resource ] ;
if ( preResource === postResource ) continue ;
if ( preResource === undefined ) {
const [ name , action1 , action2 ] = prettyResourceNames ( postResource ) ;
const hint = resourceCloseHint ( postResource ) ;
const detail =
` ${ name } (rid ${ resource } ) was ${ action1 } during the test, but not ${ action2 } during the test. ${ hint } ` ;
details . push ( detail ) ;
} else {
const [ name , action1 , action2 ] = prettyResourceNames ( preResource ) ;
const detail =
` ${ name } (rid ${ resource } ) was ${ action1 } before the test started, but was ${ action2 } during the test. Do not close resources in a test that were not created during that test. ` ;
details . push ( detail ) ;
}
}
const message = ` Test case is leaking ${ details . length } resource ${
details . length === 1 ? "" : "s"
} :
- $ { details . join ( "\n - " ) }
` ;
assert ( details . length === 0 , message ) ;
2020-07-19 19:49:44 +02:00
} ;
}
2021-02-24 20:55:50 +08:00
// Wrap test function in additional assertion that makes sure
// that the test case does not accidentally exit prematurely.
function assertExit ( fn ) {
2021-10-11 09:45:02 -04:00
return async function exitSanitizer ( ... params ) {
2021-02-24 20:55:50 +08:00
setExitHandler ( ( exitCode ) => {
assert (
false ,
` Test case attempted to exit with exit code: ${ exitCode } ` ,
) ;
} ) ;
try {
2021-10-11 09:45:02 -04:00
await fn ( ... params ) ;
2021-02-24 20:55:50 +08:00
} catch ( err ) {
throw err ;
} finally {
setExitHandler ( null ) ;
}
} ;
}
2021-10-11 09:45:02 -04:00
function assertTestStepScopes ( fn ) {
/** @param step {TestStep} */
return async function testStepSanitizer ( step ) {
preValidation ( ) ;
// only report waiting after pre-validation
if ( step . canStreamReporting ( ) ) {
step . reportWait ( ) ;
}
await fn ( createTestContext ( step ) ) ;
postValidation ( ) ;
function preValidation ( ) {
const runningSteps = getPotentialConflictingRunningSteps ( ) ;
const runningStepsWithSanitizers = ArrayPrototypeFilter (
runningSteps ,
( t ) => t . usesSanitizer ,
) ;
if ( runningStepsWithSanitizers . length > 0 ) {
throw new Error (
"Cannot start test step while another test step with sanitizers is running.\n" +
runningStepsWithSanitizers
. map ( ( s ) => ` * ${ s . getFullName ( ) } ` )
. join ( "\n" ) ,
) ;
}
if ( step . usesSanitizer && runningSteps . length > 0 ) {
throw new Error (
"Cannot start test step with sanitizers while another test step is running.\n" +
runningSteps . map ( ( s ) => ` * ${ s . getFullName ( ) } ` ) . join ( "\n" ) ,
) ;
}
function getPotentialConflictingRunningSteps ( ) {
/** @type {TestStep[]} */
const results = [ ] ;
let childStep = step ;
for ( const ancestor of step . ancestors ( ) ) {
for ( const siblingStep of ancestor . children ) {
if ( siblingStep === childStep ) {
continue ;
}
if ( ! siblingStep . finalized ) {
ArrayPrototypePush ( results , siblingStep ) ;
}
}
childStep = ancestor ;
}
return results ;
}
}
function postValidation ( ) {
// check for any running steps
2021-10-12 09:58:04 -04:00
if ( step . hasRunningChildren ) {
2021-10-11 09:45:02 -04:00
throw new Error (
"There were still test steps running after the current scope finished execution. " +
"Ensure all steps are awaited (ex. `await t.step(...)`)." ,
) ;
}
// check if an ancestor already completed
for ( const ancestor of step . ancestors ( ) ) {
if ( ancestor . finalized ) {
throw new Error (
"Parent scope completed before test step finished execution. " +
"Ensure all steps are awaited (ex. `await t.step(...)`)." ,
) ;
}
}
}
} ;
}
2021-07-14 06:11:02 +08:00
function withPermissions ( fn , permissions ) {
function pledgePermissions ( permissions ) {
return core . opSync (
"op_pledge_test_permissions" ,
2021-10-13 18:04:44 +01:00
serializePermissions ( permissions ) ,
2021-07-14 06:11:02 +08:00
) ;
}
function restorePermissions ( token ) {
core . opSync ( "op_restore_test_permissions" , token ) ;
}
2021-10-11 09:45:02 -04:00
return async function applyPermissions ( ... params ) {
2021-07-14 06:11:02 +08:00
const token = pledgePermissions ( permissions ) ;
try {
2021-10-11 09:45:02 -04:00
await fn ( ... params ) ;
2021-07-14 06:11:02 +08:00
} finally {
restorePermissions ( token ) ;
}
} ;
}
2021-04-29 02:17:04 +08:00
const tests = [ ] ;
2020-07-19 19:49:44 +02:00
2021-10-11 09:45:02 -04:00
// Main test function provided by Deno.
2020-07-19 19:49:44 +02:00
function test (
feat(test): Add more overloads for "Deno.test" (#12749)
This commit adds 4 more overloads to "Deno.test()" API.
```
// Deno.test(function testName() { });
export function test(fn: (t: TestContext) => void | Promise<void>): void;
// Deno.test("test name", { only: true }, function() { });
export function test(
name: string,
options: Omit<TestDefinition, "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ name: "test name" }, function() { });
export function test(
options: Omit<TestDefinition, "fn">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ only: true }, function testName() { });
export function test(
options: Omit<TestDefinition, "fn" | "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
```
2021-11-23 14:57:51 +01:00
nameOrFnOrOptions ,
optionsOrFn ,
maybeFn ,
2020-07-19 19:49:44 +02:00
) {
let testDef ;
const defaults = {
ignore : false ,
only : false ,
sanitizeOps : true ,
sanitizeResources : true ,
2021-02-24 20:55:50 +08:00
sanitizeExit : true ,
2021-04-26 05:38:59 +08:00
permissions : null ,
2020-07-19 19:49:44 +02:00
} ;
feat(test): Add more overloads for "Deno.test" (#12749)
This commit adds 4 more overloads to "Deno.test()" API.
```
// Deno.test(function testName() { });
export function test(fn: (t: TestContext) => void | Promise<void>): void;
// Deno.test("test name", { only: true }, function() { });
export function test(
name: string,
options: Omit<TestDefinition, "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ name: "test name" }, function() { });
export function test(
options: Omit<TestDefinition, "fn">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ only: true }, function testName() { });
export function test(
options: Omit<TestDefinition, "fn" | "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
```
2021-11-23 14:57:51 +01:00
if ( typeof nameOrFnOrOptions === "string" ) {
if ( ! nameOrFnOrOptions ) {
2020-07-19 19:49:44 +02:00
throw new TypeError ( "The test name can't be empty" ) ;
}
feat(test): Add more overloads for "Deno.test" (#12749)
This commit adds 4 more overloads to "Deno.test()" API.
```
// Deno.test(function testName() { });
export function test(fn: (t: TestContext) => void | Promise<void>): void;
// Deno.test("test name", { only: true }, function() { });
export function test(
name: string,
options: Omit<TestDefinition, "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ name: "test name" }, function() { });
export function test(
options: Omit<TestDefinition, "fn">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ only: true }, function testName() { });
export function test(
options: Omit<TestDefinition, "fn" | "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
```
2021-11-23 14:57:51 +01:00
if ( typeof optionsOrFn === "function" ) {
testDef = { fn : optionsOrFn , name : nameOrFnOrOptions , ... defaults } ;
} else {
if ( ! maybeFn || typeof maybeFn !== "function" ) {
throw new TypeError ( "Missing test function" ) ;
}
if ( optionsOrFn . fn != undefined ) {
throw new TypeError (
"Unexpected 'fn' field in options, test function is already provided as the third argument." ,
) ;
}
if ( optionsOrFn . name != undefined ) {
throw new TypeError (
"Unexpected 'name' field in options, test name is already provided as the first argument." ,
) ;
}
testDef = {
... defaults ,
... optionsOrFn ,
fn : maybeFn ,
name : nameOrFnOrOptions ,
} ;
}
} else if ( typeof nameOrFnOrOptions === "function" ) {
if ( ! nameOrFnOrOptions . name ) {
throw new TypeError ( "The test function must have a name" ) ;
}
if ( optionsOrFn != undefined ) {
throw new TypeError ( "Unexpected second argument to Deno.test()" ) ;
}
if ( maybeFn != undefined ) {
throw new TypeError ( "Unexpected third argument to Deno.test()" ) ;
}
testDef = {
... defaults ,
fn : nameOrFnOrOptions ,
name : nameOrFnOrOptions . name ,
} ;
2020-07-19 19:49:44 +02:00
} else {
feat(test): Add more overloads for "Deno.test" (#12749)
This commit adds 4 more overloads to "Deno.test()" API.
```
// Deno.test(function testName() { });
export function test(fn: (t: TestContext) => void | Promise<void>): void;
// Deno.test("test name", { only: true }, function() { });
export function test(
name: string,
options: Omit<TestDefinition, "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ name: "test name" }, function() { });
export function test(
options: Omit<TestDefinition, "fn">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ only: true }, function testName() { });
export function test(
options: Omit<TestDefinition, "fn" | "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
```
2021-11-23 14:57:51 +01:00
let fn ;
let name ;
if ( typeof optionsOrFn === "function" ) {
fn = optionsOrFn ;
if ( nameOrFnOrOptions . fn != undefined ) {
throw new TypeError (
"Unexpected 'fn' field in options, test function is already provided as the second argument." ,
) ;
}
name = nameOrFnOrOptions . name ? ? fn . name ;
} else {
if (
! nameOrFnOrOptions . fn || typeof nameOrFnOrOptions . fn !== "function"
) {
throw new TypeError (
"Expected 'fn' field in the first argument to be a test function." ,
) ;
}
fn = nameOrFnOrOptions . fn ;
name = nameOrFnOrOptions . name ? ? fn . name ;
2020-07-19 19:49:44 +02:00
}
feat(test): Add more overloads for "Deno.test" (#12749)
This commit adds 4 more overloads to "Deno.test()" API.
```
// Deno.test(function testName() { });
export function test(fn: (t: TestContext) => void | Promise<void>): void;
// Deno.test("test name", { only: true }, function() { });
export function test(
name: string,
options: Omit<TestDefinition, "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ name: "test name" }, function() { });
export function test(
options: Omit<TestDefinition, "fn">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ only: true }, function testName() { });
export function test(
options: Omit<TestDefinition, "fn" | "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
```
2021-11-23 14:57:51 +01:00
if ( ! name ) {
2020-07-19 19:49:44 +02:00
throw new TypeError ( "The test name can't be empty" ) ;
}
feat(test): Add more overloads for "Deno.test" (#12749)
This commit adds 4 more overloads to "Deno.test()" API.
```
// Deno.test(function testName() { });
export function test(fn: (t: TestContext) => void | Promise<void>): void;
// Deno.test("test name", { only: true }, function() { });
export function test(
name: string,
options: Omit<TestDefinition, "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ name: "test name" }, function() { });
export function test(
options: Omit<TestDefinition, "fn">,
fn: (t: TestContext) => void | Promise<void>,
): void;
// Deno.test({ only: true }, function testName() { });
export function test(
options: Omit<TestDefinition, "fn" | "name">,
fn: (t: TestContext) => void | Promise<void>,
): void;
```
2021-11-23 14:57:51 +01:00
testDef = { ... defaults , ... nameOrFnOrOptions , fn , name } ;
2020-07-19 19:49:44 +02:00
}
2021-10-11 09:45:02 -04:00
testDef . fn = wrapTestFnWithSanitizers ( testDef . fn , testDef ) ;
2021-02-24 20:55:50 +08:00
2021-07-14 06:11:02 +08:00
if ( testDef . permissions ) {
testDef . fn = withPermissions (
testDef . fn ,
2021-10-13 18:04:44 +01:00
testDef . permissions ,
2021-07-14 06:11:02 +08:00
) ;
}
2021-07-04 00:17:52 +02:00
ArrayPrototypePush ( tests , testDef ) ;
2020-07-19 19:49:44 +02:00
}
2021-10-11 09:45:02 -04:00
function formatError ( error ) {
2021-11-20 15:43:40 +01:00
if ( error instanceof AggregateError ) {
2021-10-01 03:54:56 +08:00
const message = error
. errors
. map ( ( error ) =>
inspectArgs ( [ error ] ) . replace ( /^(?!\s*$)/gm , " " . repeat ( 4 ) )
)
. join ( "\n" ) ;
2021-10-11 09:45:02 -04:00
return error . name + "\n" + message + error . stack ;
2021-10-01 03:54:56 +08:00
}
2021-10-11 09:45:02 -04:00
return inspectArgs ( [ error ] ) ;
2021-10-01 03:54:56 +08:00
}
2021-04-29 02:17:04 +08:00
function createTestFilter ( filter ) {
return ( def ) => {
if ( filter ) {
2021-07-04 00:17:52 +02:00
if (
StringPrototypeStartsWith ( filter , "/" ) &&
StringPrototypeEndsWith ( filter , "/" )
) {
const regex = new RegExp (
StringPrototypeSlice ( filter , 1 , filter . length - 1 ) ,
) ;
return RegExpPrototypeTest ( regex , def . name ) ;
2020-07-19 19:49:44 +02:00
}
2021-07-04 00:17:52 +02:00
return StringPrototypeIncludes ( def . name , filter ) ;
2020-07-19 19:49:44 +02:00
}
2021-04-29 02:17:04 +08:00
return true ;
} ;
2020-07-19 19:49:44 +02:00
}
2021-10-11 09:45:02 -04:00
async function runTest ( test , description ) {
if ( test . ignore ) {
2021-07-06 00:36:43 +08:00
return "ignored" ;
}
2021-10-11 09:45:02 -04:00
const step = new TestStep ( {
name : test . name ,
parent : undefined ,
rootTestDescription : description ,
sanitizeOps : test . sanitizeOps ,
sanitizeResources : test . sanitizeResources ,
sanitizeExit : test . sanitizeExit ,
} ) ;
2021-04-29 02:17:04 +08:00
try {
2021-10-11 09:45:02 -04:00
await test . fn ( step ) ;
const failCount = step . failedChildStepsCount ( ) ;
return failCount === 0 ? "ok" : {
"failed" : formatError (
new Error (
` ${ failCount } test step ${ failCount === 1 ? "" : "s" } failed. ` ,
) ,
) ,
} ;
2021-04-29 02:17:04 +08:00
} catch ( error ) {
2021-10-11 09:45:02 -04:00
return {
"failed" : formatError ( error ) ,
} ;
} finally {
2021-10-13 08:55:12 -04:00
step . finalized = true ;
2021-10-11 09:45:02 -04:00
// ensure the children report their result
for ( const child of step . children ) {
child . reportResult ( ) ;
}
2021-04-29 02:17:04 +08:00
}
2020-07-19 19:49:44 +02:00
}
2021-07-15 03:05:16 +08:00
function getTestOrigin ( ) {
return core . opSync ( "op_get_test_origin" ) ;
}
2021-09-06 04:42:35 +08:00
function reportTestPlan ( plan ) {
core . opSync ( "op_dispatch_test_event" , {
plan ,
} ) ;
}
function reportTestConsoleOutput ( console ) {
core . opSync ( "op_dispatch_test_event" , {
output : { console } ,
} ) ;
}
function reportTestWait ( test ) {
core . opSync ( "op_dispatch_test_event" , {
wait : test ,
} ) ;
}
function reportTestResult ( test , result , elapsed ) {
core . opSync ( "op_dispatch_test_event" , {
result : [ test , result , elapsed ] ,
} ) ;
2021-07-15 03:05:16 +08:00
}
2021-10-11 09:45:02 -04:00
function reportTestStepWait ( testDescription ) {
core . opSync ( "op_dispatch_test_event" , {
stepWait : testDescription ,
} ) ;
}
function reportTestStepResult ( testDescription , result , elapsed ) {
core . opSync ( "op_dispatch_test_event" , {
stepResult : [ testDescription , result , elapsed ] ,
} ) ;
}
2020-07-19 19:49:44 +02:00
async function runTests ( {
2021-04-29 02:17:04 +08:00
filter = null ,
2021-07-06 09:20:33 +08:00
shuffle = null ,
2020-07-19 19:49:44 +02:00
} = { } ) {
2021-11-30 01:27:30 +01:00
core . setMacrotaskCallback ( handleOpSanitizerDelayMacrotask ) ;
2021-07-15 03:05:16 +08:00
const origin = getTestOrigin ( ) ;
2020-07-19 19:49:44 +02:00
const originalConsole = globalThis . console ;
2021-09-04 21:16:35 +08:00
2021-09-06 04:42:35 +08:00
globalThis . console = new Console ( reportTestConsoleOutput ) ;
2020-07-19 19:49:44 +02:00
2021-07-04 00:17:52 +02:00
const only = ArrayPrototypeFilter ( tests , ( test ) => test . only ) ;
2021-07-15 03:05:16 +08:00
const filtered = ArrayPrototypeFilter (
2021-10-12 09:58:04 -04:00
only . length > 0 ? only : tests ,
2021-04-29 02:17:04 +08:00
createTestFilter ( filter ) ,
) ;
2021-07-15 03:05:16 +08:00
2021-09-06 04:42:35 +08:00
reportTestPlan ( {
origin ,
total : filtered . length ,
filteredOut : tests . length - filtered . length ,
usedOnly : only . length > 0 ,
2021-04-29 02:17:04 +08:00
} ) ;
2020-07-19 19:49:44 +02:00
2021-07-06 09:20:33 +08:00
if ( shuffle !== null ) {
// http://en.wikipedia.org/wiki/Linear_congruential_generator
const nextInt = ( function ( state ) {
const m = 0x80000000 ;
const a = 1103515245 ;
const c = 12345 ;
return function ( max ) {
return state = ( ( a * state + c ) % m ) % max ;
} ;
} ( shuffle ) ) ;
2021-07-15 03:05:16 +08:00
for ( let i = filtered . length - 1 ; i > 0 ; i -- ) {
2021-07-06 09:20:33 +08:00
const j = nextInt ( i ) ;
2021-07-15 03:05:16 +08:00
[ filtered [ i ] , filtered [ j ] ] = [ filtered [ j ] , filtered [ i ] ] ;
2021-07-06 09:20:33 +08:00
}
}
2021-07-15 03:05:16 +08:00
for ( const test of filtered ) {
const description = {
origin ,
name : test . name ,
} ;
2021-07-05 16:26:57 +08:00
const earlier = DateNow ( ) ;
2021-09-06 04:42:35 +08:00
reportTestWait ( description ) ;
2021-07-05 16:26:57 +08:00
2021-10-11 09:45:02 -04:00
const result = await runTest ( test , description ) ;
2021-07-15 03:05:16 +08:00
const elapsed = DateNow ( ) - earlier ;
2021-07-05 16:26:57 +08:00
2021-09-06 04:42:35 +08:00
reportTestResult ( description , result , elapsed ) ;
2020-07-19 19:49:44 +02:00
}
2021-09-04 21:16:35 +08:00
globalThis . console = originalConsole ;
2020-07-19 19:49:44 +02:00
}
2021-10-11 09:45:02 -04:00
/ * *
* @ typedef { {
* fn : ( t : TestContext ) => void | Promise < void > ,
* name : string ,
* ignore ? : boolean ,
* sanitizeOps ? : boolean ,
* sanitizeResources ? : boolean ,
* sanitizeExit ? : boolean ,
* } } TestStepDefinition
*
* @ typedef { {
* name : string ;
* parent : TestStep | undefined ,
* rootTestDescription : { origin : string ; name : string } ;
* sanitizeOps : boolean ,
* sanitizeResources : boolean ,
* sanitizeExit : boolean ,
* } } TestStepParams
* /
class TestStep {
/** @type {TestStepParams} */
# params ;
reportedWait = false ;
# reportedResult = false ;
finalized = false ;
elapsed = 0 ;
2021-10-12 09:58:04 -04:00
/** @type "ok" | "ignored" | "pending" | "failed" */
2021-10-11 09:45:02 -04:00
status = "pending" ;
error = undefined ;
/** @type {TestStep[]} */
children = [ ] ;
/** @param params {TestStepParams} */
constructor ( params ) {
this . # params = params ;
}
get name ( ) {
return this . # params . name ;
}
get parent ( ) {
return this . # params . parent ;
}
get rootTestDescription ( ) {
return this . # params . rootTestDescription ;
}
get sanitizerOptions ( ) {
return {
sanitizeResources : this . # params . sanitizeResources ,
sanitizeOps : this . # params . sanitizeOps ,
sanitizeExit : this . # params . sanitizeExit ,
} ;
}
get usesSanitizer ( ) {
return this . # params . sanitizeResources ||
this . # params . sanitizeOps ||
this . # params . sanitizeExit ;
}
2021-10-13 08:55:12 -04:00
/ * * I f a t e s t v a l i d a t i o n e r r o r a l r e a d y o c c u r r e d t h e n d o n ' t b o t h e r c h e c k i n g
* the sanitizers as that will create extra noise .
* /
get shouldSkipSanitizers ( ) {
return this . hasRunningChildren || this . parent ? . finalized ;
}
2021-10-12 09:58:04 -04:00
get hasRunningChildren ( ) {
return ArrayPrototypeSome (
this . children ,
/** @param step {TestStep} */
( step ) => step . status === "pending" ,
) ;
}
2021-10-11 09:45:02 -04:00
failedChildStepsCount ( ) {
return ArrayPrototypeFilter (
this . children ,
/** @param step {TestStep} */
( step ) => step . status === "failed" ,
) . length ;
}
canStreamReporting ( ) {
// there should only ever be one sub step running when running with
// sanitizers, so we can use this to tell if we can stream reporting
return this . selfAndAllAncestorsUseSanitizer ( ) &&
this . children . every ( ( c ) => c . usesSanitizer || c . finalized ) ;
}
selfAndAllAncestorsUseSanitizer ( ) {
if ( ! this . usesSanitizer ) {
return false ;
}
for ( const ancestor of this . ancestors ( ) ) {
if ( ! ancestor . usesSanitizer ) {
return false ;
}
}
return true ;
}
* ancestors ( ) {
let ancestor = this . parent ;
while ( ancestor ) {
yield ancestor ;
ancestor = ancestor . parent ;
}
}
getFullName ( ) {
if ( this . parent ) {
return ` ${ this . parent . getFullName ( ) } > ${ this . name } ` ;
} else {
return this . name ;
}
}
reportWait ( ) {
if ( this . reportedWait || ! this . parent ) {
return ;
}
reportTestStepWait ( this . # getTestStepDescription ( ) ) ;
this . reportedWait = true ;
}
reportResult ( ) {
if ( this . # reportedResult || ! this . parent ) {
return ;
}
this . reportWait ( ) ;
for ( const child of this . children ) {
child . reportResult ( ) ;
}
reportTestStepResult (
this . # getTestStepDescription ( ) ,
this . # getStepResult ( ) ,
this . elapsed ,
) ;
this . # reportedResult = true ;
}
# getStepResult ( ) {
switch ( this . status ) {
case "ok" :
return "ok" ;
case "ignored" :
return "ignored" ;
case "pending" :
return {
"pending" : this . error && formatError ( this . error ) ,
} ;
case "failed" :
return {
"failed" : this . error && formatError ( this . error ) ,
} ;
default :
throw new Error ( ` Unhandled status: ${ this . status } ` ) ;
}
}
# getTestStepDescription ( ) {
return {
test : this . rootTestDescription ,
name : this . name ,
level : this . # getLevel ( ) ,
} ;
}
# getLevel ( ) {
let count = 0 ;
for ( const _ of this . ancestors ( ) ) {
count ++ ;
}
return count ;
}
}
/** @param parentStep {TestStep} */
function createTestContext ( parentStep ) {
return {
[ SymbolToStringTag ] : "TestContext" ,
/ * *
* @ param nameOrTestDefinition { string | TestStepDefinition }
* @ param fn { ( t : TestContext ) => void | Promise < void > }
* /
async step ( nameOrTestDefinition , fn ) {
if ( parentStep . finalized ) {
throw new Error (
"Cannot run test step after parent scope has finished execution. " +
"Ensure any `.step(...)` calls are executed before their parent scope completes execution." ,
) ;
}
const definition = getDefinition ( ) ;
const subStep = new TestStep ( {
name : definition . name ,
parent : parentStep ,
rootTestDescription : parentStep . rootTestDescription ,
sanitizeOps : getOrDefault (
definition . sanitizeOps ,
parentStep . sanitizerOptions . sanitizeOps ,
) ,
sanitizeResources : getOrDefault (
definition . sanitizeResources ,
parentStep . sanitizerOptions . sanitizeResources ,
) ,
sanitizeExit : getOrDefault (
definition . sanitizeExit ,
parentStep . sanitizerOptions . sanitizeExit ,
) ,
} ) ;
ArrayPrototypePush ( parentStep . children , subStep ) ;
try {
if ( definition . ignore ) {
subStep . status = "ignored" ;
subStep . finalized = true ;
if ( subStep . canStreamReporting ( ) ) {
subStep . reportResult ( ) ;
}
return false ;
}
const testFn = wrapTestFnWithSanitizers (
definition . fn ,
subStep . sanitizerOptions ,
) ;
const start = DateNow ( ) ;
try {
await testFn ( subStep ) ;
if ( subStep . failedChildStepsCount ( ) > 0 ) {
subStep . status = "failed" ;
} else {
subStep . status = "ok" ;
}
} catch ( error ) {
subStep . error = formatError ( error ) ;
subStep . status = "failed" ;
}
subStep . elapsed = DateNow ( ) - start ;
if ( subStep . parent ? . finalized ) {
// always point this test out as one that was still running
// if the parent step finalized
subStep . status = "pending" ;
}
subStep . finalized = true ;
if ( subStep . reportedWait && subStep . canStreamReporting ( ) ) {
subStep . reportResult ( ) ;
}
return subStep . status === "ok" ;
} finally {
if ( parentStep . canStreamReporting ( ) ) {
// flush any buffered steps
for ( const parentChild of parentStep . children ) {
parentChild . reportResult ( ) ;
}
}
}
/** @returns {TestStepDefinition} */
function getDefinition ( ) {
if ( typeof nameOrTestDefinition === "string" ) {
if ( ! ( fn instanceof Function ) ) {
throw new TypeError ( "Expected function for second argument." ) ;
}
return {
name : nameOrTestDefinition ,
fn ,
} ;
} else if ( typeof nameOrTestDefinition === "object" ) {
return nameOrTestDefinition ;
} else {
throw new TypeError (
"Expected a test definition or name and function." ,
) ;
}
}
} ,
} ;
}
/ * *
* @ template T { Function }
* @ param testFn { T }
* @ param opts { {
* sanitizeOps : boolean ,
* sanitizeResources : boolean ,
* sanitizeExit : boolean ,
* } }
* @ returns { T }
* /
function wrapTestFnWithSanitizers ( testFn , opts ) {
testFn = assertTestStepScopes ( testFn ) ;
if ( opts . sanitizeOps ) {
testFn = assertOps ( testFn ) ;
}
if ( opts . sanitizeResources ) {
testFn = assertResources ( testFn ) ;
}
if ( opts . sanitizeExit ) {
testFn = assertExit ( testFn ) ;
}
return testFn ;
}
/ * *
* @ template T
* @ param value { T | undefined }
* @ param defaultValue { T }
* @ returns T
* /
function getOrDefault ( value , defaultValue ) {
return value == null ? defaultValue : value ;
}
2021-03-12 21:23:59 +01:00
window . _ _bootstrap . internals = {
... window . _ _bootstrap . internals ? ? { } ,
runTests ,
} ;
2020-07-19 19:49:44 +02:00
window . _ _bootstrap . testing = {
test ,
} ;
} ) ( this ) ;