2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2020-03-03 18:22:53 +01:00
|
|
|
import {
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest,
|
2020-03-03 18:22:53 +01:00
|
|
|
createResolvable,
|
|
|
|
assert,
|
|
|
|
assertEquals,
|
|
|
|
assertNotEquals
|
|
|
|
} from "./test_util.ts";
|
2018-09-16 13:35:16 -07:00
|
|
|
|
2019-03-10 04:30:38 +11:00
|
|
|
function deferred(): {
|
|
|
|
promise: Promise<{}>;
|
|
|
|
resolve: (value?: {} | PromiseLike<{}>) => void;
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
reject: (reason?: any) => void;
|
|
|
|
} {
|
2020-02-19 21:36:18 +01:00
|
|
|
let resolve: (value?: {} | PromiseLike<{}>) => void;
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
let reject: ((reason?: any) => void) | undefined = undefined;
|
|
|
|
const promise = new Promise<{}>((res, rej): void => {
|
2019-11-14 05:42:34 +11:00
|
|
|
resolve = res;
|
|
|
|
reject = rej;
|
|
|
|
});
|
2018-09-16 13:35:16 -07:00
|
|
|
return {
|
|
|
|
promise,
|
2020-02-19 21:36:18 +01:00
|
|
|
resolve: resolve!,
|
|
|
|
reject: reject!
|
2018-09-16 13:35:16 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-02-19 21:36:18 +01:00
|
|
|
async function waitForMs(ms: number): Promise<number> {
|
|
|
|
return new Promise((resolve: () => void): number => setTimeout(resolve, ms));
|
2018-09-16 13:35:16 -07:00
|
|
|
}
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timeoutSuccess(): Promise<void> {
|
2018-09-16 13:35:16 -07:00
|
|
|
const { promise, resolve } = deferred();
|
|
|
|
let count = 0;
|
2019-04-21 16:40:10 -04:00
|
|
|
setTimeout((): void => {
|
2018-09-16 13:35:16 -07:00
|
|
|
count++;
|
|
|
|
resolve();
|
|
|
|
}, 500);
|
|
|
|
await promise;
|
|
|
|
// count should increment
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals(count, 1);
|
2018-09-16 13:35:16 -07:00
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timeoutArgs(): Promise<void> {
|
2018-10-08 14:58:44 +08:00
|
|
|
const { promise, resolve } = deferred();
|
2018-10-05 07:29:55 -04:00
|
|
|
const arg = 1;
|
2018-10-08 14:58:44 +08:00
|
|
|
setTimeout(
|
2019-04-21 16:40:10 -04:00
|
|
|
(a, b, c): void => {
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals(a, arg);
|
|
|
|
assertEquals(b, arg.toString());
|
|
|
|
assertEquals(c, [arg]);
|
2018-10-08 14:58:44 +08:00
|
|
|
resolve();
|
|
|
|
},
|
|
|
|
10,
|
|
|
|
arg,
|
|
|
|
arg.toString(),
|
|
|
|
[arg]
|
|
|
|
);
|
|
|
|
await promise;
|
2018-09-05 13:35:29 +08:00
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timeoutCancelSuccess(): Promise<void> {
|
2018-09-16 13:35:16 -07:00
|
|
|
let count = 0;
|
2019-04-21 16:40:10 -04:00
|
|
|
const id = setTimeout((): void => {
|
2018-09-16 13:35:16 -07:00
|
|
|
count++;
|
2019-04-21 14:06:57 -04:00
|
|
|
}, 1);
|
2018-09-16 13:35:16 -07:00
|
|
|
// Cancelled, count should not increment
|
|
|
|
clearTimeout(id);
|
|
|
|
await waitForMs(600);
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals(count, 0);
|
2018-09-16 13:35:16 -07:00
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timeoutCancelMultiple(): Promise<void> {
|
2019-03-10 04:30:38 +11:00
|
|
|
function uncalled(): never {
|
|
|
|
throw new Error("This function should not be called.");
|
|
|
|
}
|
|
|
|
|
2018-10-08 19:52:16 +02:00
|
|
|
// 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 waitForMs(50);
|
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timeoutCancelInvalidSilentFail(): Promise<void> {
|
2018-09-16 13:35:16 -07:00
|
|
|
// Expect no panic
|
|
|
|
const { promise, resolve } = deferred();
|
|
|
|
let count = 0;
|
2019-04-21 16:40:10 -04:00
|
|
|
const id = setTimeout((): void => {
|
2018-09-16 13:35:16 -07:00
|
|
|
count++;
|
|
|
|
// Should have no effect
|
|
|
|
clearTimeout(id);
|
|
|
|
resolve();
|
|
|
|
}, 500);
|
|
|
|
await promise;
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals(count, 1);
|
2018-09-16 13:35:16 -07:00
|
|
|
|
|
|
|
// Should silently fail (no panic)
|
|
|
|
clearTimeout(2147483647);
|
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function intervalSuccess(): Promise<void> {
|
2018-09-16 13:35:16 -07:00
|
|
|
const { promise, resolve } = deferred();
|
|
|
|
let count = 0;
|
2019-04-21 16:40:10 -04:00
|
|
|
const id = setInterval((): void => {
|
2018-09-16 13:35:16 -07:00
|
|
|
count++;
|
2019-04-21 14:06:57 -04:00
|
|
|
clearInterval(id);
|
|
|
|
resolve();
|
|
|
|
}, 100);
|
2018-09-16 13:35:16 -07:00
|
|
|
await promise;
|
|
|
|
// Clear interval
|
|
|
|
clearInterval(id);
|
|
|
|
// count should increment twice
|
2019-04-21 14:06:57 -04:00
|
|
|
assertEquals(count, 1);
|
2018-09-16 13:35:16 -07:00
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function intervalCancelSuccess(): Promise<void> {
|
2018-09-16 13:35:16 -07:00
|
|
|
let count = 0;
|
2019-04-21 16:40:10 -04:00
|
|
|
const id = setInterval((): void => {
|
2018-09-16 13:35:16 -07:00
|
|
|
count++;
|
2019-04-17 13:02:32 -04:00
|
|
|
}, 1);
|
2018-09-16 13:35:16 -07:00
|
|
|
clearInterval(id);
|
2019-04-17 13:02:32 -04:00
|
|
|
await waitForMs(500);
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals(count, 0);
|
2018-09-16 13:35:16 -07:00
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function intervalOrdering(): Promise<void> {
|
2020-02-19 21:36:18 +01:00
|
|
|
const timers: number[] = [];
|
2018-09-05 13:35:29 +08:00
|
|
|
let timeouts = 0;
|
2019-03-10 04:30:38 +11:00
|
|
|
function onTimeout(): void {
|
2018-09-05 13:35:29 +08:00
|
|
|
++timeouts;
|
|
|
|
for (let i = 1; i < timers.length; i++) {
|
|
|
|
clearTimeout(timers[i]);
|
|
|
|
}
|
|
|
|
}
|
2019-03-10 04:30:38 +11:00
|
|
|
for (let i = 0; i < 10; i++) {
|
2019-04-17 13:02:32 -04:00
|
|
|
timers[i] = setTimeout(onTimeout, 1);
|
2019-03-10 04:30:38 +11:00
|
|
|
}
|
2019-04-17 13:02:32 -04:00
|
|
|
await waitForMs(500);
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals(timeouts, 1);
|
2018-09-05 13:35:29 +08:00
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function intervalCancelInvalidSilentFail(): Promise<void> {
|
2018-09-16 13:35:16 -07:00
|
|
|
// Should silently fail (no panic)
|
|
|
|
clearInterval(2147483647);
|
|
|
|
});
|
2019-01-27 06:10:38 +09:00
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function fireCallbackImmediatelyWhenDelayOverMaxValue(): Promise<
|
2019-04-21 16:40:10 -04:00
|
|
|
void
|
|
|
|
> {
|
2019-01-27 06:10:38 +09:00
|
|
|
let count = 0;
|
2019-04-21 16:40:10 -04:00
|
|
|
setTimeout((): void => {
|
2019-01-27 06:10:38 +09:00
|
|
|
count++;
|
|
|
|
}, 2 ** 31);
|
|
|
|
await waitForMs(1);
|
2019-03-06 20:48:46 -05:00
|
|
|
assertEquals(count, 1);
|
2019-01-27 06:10:38 +09:00
|
|
|
});
|
2019-06-11 15:50:36 +08:00
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timeoutCallbackThis(): Promise<void> {
|
2019-06-11 15:50:36 +08:00
|
|
|
const { promise, resolve } = deferred();
|
|
|
|
const obj = {
|
|
|
|
foo(): void {
|
|
|
|
assertEquals(this, window);
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
setTimeout(obj.foo, 1);
|
|
|
|
await promise;
|
|
|
|
});
|
2019-06-13 23:08:27 +08:00
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timeoutBindThis(): Promise<void> {
|
2019-06-13 23:08:27 +08:00
|
|
|
const thisCheckPassed = [null, undefined, window, globalThis];
|
|
|
|
|
|
|
|
const thisCheckFailed = [
|
|
|
|
0,
|
|
|
|
"",
|
|
|
|
true,
|
|
|
|
false,
|
|
|
|
{},
|
|
|
|
[],
|
|
|
|
"foo",
|
|
|
|
(): void => {},
|
|
|
|
Object.prototype
|
|
|
|
];
|
|
|
|
|
2020-03-03 18:22:53 +01:00
|
|
|
for (const thisArg of thisCheckPassed) {
|
|
|
|
const resolvable = createResolvable();
|
|
|
|
let hasThrown = 0;
|
|
|
|
try {
|
|
|
|
setTimeout.call(thisArg, () => resolvable.resolve(), 1);
|
|
|
|
hasThrown = 1;
|
|
|
|
} catch (err) {
|
|
|
|
if (err instanceof TypeError) {
|
|
|
|
hasThrown = 2;
|
|
|
|
} else {
|
|
|
|
hasThrown = 3;
|
2019-06-13 23:08:27 +08:00
|
|
|
}
|
|
|
|
}
|
2020-03-03 18:22:53 +01:00
|
|
|
await resolvable;
|
|
|
|
assertEquals(hasThrown, 1);
|
|
|
|
}
|
2019-06-13 23:08:27 +08:00
|
|
|
|
2020-03-03 18:22:53 +01:00
|
|
|
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;
|
2019-06-13 23:08:27 +08:00
|
|
|
}
|
|
|
|
}
|
2020-03-03 18:22:53 +01:00
|
|
|
assertEquals(hasThrown, 2);
|
|
|
|
}
|
2019-06-13 23:08:27 +08:00
|
|
|
});
|
2019-06-18 01:42:20 +08:00
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function clearTimeoutShouldConvertToNumber(): Promise<void> {
|
2019-06-18 01:42:20 +08:00
|
|
|
let called = false;
|
|
|
|
const obj = {
|
|
|
|
valueOf(): number {
|
|
|
|
called = true;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
clearTimeout((obj as unknown) as number);
|
|
|
|
assert(called);
|
|
|
|
});
|
2019-06-18 21:24:20 +08:00
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(function setTimeoutShouldThrowWithBigint(): void {
|
2019-08-29 22:57:09 +08:00
|
|
|
let hasThrown = 0;
|
|
|
|
try {
|
|
|
|
setTimeout((): void => {}, (1n as unknown) as number);
|
|
|
|
hasThrown = 1;
|
|
|
|
} catch (err) {
|
|
|
|
if (err instanceof TypeError) {
|
|
|
|
hasThrown = 2;
|
|
|
|
} else {
|
|
|
|
hasThrown = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assertEquals(hasThrown, 2);
|
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(function clearTimeoutShouldThrowWithBigint(): void {
|
2019-08-30 23:51:53 +08:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(function testFunctionName(): void {
|
2019-06-18 21:24:20 +08:00
|
|
|
assertEquals(clearTimeout.name, "clearTimeout");
|
|
|
|
assertEquals(clearInterval.name, "clearInterval");
|
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(function testFunctionParamsLength(): void {
|
2019-07-18 18:09:32 +08:00
|
|
|
assertEquals(setTimeout.length, 1);
|
|
|
|
assertEquals(setInterval.length, 1);
|
|
|
|
assertEquals(clearTimeout.length, 0);
|
|
|
|
assertEquals(clearInterval.length, 0);
|
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(function clearTimeoutAndClearIntervalNotBeEquals(): void {
|
2019-06-18 21:24:20 +08:00
|
|
|
assertNotEquals(clearTimeout, clearInterval);
|
|
|
|
});
|
2019-10-19 17:09:24 -04:00
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timerMaxCpuBug(): Promise<void> {
|
2019-10-19 17:09:24 -04:00
|
|
|
// 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 { opsDispatched } = Deno.metrics();
|
|
|
|
await waitForMs(100);
|
|
|
|
const opsDispatched_ = Deno.metrics().opsDispatched;
|
|
|
|
assert(opsDispatched_ - opsDispatched < 10);
|
|
|
|
});
|
2019-12-03 19:19:03 -08:00
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timerBasicMicrotaskOrdering(): Promise<void> {
|
2019-12-03 19:19:03 -08:00
|
|
|
let s = "";
|
|
|
|
let count = 0;
|
|
|
|
const { promise, resolve } = deferred();
|
|
|
|
setTimeout(() => {
|
|
|
|
Promise.resolve().then(() => {
|
|
|
|
count++;
|
|
|
|
s += "de";
|
|
|
|
if (count === 2) {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
setTimeout(() => {
|
|
|
|
count++;
|
|
|
|
s += "no";
|
|
|
|
if (count === 2) {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
await promise;
|
|
|
|
assertEquals(s, "deno");
|
|
|
|
});
|
|
|
|
|
2020-03-04 17:31:14 +01:00
|
|
|
unitTest(async function timerNestedMicrotaskOrdering(): Promise<void> {
|
2019-12-03 19:19:03 -08:00
|
|
|
let s = "";
|
|
|
|
const { promise, resolve } = deferred();
|
|
|
|
s += "0";
|
|
|
|
setTimeout(() => {
|
|
|
|
s += "4";
|
|
|
|
setTimeout(() => (s += "8"));
|
|
|
|
Promise.resolve().then(() => {
|
|
|
|
setTimeout(() => {
|
|
|
|
s += "9";
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
setTimeout(() => (s += "5"));
|
|
|
|
Promise.resolve().then(() => (s += "2"));
|
|
|
|
Promise.resolve().then(() =>
|
|
|
|
setTimeout(() => {
|
|
|
|
s += "6";
|
|
|
|
Promise.resolve().then(() => (s += "7"));
|
|
|
|
})
|
|
|
|
);
|
|
|
|
Promise.resolve().then(() => Promise.resolve().then(() => (s += "3")));
|
|
|
|
s += "1";
|
|
|
|
await promise;
|
|
|
|
assertEquals(s, "0123456789");
|
|
|
|
});
|