2022-04-05 23:02:17 +08:00
|
|
|
import { safeRun } from "../util";
|
2022-05-09 20:59:12 +08:00
|
|
|
import { ConsoleLogger } from "./custom_logger";
|
2022-04-05 23:02:17 +08:00
|
|
|
import { ControllerMessage, WorkerMessage } from "./worker";
|
2022-03-20 16:56:28 +08:00
|
|
|
|
|
|
|
let loadedFunctions = new Map<string, Function>();
|
|
|
|
let pendingRequests = new Map<
|
|
|
|
number,
|
|
|
|
{
|
|
|
|
resolve: (result: unknown) => void;
|
|
|
|
reject: (e: any) => void;
|
|
|
|
}
|
|
|
|
>();
|
|
|
|
|
2022-04-04 21:25:07 +08:00
|
|
|
function workerPostMessage(msg: ControllerMessage) {
|
|
|
|
if (typeof window !== "undefined" && window.parent !== window) {
|
|
|
|
window.parent.postMessage(msg, "*");
|
|
|
|
} else {
|
|
|
|
self.postMessage(msg);
|
|
|
|
}
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
|
|
|
|
2022-03-24 17:48:56 +08:00
|
|
|
declare global {
|
|
|
|
function syscall(name: string, ...args: any[]): Promise<any>;
|
2022-04-13 20:46:52 +08:00
|
|
|
// function require(moduleName: string): any;
|
2022-03-24 17:48:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
let syscallReqId = 0;
|
|
|
|
|
|
|
|
self.syscall = async (name: string, ...args: any[]) => {
|
2022-03-20 16:56:28 +08:00
|
|
|
return await new Promise((resolve, reject) => {
|
2022-03-24 17:48:56 +08:00
|
|
|
syscallReqId++;
|
|
|
|
pendingRequests.set(syscallReqId, { resolve, reject });
|
2022-04-04 21:25:07 +08:00
|
|
|
workerPostMessage({
|
|
|
|
type: "syscall",
|
|
|
|
id: syscallReqId,
|
|
|
|
name,
|
|
|
|
args,
|
|
|
|
});
|
2022-03-20 16:56:28 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2022-04-13 20:46:52 +08:00
|
|
|
const preloadedModules: { [key: string]: any } = {
|
|
|
|
"@lezer/lr": require("@lezer/lr"),
|
|
|
|
yaml: require("yaml"),
|
2022-04-28 17:55:38 +08:00
|
|
|
handlebars: require("handlebars/dist/handlebars"),
|
2022-04-13 20:46:52 +08:00
|
|
|
};
|
|
|
|
// for (const moduleName of preloadModules) {
|
|
|
|
// preloadedModules[moduleName] = require(moduleName);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
self.require = (moduleName: string): any => {
|
2022-04-21 17:46:33 +08:00
|
|
|
// console.log("Loading", moduleName, preloadedModules[moduleName]);
|
2022-04-13 20:46:52 +08:00
|
|
|
return preloadedModules[moduleName];
|
|
|
|
};
|
|
|
|
|
2022-05-09 20:59:12 +08:00
|
|
|
// @ts-ignore
|
|
|
|
self.console = new ConsoleLogger((level, message) => {
|
|
|
|
workerPostMessage({ type: "log", level, message });
|
|
|
|
}, false);
|
|
|
|
|
2022-04-01 21:02:35 +08:00
|
|
|
function wrapScript(code: string) {
|
|
|
|
return `return (${code})["default"]`;
|
2022-03-20 16:56:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
self.addEventListener("message", (event: { data: WorkerMessage }) => {
|
|
|
|
safeRun(async () => {
|
2022-03-29 18:13:46 +08:00
|
|
|
let data = event.data;
|
2022-03-20 16:56:28 +08:00
|
|
|
switch (data.type) {
|
|
|
|
case "load":
|
2022-04-01 21:02:35 +08:00
|
|
|
let fn2 = new Function(wrapScript(data.code!));
|
|
|
|
loadedFunctions.set(data.name!, fn2());
|
2022-04-04 21:25:07 +08:00
|
|
|
workerPostMessage({
|
|
|
|
type: "inited",
|
|
|
|
name: data.name,
|
|
|
|
});
|
2022-03-20 16:56:28 +08:00
|
|
|
break;
|
|
|
|
case "invoke":
|
|
|
|
let fn = loadedFunctions.get(data.name!);
|
|
|
|
if (!fn) {
|
|
|
|
throw new Error(`Function not loaded: ${data.name}`);
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
let result = await Promise.resolve(fn(...(data.args || [])));
|
2022-04-04 21:25:07 +08:00
|
|
|
workerPostMessage({
|
|
|
|
type: "result",
|
|
|
|
id: data.id,
|
|
|
|
result: result,
|
|
|
|
} as ControllerMessage);
|
2022-03-20 16:56:28 +08:00
|
|
|
} catch (e: any) {
|
2022-04-04 21:25:07 +08:00
|
|
|
workerPostMessage({
|
|
|
|
type: "result",
|
|
|
|
id: data.id,
|
|
|
|
error: e.message,
|
|
|
|
});
|
2022-03-20 16:56:28 +08:00
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
|
|
|
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) {
|
|
|
|
lookup.reject(new Error(data.error));
|
|
|
|
} else {
|
|
|
|
lookup.resolve(data.result);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|