2022-04-13 20:46:52 +08:00
|
|
|
import { preloadModules } from "../../common/preload_modules";
|
|
|
|
|
2022-03-27 15:55:29 +08:00
|
|
|
const { parentPort, workerData } = require("worker_threads");
|
2022-04-09 20:28:41 +08:00
|
|
|
// @ts-ignore
|
2022-03-27 15:55:29 +08:00
|
|
|
let vm2 = `${workerData}/vm2`;
|
|
|
|
const { VM, VMScript } = require(vm2);
|
|
|
|
|
|
|
|
// console.log("Process env", process.env);
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
let loadedFunctions = new Map<string, Function>();
|
|
|
|
let pendingRequests = new Map<
|
2022-03-21 22:21:34 +08:00
|
|
|
number,
|
|
|
|
{
|
|
|
|
resolve: (result: unknown) => void;
|
|
|
|
reject: (e: any) => void;
|
|
|
|
}
|
|
|
|
>();
|
2022-03-20 16:56:28 +08:00
|
|
|
|
2022-03-24 17:48:56 +08:00
|
|
|
let syscallReqId = 0;
|
|
|
|
|
2022-03-20 16:56:28 +08:00
|
|
|
let vm = new VM({
|
|
|
|
sandbox: {
|
2022-04-09 20:28:41 +08:00
|
|
|
console,
|
2022-04-13 20:46:52 +08:00
|
|
|
require: (moduleName: string): any => {
|
|
|
|
console.log("Loading", moduleName);
|
|
|
|
if (preloadModules.includes(moduleName)) {
|
|
|
|
return require(`${workerData}/${moduleName}`);
|
|
|
|
} else {
|
|
|
|
throw Error("Cannot import arbitrary modules");
|
|
|
|
}
|
|
|
|
},
|
2022-03-20 16:56:28 +08:00
|
|
|
self: {
|
2022-03-24 17:48:56 +08:00
|
|
|
syscall: (name: string, ...args: any[]) => {
|
2022-03-20 16:56:28 +08:00
|
|
|
return new Promise((resolve, reject) => {
|
2022-03-24 17:48:56 +08:00
|
|
|
syscallReqId++;
|
|
|
|
pendingRequests.set(syscallReqId, { resolve, reject });
|
2022-03-20 16:56:28 +08:00
|
|
|
parentPort.postMessage({
|
|
|
|
type: "syscall",
|
2022-03-24 17:48:56 +08:00
|
|
|
id: syscallReqId,
|
2022-03-20 16:56:28 +08:00
|
|
|
name,
|
|
|
|
// TODO: Figure out why this is necessary (to avoide a CloneError)
|
|
|
|
args: JSON.parse(JSON.stringify(args)),
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-03-21 22:21:34 +08:00
|
|
|
function wrapScript(code: string) {
|
2022-03-20 16:56:28 +08:00
|
|
|
return `(${code})["default"]`;
|
|
|
|
}
|
|
|
|
|
2022-03-21 22:21:34 +08:00
|
|
|
function safeRun(fn: any) {
|
|
|
|
fn().catch((e: any) => {
|
2022-03-20 16:56:28 +08:00
|
|
|
console.error(e);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-03-21 22:21:34 +08:00
|
|
|
parentPort.on("message", (data: any) => {
|
2022-03-20 16:56:28 +08:00
|
|
|
safeRun(async () => {
|
|
|
|
switch (data.type) {
|
|
|
|
case "load":
|
|
|
|
loadedFunctions.set(data.name, new VMScript(wrapScript(data.code)));
|
|
|
|
parentPort.postMessage({
|
|
|
|
type: "inited",
|
|
|
|
name: data.name,
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case "invoke":
|
|
|
|
let fn = loadedFunctions.get(data.name);
|
|
|
|
if (!fn) {
|
|
|
|
throw new Error(`Function not loaded: ${data.name}`);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
let r = vm.run(fn);
|
|
|
|
let result = await Promise.resolve(r(...data.args));
|
|
|
|
parentPort.postMessage({
|
|
|
|
type: "result",
|
|
|
|
id: data.id,
|
2022-03-21 22:21:34 +08:00
|
|
|
// TOOD: Figure out if this is necessary, because it's expensive
|
|
|
|
result: result && JSON.parse(JSON.stringify(result)),
|
2022-03-20 16:56:28 +08:00
|
|
|
});
|
2022-03-21 22:21:34 +08:00
|
|
|
} catch (e: any) {
|
2022-03-20 16:56:28 +08:00
|
|
|
parentPort.postMessage({
|
|
|
|
type: "result",
|
|
|
|
id: data.id,
|
|
|
|
error: e.message,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "syscall-response":
|
|
|
|
let syscallId = data.id;
|
|
|
|
const lookup = pendingRequests.get(syscallId);
|
|
|
|
if (!lookup) {
|
|
|
|
console.log(
|
|
|
|
"Current outstanding requests",
|
|
|
|
pendingRequests,
|
|
|
|
"looking up",
|
|
|
|
syscallId
|
|
|
|
);
|
|
|
|
throw Error("Invalid request id");
|
|
|
|
}
|
|
|
|
pendingRequests.delete(syscallId);
|
|
|
|
if (data.error) {
|
2022-03-28 21:25:05 +08:00
|
|
|
console.log("Got rejection", data.error);
|
2022-03-20 16:56:28 +08:00
|
|
|
lookup.reject(new Error(data.error));
|
|
|
|
} else {
|
|
|
|
lookup.resolve(data.result);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|