mirror of
https://github.com/denoland/deno.git
synced 2025-01-11 16:42:21 -05:00
22f951aa67
Fixes panic occurring in worker when "self.close()" is called at the top level, ie. worker shuts down while module evaluation promise hasn't yet resolved.
359 lines
8.8 KiB
TypeScript
359 lines
8.8 KiB
TypeScript
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
|
|
// Requires to be run with `--allow-net` flag
|
|
|
|
// FIXME(bartlomieju): this file is an integration test only because
|
|
// workers are leaking ops at the moment - `worker.terminate()` is not
|
|
// yet implemented. Once it gets implemented this file should be
|
|
// again moved to `cli/js/` as an unit test file.
|
|
|
|
import { assert, assertEquals } from "../../std/testing/asserts.ts";
|
|
import { deferred } from "../../std/async/deferred.ts";
|
|
|
|
Deno.test({
|
|
name: "worker terminate",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
|
|
const jsWorker = new Worker(
|
|
new URL("subdir/test_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
const tsWorker = new Worker(
|
|
new URL("subdir/test_worker.ts", import.meta.url).href,
|
|
{ type: "module", name: "tsWorker" },
|
|
);
|
|
|
|
tsWorker.onmessage = (e): void => {
|
|
assertEquals(e.data, "Hello World");
|
|
promise.resolve();
|
|
};
|
|
|
|
jsWorker.onmessage = (e): void => {
|
|
assertEquals(e.data, "Hello World");
|
|
tsWorker.postMessage("Hello World");
|
|
};
|
|
|
|
jsWorker.onerror = (e: Event): void => {
|
|
e.preventDefault();
|
|
jsWorker.postMessage("Hello World");
|
|
};
|
|
|
|
jsWorker.postMessage("Hello World");
|
|
await promise;
|
|
tsWorker.terminate();
|
|
jsWorker.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker nested",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
|
|
const nestedWorker = new Worker(
|
|
new URL("subdir/nested_worker.js", import.meta.url).href,
|
|
{ type: "module", name: "nested" },
|
|
);
|
|
|
|
nestedWorker.onmessage = (e): void => {
|
|
assert(e.data.type !== "error");
|
|
promise.resolve();
|
|
};
|
|
|
|
nestedWorker.postMessage("Hello World");
|
|
await promise;
|
|
nestedWorker.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker throws when executing",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
const throwingWorker = new Worker(
|
|
new URL("subdir/throwing_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
throwingWorker.onerror = (e: any): void => {
|
|
e.preventDefault();
|
|
assert(/Uncaught Error: Thrown error/.test(e.message));
|
|
promise.resolve();
|
|
};
|
|
|
|
await promise;
|
|
throwingWorker.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker globals",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
const w = new Worker(
|
|
new URL("subdir/worker_globals.ts", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
w.onmessage = (e): void => {
|
|
assertEquals(e.data, "true, true, true");
|
|
promise.resolve();
|
|
};
|
|
w.postMessage("Hello, world!");
|
|
await promise;
|
|
w.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker fetch API",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
|
|
const fetchingWorker = new Worker(
|
|
new URL("subdir/fetching_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
|
|
// deno-lint-ignore no-explicit-any
|
|
fetchingWorker.onerror = (e: any): void => {
|
|
e.preventDefault();
|
|
promise.reject(e.message);
|
|
};
|
|
|
|
// Defer promise.resolve() to allow worker to shut down
|
|
fetchingWorker.onmessage = (e): void => {
|
|
assert(e.data === "Done!");
|
|
promise.resolve();
|
|
};
|
|
|
|
await promise;
|
|
fetchingWorker.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker terminate busy loop",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
|
|
const busyWorker = new Worker(
|
|
new URL("subdir/busy_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
|
|
let testResult = 0;
|
|
|
|
busyWorker.onmessage = (e): void => {
|
|
testResult = e.data;
|
|
if (testResult >= 10000) {
|
|
busyWorker.terminate();
|
|
busyWorker.onmessage = (_e): void => {
|
|
throw new Error("unreachable");
|
|
};
|
|
setTimeout(() => {
|
|
assertEquals(testResult, 10000);
|
|
promise.resolve();
|
|
}, 100);
|
|
}
|
|
};
|
|
|
|
busyWorker.postMessage("ping");
|
|
await promise;
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker race condition",
|
|
fn: async function (): Promise<void> {
|
|
// See issue for details
|
|
// https://github.com/denoland/deno/issues/4080
|
|
const promise = deferred();
|
|
|
|
const racyWorker = new Worker(
|
|
new URL("subdir/racy_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
|
|
racyWorker.onmessage = (e): void => {
|
|
assertEquals(e.data.buf.length, 999999);
|
|
racyWorker.onmessage = (_e): void => {
|
|
throw new Error("unreachable");
|
|
};
|
|
setTimeout(() => {
|
|
promise.resolve();
|
|
}, 100);
|
|
};
|
|
|
|
await promise;
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker is event listener",
|
|
fn: async function (): Promise<void> {
|
|
let messageHandlersCalled = 0;
|
|
let errorHandlersCalled = 0;
|
|
|
|
const promise1 = deferred();
|
|
const promise2 = deferred();
|
|
|
|
const worker = new Worker(
|
|
new URL("subdir/event_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
|
|
worker.onmessage = (_e: Event): void => {
|
|
messageHandlersCalled++;
|
|
};
|
|
worker.addEventListener("message", (_e: Event) => {
|
|
messageHandlersCalled++;
|
|
});
|
|
worker.addEventListener("message", (_e: Event) => {
|
|
messageHandlersCalled++;
|
|
promise1.resolve();
|
|
});
|
|
|
|
worker.onerror = (e): void => {
|
|
errorHandlersCalled++;
|
|
e.preventDefault();
|
|
};
|
|
worker.addEventListener("error", (_e: Event) => {
|
|
errorHandlersCalled++;
|
|
});
|
|
worker.addEventListener("error", (_e: Event) => {
|
|
errorHandlersCalled++;
|
|
promise2.resolve();
|
|
});
|
|
|
|
worker.postMessage("ping");
|
|
await promise1;
|
|
assertEquals(messageHandlersCalled, 3);
|
|
|
|
worker.postMessage("boom");
|
|
await promise2;
|
|
assertEquals(errorHandlersCalled, 3);
|
|
worker.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker scope is event listener",
|
|
fn: async function (): Promise<void> {
|
|
const promise1 = deferred();
|
|
|
|
const worker = new Worker(
|
|
new URL("subdir/event_worker_scope.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
|
|
worker.onmessage = (e: MessageEvent): void => {
|
|
const { messageHandlersCalled, errorHandlersCalled } = e.data;
|
|
assertEquals(messageHandlersCalled, 4);
|
|
assertEquals(errorHandlersCalled, 4);
|
|
promise1.resolve();
|
|
};
|
|
|
|
worker.onerror = (_e): void => {
|
|
throw new Error("unreachable");
|
|
};
|
|
|
|
worker.postMessage("boom");
|
|
worker.postMessage("ping");
|
|
await promise1;
|
|
worker.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker with Deno namespace",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
const promise2 = deferred();
|
|
|
|
const regularWorker = new Worker(
|
|
new URL("subdir/non_deno_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
const denoWorker = new Worker(
|
|
new URL("subdir/deno_worker.ts", import.meta.url).href,
|
|
{ type: "module", deno: true },
|
|
);
|
|
|
|
regularWorker.onmessage = (e): void => {
|
|
assertEquals(e.data, "Hello World");
|
|
regularWorker.terminate();
|
|
promise.resolve();
|
|
};
|
|
|
|
denoWorker.onmessage = (e): void => {
|
|
assertEquals(e.data, "Hello World");
|
|
denoWorker.terminate();
|
|
promise2.resolve();
|
|
};
|
|
|
|
regularWorker.postMessage("Hello World");
|
|
await promise;
|
|
denoWorker.postMessage("Hello World");
|
|
await promise2;
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "worker with crypto in scope",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
const w = new Worker(
|
|
new URL("subdir/worker_crypto.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
w.onmessage = (e): void => {
|
|
assertEquals(e.data, true);
|
|
promise.resolve();
|
|
};
|
|
w.postMessage(null);
|
|
await promise;
|
|
w.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "Worker event handler order",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
const w = new Worker(
|
|
new URL("subdir/test_worker.ts", import.meta.url).href,
|
|
{ type: "module", name: "tsWorker" },
|
|
);
|
|
const arr: number[] = [];
|
|
w.addEventListener("message", () => arr.push(1));
|
|
w.onmessage = (e): void => {
|
|
arr.push(2);
|
|
};
|
|
w.addEventListener("message", () => arr.push(3));
|
|
w.addEventListener("message", () => {
|
|
assertEquals(arr, [1, 2, 3]);
|
|
promise.resolve();
|
|
});
|
|
w.postMessage("Hello World");
|
|
await promise;
|
|
w.terminate();
|
|
},
|
|
});
|
|
|
|
Deno.test({
|
|
name: "Worker immediate close",
|
|
fn: async function (): Promise<void> {
|
|
const promise = deferred();
|
|
const w = new Worker(
|
|
new URL("./immediately_close_worker.js", import.meta.url).href,
|
|
{ type: "module" },
|
|
);
|
|
setTimeout(() => {
|
|
promise.resolve();
|
|
}, 1000);
|
|
await promise;
|
|
w.terminate();
|
|
},
|
|
});
|