worker-testbed
Version:
The AI-ready testbed which use Worker Threads to create isolated vm context
88 lines (84 loc) • 2.71 kB
JavaScript
;
var worker_threads = require('worker_threads');
var url = require('url');
var functoolsKit = require('functools-kit');
var tape = require('tape');
const workerFileSubject = new functoolsKit.BehaviorSubject();
const finishSubject = new functoolsKit.Subject();
let testRegistry = new functoolsKit.ToolRegistry("workertest");
let testCounter = 0;
const waitForFile = async () => {
if (workerFileSubject.data) {
return workerFileSubject.data;
}
return await workerFileSubject.toPromise();
};
class TestWrapper {
testName;
cb;
constructor(testName, cb) {
this.testName = testName;
this.cb = cb;
}
}
const test = async (testName, cb) => {
if (!worker_threads.isMainThread) {
testRegistry = testRegistry.register(testName, new TestWrapper(testName, cb));
return;
}
const workerFile = await waitForFile();
testCounter += 1;
tape(testName, async (test) => {
const [awaiter, { resolve }] = functoolsKit.createAwaiter();
let isFinished = false;
const worker = new worker_threads.Worker(workerFile, {
workerData: { testName },
});
worker.once("message", ({ status, msg }) => {
if (status === "pass") {
test.pass(msg);
}
else if (status === "fail") {
test.fail(msg);
}
isFinished = true;
worker.terminate();
resolve();
});
worker.on("error", (err) => {
test.fail(`Worker error: ${functoolsKit.getErrorMessage(err)}`);
resolve();
});
worker.on("exit", (code) => {
if (isFinished) {
return;
}
if (code !== 0) {
test.fail(`Worker stopped with exit code ${code}`);
resolve();
}
});
{
await awaiter;
testCounter -= 1;
await finishSubject.next();
}
});
};
const run = functoolsKit.singleshot(async (__filename, cb = () => { }) => {
if (worker_threads.isMainThread) {
await workerFileSubject.next(url.fileURLToPath(__filename));
await finishSubject.filter(() => testCounter === 0).toPromise();
cb();
return;
}
if (!worker_threads.parentPort) {
throw new Error("workertest parentPort is null");
}
testRegistry.get(worker_threads.workerData.testName).cb({
pass: (msg) => worker_threads.parentPort.postMessage({ status: "pass", msg }),
fail: (msg) => worker_threads.parentPort.postMessage({ status: "fail", msg }),
});
});
exports.run = run;
exports.test = test;