8527528af4
* Manifest caching and lazy loading of plug workers * Fixes #546 Plug unloading after time out
168 lines
4.4 KiB
TypeScript
168 lines
4.4 KiB
TypeScript
import { Hook, Manifest } from "./types.ts";
|
|
import { EventEmitter } from "./event.ts";
|
|
import type { SandboxFactory } from "./sandbox.ts";
|
|
import { Plug } from "./plug.ts";
|
|
import { deepObjectMerge } from "$sb/lib/json.ts";
|
|
import { InMemoryManifestCache, ManifestCache } from "./manifest_cache.ts";
|
|
|
|
export interface SysCallMapping {
|
|
[key: string]: (ctx: SyscallContext, ...args: any) => Promise<any> | any;
|
|
}
|
|
|
|
export type SystemEvents<HookT> = {
|
|
plugLoaded: (plug: Plug<HookT>) => void | Promise<void>;
|
|
plugUnloaded: (name: string) => void | Promise<void>;
|
|
};
|
|
|
|
// Passed to every syscall, allows to pass in additional context that the syscall may use
|
|
export type SyscallContext = {
|
|
plug: Plug<any>;
|
|
};
|
|
|
|
type SyscallSignature = (
|
|
ctx: SyscallContext,
|
|
...args: any[]
|
|
) => Promise<any> | any;
|
|
|
|
type Syscall = {
|
|
requiredPermissions: string[];
|
|
callback: SyscallSignature;
|
|
};
|
|
|
|
export type SystemOptions = {
|
|
manifestCache?: ManifestCache<any>;
|
|
plugFlushTimeout?: number;
|
|
};
|
|
|
|
export class System<HookT> extends EventEmitter<SystemEvents<HookT>> {
|
|
protected plugs = new Map<string, Plug<HookT>>();
|
|
protected registeredSyscalls = new Map<string, Syscall>();
|
|
protected enabledHooks = new Set<Hook<HookT>>();
|
|
|
|
/**
|
|
* @param env either an environment or undefined for hybrid mode
|
|
*/
|
|
constructor(
|
|
readonly env: string | undefined,
|
|
readonly options: SystemOptions = {},
|
|
) {
|
|
super();
|
|
if (!options.manifestCache) {
|
|
options.manifestCache = new InMemoryManifestCache();
|
|
}
|
|
}
|
|
|
|
get loadedPlugs(): Map<string, Plug<HookT>> {
|
|
return this.plugs;
|
|
}
|
|
|
|
addHook(feature: Hook<HookT>) {
|
|
this.enabledHooks.add(feature);
|
|
feature.apply(this);
|
|
}
|
|
|
|
registerSyscalls(
|
|
requiredCapabilities: string[],
|
|
...registrationObjects: SysCallMapping[]
|
|
) {
|
|
for (const registrationObject of registrationObjects) {
|
|
for (const [name, callback] of Object.entries(registrationObject)) {
|
|
this.registeredSyscalls.set(name, {
|
|
requiredPermissions: requiredCapabilities,
|
|
callback,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
syscallWithContext(
|
|
ctx: SyscallContext,
|
|
name: string,
|
|
args: any[],
|
|
): Promise<any> {
|
|
const syscall = this.registeredSyscalls.get(name);
|
|
if (!syscall) {
|
|
throw Error(`Unregistered syscall ${name}`);
|
|
}
|
|
for (const permission of syscall.requiredPermissions) {
|
|
if (!ctx.plug) {
|
|
throw Error(`Syscall ${name} requires permission and no plug is set`);
|
|
}
|
|
if (!ctx.plug.grantedPermissions.includes(permission)) {
|
|
throw Error(`Missing permission '${permission}' for syscall ${name}`);
|
|
}
|
|
}
|
|
return Promise.resolve(syscall.callback(ctx, ...args));
|
|
}
|
|
|
|
localSyscall(
|
|
contextPlugName: string,
|
|
syscallName: string,
|
|
args: any[],
|
|
): Promise<any> {
|
|
return this.syscallWithContext(
|
|
{ plug: this.plugs.get(contextPlugName)! },
|
|
syscallName,
|
|
args,
|
|
);
|
|
}
|
|
|
|
async load(
|
|
workerUrl: URL,
|
|
name: string,
|
|
hash: number,
|
|
sandboxFactory: SandboxFactory<HookT>,
|
|
// Mapping plug name -> manifest overrides
|
|
manifestOverrides?: Record<string, Partial<Manifest<HookT>>>,
|
|
): Promise<Plug<HookT>> {
|
|
const plug = new Plug(this, workerUrl, name, hash, sandboxFactory);
|
|
|
|
// Wait for worker to boot, and pass back its manifest
|
|
await plug.ready;
|
|
|
|
if (manifestOverrides && manifestOverrides[plug.manifest!.name]) {
|
|
plug.manifest = deepObjectMerge(
|
|
plug.manifest,
|
|
manifestOverrides[plug.manifest!.name],
|
|
);
|
|
// console.log("New manifest", plug.manifest);
|
|
}
|
|
// and there it is!
|
|
const manifest = plug.manifest!;
|
|
|
|
// Validate the manifest
|
|
let errors: string[] = [];
|
|
for (const feature of this.enabledHooks) {
|
|
errors = [...errors, ...feature.validateManifest(plug.manifest!)];
|
|
}
|
|
if (errors.length > 0) {
|
|
throw new Error(`Invalid manifest: ${errors.join(", ")}`);
|
|
}
|
|
if (this.plugs.has(manifest.name)) {
|
|
this.unload(manifest.name);
|
|
}
|
|
console.log("Loaded plug", manifest.name);
|
|
this.plugs.set(manifest.name, plug);
|
|
|
|
await this.emit("plugLoaded", plug);
|
|
return plug;
|
|
}
|
|
|
|
unload(name: string) {
|
|
// console.log("Unloading", name);
|
|
const plug = this.plugs.get(name);
|
|
if (!plug) {
|
|
return;
|
|
}
|
|
plug.stop();
|
|
this.emit("plugUnloaded", name);
|
|
this.plugs.delete(name);
|
|
}
|
|
|
|
unloadAll(): Promise<void[]> {
|
|
return Promise.all(
|
|
Array.from(this.plugs.keys()).map(this.unload.bind(this)),
|
|
);
|
|
}
|
|
}
|