mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 09:03:42 -05:00
Add worker benchmarks (#2059)
This commit is contained in:
parent
031411b449
commit
7a3df0a184
6 changed files with 139 additions and 3 deletions
|
@ -150,11 +150,13 @@ export interface Worker {
|
||||||
onmessage?: (e: { data: any }) => void;
|
onmessage?: (e: { data: any }) => void;
|
||||||
onmessageerror?: () => void;
|
onmessageerror?: () => void;
|
||||||
postMessage(data: any): void;
|
postMessage(data: any): void;
|
||||||
|
closed: Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class WorkerImpl implements Worker {
|
export class WorkerImpl implements Worker {
|
||||||
private readonly rid: number;
|
private readonly rid: number;
|
||||||
private isClosing: boolean = false;
|
private isClosing: boolean = false;
|
||||||
|
private readonly isClosedPromise: Promise<void>;
|
||||||
public onerror?: () => void;
|
public onerror?: () => void;
|
||||||
public onmessage?: (data: any) => void;
|
public onmessage?: (data: any) => void;
|
||||||
public onmessageerror?: () => void;
|
public onmessageerror?: () => void;
|
||||||
|
@ -162,11 +164,16 @@ export class WorkerImpl implements Worker {
|
||||||
constructor(specifier: string) {
|
constructor(specifier: string) {
|
||||||
this.rid = createWorker(specifier);
|
this.rid = createWorker(specifier);
|
||||||
this.run();
|
this.run();
|
||||||
hostGetWorkerClosed(this.rid).then(() => {
|
this.isClosedPromise = hostGetWorkerClosed(this.rid);
|
||||||
|
this.isClosedPromise.then(() => {
|
||||||
this.isClosing = true;
|
this.isClosing = true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get closed(): Promise<void> {
|
||||||
|
return this.isClosedPromise;
|
||||||
|
}
|
||||||
|
|
||||||
postMessage(data: any): void {
|
postMessage(data: any): void {
|
||||||
hostPostMessage(this.rid, data);
|
hostPostMessage(this.rid, data);
|
||||||
}
|
}
|
||||||
|
|
20
tests/subdir/bench_worker.ts
Normal file
20
tests/subdir/bench_worker.ts
Normal file
|
@ -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;
|
||||||
|
}
|
||||||
|
};
|
75
tests/workers_round_robin_bench.ts
Normal file
75
tests/workers_round_robin_bench.ts
Normal file
|
@ -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();
|
25
tests/workers_startup_bench.ts
Normal file
25
tests/workers_startup_bench.ts
Normal file
|
@ -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();
|
|
@ -25,6 +25,8 @@ exec_time_benchmarks = [
|
||||||
("error_001", ["tests/error_001.ts"]),
|
("error_001", ["tests/error_001.ts"]),
|
||||||
("cold_hello", ["tests/002_hello.ts", "--reload"]),
|
("cold_hello", ["tests/002_hello.ts", "--reload"]),
|
||||||
("cold_relative_import", ["tests/003_relative_import.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"
|
gh_pages_data_file = "gh-pages/data.json"
|
||||||
|
|
|
@ -29,11 +29,18 @@
|
||||||
href="https://github.com/denoland/deno/blob/master/tests/002_hello.ts"
|
href="https://github.com/denoland/deno/blob/master/tests/002_hello.ts"
|
||||||
>
|
>
|
||||||
tests/002_hello.ts
|
tests/002_hello.ts
|
||||||
</a>
|
</a>,
|
||||||
and
|
|
||||||
<a
|
<a
|
||||||
href="https://github.com/denoland/deno/blob/master/tests/003_relative_import.ts"
|
href="https://github.com/denoland/deno/blob/master/tests/003_relative_import.ts"
|
||||||
>tests/003_relative_import.ts</a
|
>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
|
>. 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
|
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
|
be fast because it bypasses the TS compiler. A cold startup is when deno
|
||||||
|
|
Loading…
Reference in a new issue