mirror of
https://github.com/denoland/deno.git
synced 2024-12-11 18:17:48 -05:00
53088e16de
This commit improves the error messages for the `deno test` async op sanitizer. It does this in two ways: - it uses handwritten error messages for each op that could be leaking - it includes traces showing where each op was started This "async op tracing" functionality is a new feature in deno_core. It likely has a significant performance impact, which is why it is only enabled in tests.
697 lines
16 KiB
TypeScript
697 lines
16 KiB
TypeScript
// Copyright 2018-2022 the Deno authors. All rights reserved. MIT license.
|
|
import {
|
|
assert,
|
|
assertEquals,
|
|
assertNotEquals,
|
|
Deferred,
|
|
deferred,
|
|
delay,
|
|
unreachable,
|
|
} from "./test_util.ts";
|
|
|
|
const decoder = new TextDecoder();
|
|
|
|
Deno.test(async function functionParameterBindingSuccess() {
|
|
const promise = deferred();
|
|
let count = 0;
|
|
|
|
const nullProto = (newCount: number) => {
|
|
count = newCount;
|
|
promise.resolve();
|
|
};
|
|
|
|
Reflect.setPrototypeOf(nullProto, null);
|
|
|
|
setTimeout(nullProto, 500, 1);
|
|
await promise;
|
|
// count should be reassigned
|
|
assertEquals(count, 1);
|
|
});
|
|
|
|
Deno.test(async function stringifyAndEvalNonFunctions() {
|
|
// eval can only access global scope
|
|
const global = globalThis as unknown as {
|
|
globalPromise: ReturnType<typeof deferred>;
|
|
globalCount: number;
|
|
};
|
|
global.globalPromise = deferred();
|
|
global.globalCount = 0;
|
|
|
|
const notAFunction =
|
|
"globalThis.globalCount++; globalThis.globalPromise.resolve();" as unknown as () =>
|
|
void;
|
|
|
|
setTimeout(notAFunction, 500);
|
|
|
|
await global.globalPromise;
|
|
|
|
// count should be incremented
|
|
assertEquals(global.globalCount, 1);
|
|
|
|
Reflect.deleteProperty(global, "globalPromise");
|
|
Reflect.deleteProperty(global, "globalCount");
|
|
});
|
|
|
|
Deno.test(async function timeoutSuccess() {
|
|
const promise = deferred();
|
|
let count = 0;
|
|
setTimeout(() => {
|
|
count++;
|
|
promise.resolve();
|
|
}, 500);
|
|
await promise;
|
|
// count should increment
|
|
assertEquals(count, 1);
|
|
});
|
|
|
|
Deno.test(async function timeoutEvalNoScopeLeak() {
|
|
// eval can only access global scope
|
|
const global = globalThis as unknown as {
|
|
globalPromise: Deferred<Error>;
|
|
};
|
|
global.globalPromise = deferred();
|
|
setTimeout(
|
|
`
|
|
try {
|
|
console.log(core);
|
|
globalThis.globalPromise.reject(new Error("Didn't throw."));
|
|
} catch (error) {
|
|
globalThis.globalPromise.resolve(error);
|
|
}` as unknown as () => void,
|
|
0,
|
|
);
|
|
const error = await global.globalPromise;
|
|
assertEquals(error.name, "ReferenceError");
|
|
Reflect.deleteProperty(global, "globalPromise");
|
|
});
|
|
|
|
Deno.test(async function timeoutArgs() {
|
|
const promise = deferred();
|
|
const arg = 1;
|
|
let capturedArgs: unknown[] = [];
|
|
setTimeout(
|
|
function () {
|
|
capturedArgs = [...arguments];
|
|
promise.resolve();
|
|
},
|
|
10,
|
|
arg,
|
|
arg.toString(),
|
|
[arg],
|
|
);
|
|
await promise;
|
|
assertEquals(capturedArgs, [
|
|
arg,
|
|
arg.toString(),
|
|
[arg],
|
|
]);
|
|
});
|
|
|
|
Deno.test(async function timeoutCancelSuccess() {
|
|
let count = 0;
|
|
const id = setTimeout(() => {
|
|
count++;
|
|
}, 1);
|
|
// Cancelled, count should not increment
|
|
clearTimeout(id);
|
|
await delay(600);
|
|
assertEquals(count, 0);
|
|
});
|
|
|
|
Deno.test(async function timeoutCancelMultiple() {
|
|
function uncalled(): never {
|
|
throw new Error("This function should not be called.");
|
|
}
|
|
|
|
// Set timers and cancel them in the same order.
|
|
const t1 = setTimeout(uncalled, 10);
|
|
const t2 = setTimeout(uncalled, 10);
|
|
const t3 = setTimeout(uncalled, 10);
|
|
clearTimeout(t1);
|
|
clearTimeout(t2);
|
|
clearTimeout(t3);
|
|
|
|
// Set timers and cancel them in reverse order.
|
|
const t4 = setTimeout(uncalled, 20);
|
|
const t5 = setTimeout(uncalled, 20);
|
|
const t6 = setTimeout(uncalled, 20);
|
|
clearTimeout(t6);
|
|
clearTimeout(t5);
|
|
clearTimeout(t4);
|
|
|
|
// Sleep until we're certain that the cancelled timers aren't gonna fire.
|
|
await delay(50);
|
|
});
|
|
|
|
Deno.test(async function timeoutCancelInvalidSilentFail() {
|
|
// Expect no panic
|
|
const promise = deferred();
|
|
let count = 0;
|
|
const id = setTimeout(() => {
|
|
count++;
|
|
// Should have no effect
|
|
clearTimeout(id);
|
|
promise.resolve();
|
|
}, 500);
|
|
await promise;
|
|
assertEquals(count, 1);
|
|
|
|
// Should silently fail (no panic)
|
|
clearTimeout(2147483647);
|
|
});
|
|
|
|
Deno.test(async function intervalSuccess() {
|
|
const promise = deferred();
|
|
let count = 0;
|
|
const id = setInterval(() => {
|
|
count++;
|
|
clearInterval(id);
|
|
promise.resolve();
|
|
}, 100);
|
|
await promise;
|
|
// Clear interval
|
|
clearInterval(id);
|
|
// count should increment twice
|
|
assertEquals(count, 1);
|
|
// Similar false async leaking alarm.
|
|
// Force next round of polling.
|
|
await delay(0);
|
|
});
|
|
|
|
Deno.test(async function intervalCancelSuccess() {
|
|
let count = 0;
|
|
const id = setInterval(() => {
|
|
count++;
|
|
}, 1);
|
|
clearInterval(id);
|
|
await delay(500);
|
|
assertEquals(count, 0);
|
|
});
|
|
|
|
Deno.test(async function intervalOrdering() {
|
|
const timers: number[] = [];
|
|
let timeouts = 0;
|
|
function onTimeout() {
|
|
++timeouts;
|
|
for (let i = 1; i < timers.length; i++) {
|
|
clearTimeout(timers[i]);
|
|
}
|
|
}
|
|
for (let i = 0; i < 10; i++) {
|
|
timers[i] = setTimeout(onTimeout, 1);
|
|
}
|
|
await delay(500);
|
|
assertEquals(timeouts, 1);
|
|
});
|
|
|
|
Deno.test(function intervalCancelInvalidSilentFail() {
|
|
// Should silently fail (no panic)
|
|
clearInterval(2147483647);
|
|
});
|
|
|
|
Deno.test(async function callbackTakesLongerThanInterval() {
|
|
const promise = deferred();
|
|
|
|
let timeEndOfFirstCallback: number | undefined;
|
|
const interval = setInterval(() => {
|
|
if (timeEndOfFirstCallback === undefined) {
|
|
// First callback
|
|
Deno.sleepSync(300);
|
|
timeEndOfFirstCallback = Date.now();
|
|
} else {
|
|
// Second callback
|
|
assert(Date.now() - 100 >= timeEndOfFirstCallback);
|
|
clearInterval(interval);
|
|
promise.resolve();
|
|
}
|
|
}, 100);
|
|
|
|
await promise;
|
|
});
|
|
|
|
// https://github.com/denoland/deno/issues/11398
|
|
Deno.test(async function clearTimeoutAfterNextTimerIsDue1() {
|
|
const promise = deferred();
|
|
|
|
setTimeout(() => {
|
|
promise.resolve();
|
|
}, 300);
|
|
|
|
const interval = setInterval(() => {
|
|
Deno.sleepSync(400);
|
|
// Both the interval and the timeout's due times are now in the past.
|
|
clearInterval(interval);
|
|
}, 100);
|
|
|
|
await promise;
|
|
});
|
|
|
|
// https://github.com/denoland/deno/issues/11398
|
|
Deno.test(async function clearTimeoutAfterNextTimerIsDue2() {
|
|
const promise = deferred();
|
|
|
|
const timeout1 = setTimeout(unreachable, 100);
|
|
|
|
setTimeout(() => {
|
|
promise.resolve();
|
|
}, 200);
|
|
|
|
Deno.sleepSync(300);
|
|
// Both of the timeouts' due times are now in the past.
|
|
clearTimeout(timeout1);
|
|
|
|
await promise;
|
|
});
|
|
|
|
Deno.test(async function fireCallbackImmediatelyWhenDelayOverMaxValue() {
|
|
let count = 0;
|
|
setTimeout(() => {
|
|
count++;
|
|
}, 2 ** 31);
|
|
await delay(1);
|
|
assertEquals(count, 1);
|
|
});
|
|
|
|
Deno.test(async function timeoutCallbackThis() {
|
|
const promise = deferred();
|
|
let capturedThis: unknown;
|
|
const obj = {
|
|
foo() {
|
|
capturedThis = this;
|
|
promise.resolve();
|
|
},
|
|
};
|
|
setTimeout(obj.foo, 1);
|
|
await promise;
|
|
assertEquals(capturedThis, window);
|
|
});
|
|
|
|
Deno.test(async function timeoutBindThis() {
|
|
const thisCheckPassed = [null, undefined, window, globalThis];
|
|
|
|
const thisCheckFailed = [
|
|
0,
|
|
"",
|
|
true,
|
|
false,
|
|
{},
|
|
[],
|
|
"foo",
|
|
() => {},
|
|
Object.prototype,
|
|
];
|
|
|
|
for (const thisArg of thisCheckPassed) {
|
|
const resolvable = deferred();
|
|
let hasThrown = 0;
|
|
try {
|
|
setTimeout.call(thisArg, () => resolvable.resolve(), 1);
|
|
hasThrown = 1;
|
|
} catch (err) {
|
|
if (err instanceof TypeError) {
|
|
hasThrown = 2;
|
|
} else {
|
|
hasThrown = 3;
|
|
}
|
|
}
|
|
await resolvable;
|
|
assertEquals(hasThrown, 1);
|
|
}
|
|
|
|
for (const thisArg of thisCheckFailed) {
|
|
let hasThrown = 0;
|
|
try {
|
|
setTimeout.call(thisArg, () => {}, 1);
|
|
hasThrown = 1;
|
|
} catch (err) {
|
|
if (err instanceof TypeError) {
|
|
hasThrown = 2;
|
|
} else {
|
|
hasThrown = 3;
|
|
}
|
|
}
|
|
assertEquals(hasThrown, 2);
|
|
}
|
|
});
|
|
|
|
Deno.test(function clearTimeoutShouldConvertToNumber() {
|
|
let called = false;
|
|
const obj = {
|
|
valueOf(): number {
|
|
called = true;
|
|
return 1;
|
|
},
|
|
};
|
|
clearTimeout((obj as unknown) as number);
|
|
assert(called);
|
|
});
|
|
|
|
Deno.test(function setTimeoutShouldThrowWithBigint() {
|
|
let hasThrown = 0;
|
|
try {
|
|
setTimeout(() => {}, (1n as unknown) as number);
|
|
hasThrown = 1;
|
|
} catch (err) {
|
|
if (err instanceof TypeError) {
|
|
hasThrown = 2;
|
|
} else {
|
|
hasThrown = 3;
|
|
}
|
|
}
|
|
assertEquals(hasThrown, 2);
|
|
});
|
|
|
|
Deno.test(function clearTimeoutShouldThrowWithBigint() {
|
|
let hasThrown = 0;
|
|
try {
|
|
clearTimeout((1n as unknown) as number);
|
|
hasThrown = 1;
|
|
} catch (err) {
|
|
if (err instanceof TypeError) {
|
|
hasThrown = 2;
|
|
} else {
|
|
hasThrown = 3;
|
|
}
|
|
}
|
|
assertEquals(hasThrown, 2);
|
|
});
|
|
|
|
Deno.test(function testFunctionName() {
|
|
assertEquals(clearTimeout.name, "clearTimeout");
|
|
assertEquals(clearInterval.name, "clearInterval");
|
|
});
|
|
|
|
Deno.test(function testFunctionParamsLength() {
|
|
assertEquals(setTimeout.length, 1);
|
|
assertEquals(setInterval.length, 1);
|
|
assertEquals(clearTimeout.length, 0);
|
|
assertEquals(clearInterval.length, 0);
|
|
});
|
|
|
|
Deno.test(function clearTimeoutAndClearIntervalNotBeEquals() {
|
|
assertNotEquals(clearTimeout, clearInterval);
|
|
});
|
|
|
|
Deno.test(async function timerMaxCpuBug() {
|
|
// There was a bug where clearing a timeout would cause Deno to use 100% CPU.
|
|
clearTimeout(setTimeout(() => {}, 1000));
|
|
// We can check this by counting how many ops have triggered in the interim.
|
|
// Certainly less than 10 ops should have been dispatched in next 100 ms.
|
|
const { ops: pre } = Deno.metrics();
|
|
await delay(100);
|
|
const { ops: post } = Deno.metrics();
|
|
const before = pre.op_sleep.opsDispatched;
|
|
const after = post.op_sleep.opsDispatched;
|
|
assert(after - before < 10);
|
|
});
|
|
|
|
Deno.test(async function timerOrdering() {
|
|
const array: number[] = [];
|
|
const donePromise = deferred();
|
|
|
|
function push(n: number) {
|
|
array.push(n);
|
|
if (array.length === 6) {
|
|
donePromise.resolve();
|
|
}
|
|
}
|
|
|
|
setTimeout(() => {
|
|
push(1);
|
|
setTimeout(() => push(4));
|
|
}, 0);
|
|
setTimeout(() => {
|
|
push(2);
|
|
setTimeout(() => push(5));
|
|
}, 0);
|
|
setTimeout(() => {
|
|
push(3);
|
|
setTimeout(() => push(6));
|
|
}, 0);
|
|
|
|
await donePromise;
|
|
|
|
assertEquals(array, [1, 2, 3, 4, 5, 6]);
|
|
});
|
|
|
|
Deno.test(async function timerBasicMicrotaskOrdering() {
|
|
let s = "";
|
|
let count = 0;
|
|
const promise = deferred();
|
|
setTimeout(() => {
|
|
Promise.resolve().then(() => {
|
|
count++;
|
|
s += "de";
|
|
if (count === 2) {
|
|
promise.resolve();
|
|
}
|
|
});
|
|
});
|
|
setTimeout(() => {
|
|
count++;
|
|
s += "no";
|
|
if (count === 2) {
|
|
promise.resolve();
|
|
}
|
|
});
|
|
await promise;
|
|
assertEquals(s, "deno");
|
|
});
|
|
|
|
Deno.test(async function timerNestedMicrotaskOrdering() {
|
|
let s = "";
|
|
const promise = deferred();
|
|
s += "0";
|
|
setTimeout(() => {
|
|
s += "4";
|
|
setTimeout(() => (s += "A"));
|
|
Promise.resolve()
|
|
.then(() => {
|
|
setTimeout(() => {
|
|
s += "B";
|
|
promise.resolve();
|
|
});
|
|
})
|
|
.then(() => {
|
|
s += "5";
|
|
});
|
|
});
|
|
setTimeout(() => (s += "6"));
|
|
Promise.resolve().then(() => (s += "2"));
|
|
Promise.resolve().then(() =>
|
|
setTimeout(() => {
|
|
s += "7";
|
|
Promise.resolve()
|
|
.then(() => (s += "8"))
|
|
.then(() => {
|
|
s += "9";
|
|
});
|
|
})
|
|
);
|
|
Promise.resolve().then(() => Promise.resolve().then(() => (s += "3")));
|
|
s += "1";
|
|
await promise;
|
|
assertEquals(s, "0123456789AB");
|
|
});
|
|
|
|
Deno.test(function testQueueMicrotask() {
|
|
assertEquals(typeof queueMicrotask, "function");
|
|
});
|
|
|
|
Deno.test(async function timerIgnoresDateOverride() {
|
|
const OriginalDate = Date;
|
|
const promise = deferred();
|
|
let hasThrown = 0;
|
|
try {
|
|
const overrideCalled: () => number = () => {
|
|
promise.reject("global Date override used over original Date object");
|
|
return 0;
|
|
};
|
|
const DateOverride = () => {
|
|
overrideCalled();
|
|
};
|
|
globalThis.Date = DateOverride as DateConstructor;
|
|
globalThis.Date.now = overrideCalled;
|
|
globalThis.Date.UTC = overrideCalled;
|
|
globalThis.Date.parse = overrideCalled;
|
|
queueMicrotask(promise.resolve);
|
|
await promise;
|
|
hasThrown = 1;
|
|
} catch (err) {
|
|
if (typeof err === "string") {
|
|
assertEquals(err, "global Date override used over original Date object");
|
|
hasThrown = 2;
|
|
} else if (err instanceof TypeError) {
|
|
hasThrown = 3;
|
|
} else {
|
|
hasThrown = 4;
|
|
}
|
|
} finally {
|
|
globalThis.Date = OriginalDate;
|
|
}
|
|
assertEquals(hasThrown, 1);
|
|
});
|
|
|
|
Deno.test({ permissions: { hrtime: true } }, function sleepSync() {
|
|
const start = performance.now();
|
|
Deno.sleepSync(10);
|
|
const after = performance.now();
|
|
assert(after - start >= 10);
|
|
});
|
|
|
|
Deno.test(
|
|
{ permissions: { hrtime: true } },
|
|
async function sleepSyncShorterPromise() {
|
|
const perf = performance;
|
|
const short = 5;
|
|
const long = 10;
|
|
|
|
const start = perf.now();
|
|
const p = delay(short).then(() => {
|
|
const after = perf.now();
|
|
// pending promises should resolve after the main thread comes out of sleep
|
|
assert(after - start >= long);
|
|
});
|
|
Deno.sleepSync(long);
|
|
|
|
await p;
|
|
},
|
|
);
|
|
|
|
Deno.test(
|
|
{ permissions: { hrtime: true } },
|
|
async function sleepSyncLongerPromise() {
|
|
const perf = performance;
|
|
const short = 5;
|
|
const long = 10;
|
|
|
|
const start = perf.now();
|
|
const p = delay(long).then(() => {
|
|
const after = perf.now();
|
|
// sleeping for less than the duration of a promise should have no impact
|
|
// on the resolution of that promise
|
|
assert(after - start >= long);
|
|
});
|
|
Deno.sleepSync(short);
|
|
|
|
await p;
|
|
},
|
|
);
|
|
|
|
async function execCode(code: string) {
|
|
const p = Deno.run({
|
|
cmd: [
|
|
Deno.execPath(),
|
|
"eval",
|
|
"--unstable",
|
|
code,
|
|
],
|
|
stdout: "piped",
|
|
});
|
|
const [status, output] = await Promise.all([p.status(), p.output()]);
|
|
p.close();
|
|
return [status.code, decoder.decode(output)];
|
|
}
|
|
|
|
Deno.test({
|
|
name: "unrefTimer",
|
|
permissions: { run: true },
|
|
fn: async () => {
|
|
const [statusCode, output] = await execCode(`
|
|
const timer = setTimeout(() => console.log("1"));
|
|
Deno.unrefTimer(timer);
|
|
`);
|
|
assertEquals(statusCode, 0);
|
|
assertEquals(output, "");
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "unrefTimer - mix ref and unref 1",
|
|
permissions: { run: true },
|
|
fn: async () => {
|
|
const [statusCode, output] = await execCode(`
|
|
const timer1 = setTimeout(() => console.log("1"), 200);
|
|
const timer2 = setTimeout(() => console.log("2"), 400);
|
|
const timer3 = setTimeout(() => console.log("3"), 600);
|
|
Deno.unrefTimer(timer3);
|
|
`);
|
|
assertEquals(statusCode, 0);
|
|
assertEquals(output, "1\n2\n");
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "unrefTimer - mix ref and unref 2",
|
|
permissions: { run: true },
|
|
fn: async () => {
|
|
const [statusCode, output] = await execCode(`
|
|
const timer1 = setTimeout(() => console.log("1"), 200);
|
|
const timer2 = setTimeout(() => console.log("2"), 400);
|
|
const timer3 = setTimeout(() => console.log("3"), 600);
|
|
Deno.unrefTimer(timer1);
|
|
Deno.unrefTimer(timer2);
|
|
`);
|
|
assertEquals(statusCode, 0);
|
|
assertEquals(output, "1\n2\n3\n");
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "unrefTimer - unref interval",
|
|
permissions: { run: true },
|
|
fn: async () => {
|
|
const [statusCode, output] = await execCode(`
|
|
let i = 0;
|
|
const timer1 = setInterval(() => {
|
|
console.log("1");
|
|
i++;
|
|
if (i === 5) {
|
|
Deno.unrefTimer(timer1);
|
|
}
|
|
}, 10);
|
|
`);
|
|
assertEquals(statusCode, 0);
|
|
assertEquals(output, "1\n1\n1\n1\n1\n");
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "unrefTimer - unref then ref 1",
|
|
permissions: { run: true },
|
|
fn: async () => {
|
|
const [statusCode, output] = await execCode(`
|
|
const timer1 = setTimeout(() => console.log("1"), 10);
|
|
Deno.unrefTimer(timer1);
|
|
Deno.refTimer(timer1);
|
|
`);
|
|
assertEquals(statusCode, 0);
|
|
assertEquals(output, "1\n");
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "unrefTimer - unref then ref",
|
|
permissions: { run: true },
|
|
fn: async () => {
|
|
const [statusCode, output] = await execCode(`
|
|
const timer1 = setTimeout(() => {
|
|
console.log("1");
|
|
Deno.refTimer(timer2);
|
|
}, 10);
|
|
const timer2 = setTimeout(() => console.log("2"), 20);
|
|
Deno.unrefTimer(timer2);
|
|
`);
|
|
assertEquals(statusCode, 0);
|
|
assertEquals(output, "1\n2\n");
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "unrefTimer - invalid calls do nothing",
|
|
permissions: { run: true },
|
|
fn: () => {
|
|
Deno.unrefTimer(NaN);
|
|
Deno.refTimer(NaN);
|
|
},
|
|
});
|