silverbullet/plugos/sandbox.ts

112 lines
2.9 KiB
TypeScript
Raw Normal View History

2022-03-23 22:41:12 +08:00
import {
ControllerMessage,
WorkerLike,
WorkerMessage,
} from "./environment/worker";
2022-03-25 19:03:06 +08:00
import { Plug } from "./plug";
export type SandboxFactory<HookT> = (plug: Plug<HookT>) => Sandbox;
2022-03-23 22:41:12 +08:00
export class Sandbox {
protected worker: WorkerLike;
protected reqId = 0;
protected outstandingInits = new Map<string, () => void>();
protected outstandingInvocations = new Map<
number,
{ resolve: (result: any) => void; reject: (e: any) => void }
>();
protected loadedFunctions = new Set<string>();
2022-03-25 19:03:06 +08:00
protected plug: Plug<any>;
2022-03-23 22:41:12 +08:00
2022-03-25 19:03:06 +08:00
constructor(plug: Plug<any>, worker: WorkerLike) {
2022-03-23 22:41:12 +08:00
worker.onMessage = this.onMessage.bind(this);
this.worker = worker;
2022-03-25 19:03:06 +08:00
this.plug = plug;
2022-03-23 22:41:12 +08:00
}
isLoaded(name: string) {
return this.loadedFunctions.has(name);
}
async load(name: string, code: string): Promise<void> {
await this.worker.ready;
2022-03-28 14:51:24 +08:00
let outstandingInit = this.outstandingInits.get(name);
if (outstandingInit) {
// Load already in progress, let's wait for it...
return new Promise((resolve) => {
this.outstandingInits.set(name, () => {
outstandingInit!();
resolve();
});
});
}
2022-03-23 22:41:12 +08:00
this.worker.postMessage({
type: "load",
name: name,
code: code,
} as WorkerMessage);
return new Promise((resolve) => {
2022-03-28 14:51:24 +08:00
this.outstandingInits.set(name, () => {
this.loadedFunctions.add(name);
this.outstandingInits.delete(name);
resolve();
});
2022-03-23 22:41:12 +08:00
});
}
async onMessage(data: ControllerMessage) {
switch (data.type) {
case "inited":
let initCb = this.outstandingInits.get(data.name!);
initCb && initCb();
this.outstandingInits.delete(data.name!);
break;
case "syscall":
try {
2022-03-25 19:03:06 +08:00
let result = await this.plug.syscall(data.name!, data.args!);
2022-03-23 22:41:12 +08:00
this.worker.postMessage({
type: "syscall-response",
id: data.id,
result: result,
} as WorkerMessage);
} catch (e: any) {
this.worker.postMessage({
type: "syscall-response",
id: data.id,
error: e.message,
} as WorkerMessage);
}
break;
case "result":
let resultCbs = this.outstandingInvocations.get(data.id!);
this.outstandingInvocations.delete(data.id!);
if (data.error) {
resultCbs && resultCbs.reject(new Error(data.error));
} else {
resultCbs && resultCbs.resolve(data.result);
}
break;
default:
console.error("Unknown message type", data);
}
}
async invoke(name: string, args: any[]): Promise<any> {
this.reqId++;
this.worker.postMessage({
type: "invoke",
id: this.reqId,
name,
args,
});
return new Promise((resolve, reject) => {
this.outstandingInvocations.set(this.reqId, { resolve, reject });
});
}
stop() {
this.worker.terminate();
}
}