silverbullet/plugos/system.ts

163 lines
4.2 KiB
TypeScript
Raw Normal View History

import { Hook, Manifest, RuntimeEnvironment } from "./types.ts";
import { EventEmitter } from "./event.ts";
import { Sandbox, SandboxFactory } from "./sandbox.ts";
import { Plug } from "./plug.ts";
2022-03-23 22:41:12 +08:00
2022-03-24 17:48:56 +08:00
export interface SysCallMapping {
2022-03-25 19:03:06 +08:00
[key: string]: (ctx: SyscallContext, ...args: any) => Promise<any> | any;
2022-03-23 22:41:12 +08:00
}
2022-04-27 01:04:36 +08:00
export type SystemJSON<HookT> = Manifest<HookT>[];
2022-03-24 17:48:56 +08:00
2022-03-23 22:41:12 +08:00
export type SystemEvents<HookT> = {
plugLoaded: (plug: Plug<HookT>) => void | Promise<void>;
sandboxInitialized(sandbox: Sandbox, plug: Plug<HookT>): void | Promise<void>;
plugUnloaded: (name: string) => void | Promise<void>;
2022-03-23 22:41:12 +08:00
};
export type SyscallContext = {
plug: Plug<any>;
2022-03-25 19:03:06 +08:00
};
type SyscallSignature = (
ctx: SyscallContext,
...args: any[]
) => Promise<any> | any;
type Syscall = {
requiredPermissions: string[];
callback: SyscallSignature;
};
2022-03-23 22:41:12 +08:00
export class System<HookT> extends EventEmitter<SystemEvents<HookT>> {
readonly runtimeEnv: RuntimeEnvironment;
2022-03-23 22:41:12 +08:00
protected plugs = new Map<string, Plug<HookT>>();
2022-03-25 19:03:06 +08:00
protected registeredSyscalls = new Map<string, Syscall>();
protected enabledHooks = new Set<Hook<HookT>>();
2022-03-23 22:41:12 +08:00
constructor(env: RuntimeEnvironment) {
super();
this.runtimeEnv = env;
}
get loadedPlugs(): Map<string, Plug<HookT>> {
return this.plugs;
}
addHook(feature: Hook<HookT>) {
this.enabledHooks.add(feature);
2022-03-23 22:41:12 +08:00
feature.apply(this);
}
2022-03-25 19:03:06 +08:00
registerSyscalls(
requiredCapabilities: string[],
...registrationObjects: SysCallMapping[]
) {
2022-03-23 22:41:12 +08:00
for (const registrationObject of registrationObjects) {
for (const [name, callback] of Object.entries(registrationObject)) {
this.registeredSyscalls.set(name, {
2022-03-25 19:03:06 +08:00
requiredPermissions: requiredCapabilities,
callback,
});
2022-03-23 22:41:12 +08:00
}
}
}
syscallWithContext(
2022-03-25 19:03:06 +08:00
ctx: SyscallContext,
name: string,
args: any[],
2022-03-25 19:03:06 +08:00
): Promise<any> {
const syscall = this.registeredSyscalls.get(name);
if (!syscall) {
2022-03-23 22:41:12 +08:00
throw Error(`Unregistered syscall ${name}`);
}
2022-03-25 19:03:06 +08:00
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}`);
}
2022-03-23 22:41:12 +08:00
}
2022-03-25 19:03:06 +08:00
return Promise.resolve(syscall.callback(ctx, ...args));
2022-03-23 22:41:12 +08:00
}
localSyscall(
contextPlugName: string,
syscallName: string,
args: any[],
): Promise<any> {
return this.syscallWithContext(
// Mock the plug
{ plug: { name: contextPlugName } as any },
syscallName,
args,
);
}
2022-03-23 22:41:12 +08:00
async load(
manifest: Manifest<HookT>,
sandboxFactory: SandboxFactory<HookT>,
2022-03-23 22:41:12 +08:00
): Promise<Plug<HookT>> {
2022-04-27 01:04:36 +08:00
const name = manifest.name;
2022-03-23 22:41:12 +08:00
if (this.plugs.has(name)) {
await this.unload(name);
}
// Validate
let errors: string[] = [];
for (const feature of this.enabledHooks) {
2022-03-23 22:41:12 +08:00
errors = [...errors, ...feature.validateManifest(manifest)];
}
if (errors.length > 0) {
throw new Error(`Invalid manifest: ${errors.join(", ")}`);
}
// Ok, let's load this thing!
2022-03-25 19:03:06 +08:00
const plug = new Plug(this, name, sandboxFactory);
2022-04-27 01:04:36 +08:00
console.log("Loading", name);
2022-03-23 22:41:12 +08:00
await plug.load(manifest);
this.plugs.set(name, plug);
await this.emit("plugLoaded", plug);
2022-03-23 22:41:12 +08:00
return plug;
}
async unload(name: string) {
2022-04-27 02:31:31 +08:00
// console.log("Unloading", name);
2022-03-23 22:41:12 +08:00
const plug = this.plugs.get(name);
if (!plug) {
throw Error(`Plug ${name} not found`);
}
await plug.stop();
2022-04-27 01:04:36 +08:00
this.emit("plugUnloaded", name);
2022-03-23 22:41:12 +08:00
this.plugs.delete(name);
}
toJSON(): SystemJSON<HookT> {
const plugJSON: Manifest<HookT>[] = [];
for (const [_, plug] of this.plugs) {
2022-03-23 22:41:12 +08:00
if (!plug.manifest) {
continue;
}
2022-04-27 01:04:36 +08:00
plugJSON.push(plug.manifest);
2022-03-23 22:41:12 +08:00
}
return plugJSON;
}
async replaceAllFromJSON(
json: SystemJSON<HookT>,
sandboxFactory: SandboxFactory<HookT>,
2022-03-23 22:41:12 +08:00
) {
await this.unloadAll();
for (const manifest of json) {
2022-07-11 15:08:22 +08:00
// console.log("Loading plug", manifest.name);
2022-04-27 01:04:36 +08:00
await this.load(manifest, sandboxFactory);
2022-03-23 22:41:12 +08:00
}
}
unloadAll(): Promise<void[]> {
2022-03-23 22:41:12 +08:00
return Promise.all(
Array.from(this.plugs.keys()).map(this.unload.bind(this)),
2022-03-23 22:41:12 +08:00
);
}
}