summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--js/workers.ts9
-rw-r--r--tests/subdir/bench_worker.ts20
-rw-r--r--tests/workers_round_robin_bench.ts75
-rw-r--r--tests/workers_startup_bench.ts25
-rwxr-xr-xtools/benchmark.py2
-rw-r--r--website/benchmarks.html11
6 files changed, 139 insertions, 3 deletions
diff --git a/js/workers.ts b/js/workers.ts
index 601ffa0b1..8c08a8506 100644
--- a/js/workers.ts
+++ b/js/workers.ts
@@ -150,11 +150,13 @@ export interface Worker {
onmessage?: (e: { data: any }) => void;
onmessageerror?: () => void;
postMessage(data: any): void;
+ closed: Promise<void>;
}
export class WorkerImpl implements Worker {
private readonly rid: number;
private isClosing: boolean = false;
+ private readonly isClosedPromise: Promise<void>;
public onerror?: () => void;
public onmessage?: (data: any) => void;
public onmessageerror?: () => void;
@@ -162,11 +164,16 @@ export class WorkerImpl implements Worker {
constructor(specifier: string) {
this.rid = createWorker(specifier);
this.run();
- hostGetWorkerClosed(this.rid).then(() => {
+ this.isClosedPromise = hostGetWorkerClosed(this.rid);
+ this.isClosedPromise.then(() => {
this.isClosing = true;
});
}
+ get closed(): Promise<void> {
+ return this.isClosedPromise;
+ }
+
postMessage(data: any): void {
hostPostMessage(this.rid, data);
}
diff --git a/tests/subdir/bench_worker.ts b/tests/subdir/bench_worker.ts
new file mode 100644
index 000000000..6dd2f9541
--- /dev/null
+++ b/tests/subdir/bench_worker.ts
@@ -0,0 +1,20 @@
+onmessage = function(e) {
+ const { cmdId, action, data } = e.data;
+ switch (action) {
+ case 0: // Static response
+ postMessage({
+ cmdId,
+ data: "HTTP/1.1 200 OK\r\nContent-Length: 12\r\n\r\nHello World\n"
+ });
+ break;
+ case 1: // Respond with request data
+ postMessage({ cmdId, data });
+ break;
+ case 2: // Ping
+ postMessage({ cmdId });
+ break;
+ case 3: // Close
+ workerClose();
+ break;
+ }
+};
diff --git a/tests/workers_round_robin_bench.ts b/tests/workers_round_robin_bench.ts
new file mode 100644
index 000000000..818a1145f
--- /dev/null
+++ b/tests/workers_round_robin_bench.ts
@@ -0,0 +1,75 @@
+// Benchmark measures time it takes to send a message to a group of workers one
+// at a time and wait for a response from all of them. Just a general
+// throughput and consistency benchmark.
+const data = "HTTP/1.1 200 OK\r\nContent-Length: 12\r\n\r\nHello World\n";
+const workerCount = 4;
+const cmdsPerWorker = 400;
+
+export interface ResolvableMethods<T> {
+ resolve: (value?: T | PromiseLike<T>) => void;
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ reject: (reason?: any) => void;
+}
+
+export type Resolvable<T> = Promise<T> & ResolvableMethods<T>;
+
+export function createResolvable<T>(): Resolvable<T> {
+ let methods: ResolvableMethods<T>;
+ const promise = new Promise<T>((resolve, reject) => {
+ methods = { resolve, reject };
+ });
+ // TypeScript doesn't know that the Promise callback occurs synchronously
+ // therefore use of not null assertion (`!`)
+ return Object.assign(promise, methods!) as Resolvable<T>;
+}
+
+function handleAsyncMsgFromWorker(
+ promiseTable: Map<number, Resolvable<string>>,
+ msg: { cmdId: number; data: string }
+): void {
+ const promise = promiseTable.get(msg.cmdId);
+ if (promise === null) {
+ throw new Error(`Failed to find promise: cmdId: ${msg.cmdId}, msg: ${msg}`);
+ }
+ promise.resolve(data);
+}
+
+async function main(): Promise<void> {
+ const workers: Array<[Map<number, Resolvable<string>>, Worker]> = [];
+ for (var i = 1; i <= workerCount; ++i) {
+ const worker = new Worker("tests/subdir/bench_worker.ts");
+ const promise = new Promise(resolve => {
+ worker.onmessage = e => {
+ if (e.data.cmdId === 0) resolve();
+ };
+ });
+ worker.postMessage({ cmdId: 0, action: 2 });
+ await promise;
+ workers.push([new Map(), worker]);
+ }
+ // assign callback function
+ for (const [promiseTable, worker] of workers) {
+ worker.onmessage = e => {
+ handleAsyncMsgFromWorker(promiseTable, e.data);
+ };
+ }
+ for (const cmdId of Array(cmdsPerWorker).keys()) {
+ const promises: Array<Promise<string>> = [];
+ for (const [promiseTable, worker] of workers) {
+ const promise = createResolvable<string>();
+ promiseTable.set(cmdId, promise);
+ worker.postMessage({ cmdId: cmdId, action: 1, data });
+ promises.push(promise);
+ }
+ for (const promise of promises) {
+ await promise;
+ }
+ }
+ for (const [, worker] of workers) {
+ worker.postMessage({ action: 3 });
+ await worker.closed; // Required to avoid a cmdId not in table error.
+ }
+ console.log("Finished!");
+}
+
+main();
diff --git a/tests/workers_startup_bench.ts b/tests/workers_startup_bench.ts
new file mode 100644
index 000000000..46b2b2801
--- /dev/null
+++ b/tests/workers_startup_bench.ts
@@ -0,0 +1,25 @@
+// Benchmark measures time it takes to start and stop a number of workers.
+const workerCount = 50;
+
+async function bench(): Promise<void> {
+ const workers: Worker[] = [];
+ for (var i = 1; i <= workerCount; ++i) {
+ const worker = new Worker("tests/subdir/bench_worker.ts");
+ const promise = new Promise(resolve => {
+ worker.onmessage = e => {
+ if (e.data.cmdId === 0) resolve();
+ };
+ });
+ worker.postMessage({ cmdId: 0, action: 2 });
+ await promise;
+ workers.push(worker);
+ }
+ console.log("Done creating workers closing workers!");
+ for (const worker of workers) {
+ worker.postMessage({ action: 3 });
+ await worker.closed; // Required to avoid a cmdId not in table error.
+ }
+ console.log("Finished!");
+}
+
+bench();
diff --git a/tools/benchmark.py b/tools/benchmark.py
index 772e35d40..db2ef0fb2 100755
--- a/tools/benchmark.py
+++ b/tools/benchmark.py
@@ -25,6 +25,8 @@ exec_time_benchmarks = [
("error_001", ["tests/error_001.ts"]),
("cold_hello", ["tests/002_hello.ts", "--reload"]),
("cold_relative_import", ["tests/003_relative_import.ts", "--reload"]),
+ ("workers_startup", ["tests/workers_startup_bench.ts"]),
+ ("workers_round_robin", ["tests/workers_round_robin_bench.ts"]),
]
gh_pages_data_file = "gh-pages/data.json"
diff --git a/website/benchmarks.html b/website/benchmarks.html
index 4d6d543dc..560b96c7c 100644
--- a/website/benchmarks.html
+++ b/website/benchmarks.html
@@ -29,11 +29,18 @@
href="https://github.com/denoland/deno/blob/master/tests/002_hello.ts"
>
tests/002_hello.ts
- </a>
- and
+ </a>,
<a
href="https://github.com/denoland/deno/blob/master/tests/003_relative_import.ts"
>tests/003_relative_import.ts</a
+ >,
+ <a
+ href="https://github.com/denoland/deno/blob/master/tests/worker_round_robin_bench.ts"
+ >tests/worker_round_robin_bench.ts</a
+ >, and
+ <a
+ href="https://github.com/denoland/deno/blob/master/tests/worker_startup_bench.ts"
+ >tests/worker_startup_bench.ts</a
>. For deno to execute typescript, it must first compile it to JS. A
warm startup is when deno has a cached JS output already, so it should
be fast because it bypasses the TS compiler. A cold startup is when deno