silverbullet/lib/plugos/system.ts

184 lines
5.1 KiB
TypeScript
Raw Normal View History

2024-07-30 23:33:33 +08:00
import type { Hook } from "./types.ts";
import { EventEmitter } from "./event.ts";
2024-01-14 20:38:39 +08:00
import type { SandboxFactory } from "./sandboxes/sandbox.ts";
import { Plug } from "./plug.ts";
2024-07-30 23:33:33 +08:00
import { InMemoryManifestCache, type ManifestCache } from "./manifest_cache.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
}
export type SystemEvents<HookT> = {
plugLoaded: (plug: Plug<HookT>) => void | Promise<void>;
plugUnloaded: (name: string) => void | Promise<void>;
2022-03-23 22:41:12 +08:00
};
// Passed to every syscall, allows to pass in additional context that the syscall may use
export type SyscallContext = {
// This is the plug that is invoking the syscall,
// which may be undefined where this cannot be determined (e.g. when running in a NoSandbox)
plug?: string;
2022-03-25 19:03:06 +08:00
};
type SyscallSignature = (
...args: any[]
) => Promise<any> | any;
type Syscall = {
requiredPermissions: string[];
callback: SyscallSignature;
};
export type SystemOptions = {
manifestCache?: ManifestCache<any>;
plugFlushTimeout?: number;
};
2022-03-23 22:41:12 +08:00
export class System<HookT> extends EventEmitter<SystemEvents<HookT>> {
protected plugs = new Map<string, Plug<HookT>>();
registeredSyscalls = new Map<string, Syscall>();
protected enabledHooks = new Set<Hook<HookT>>();
2022-03-23 22:41:12 +08:00
/**
* @param env either an environment or undefined for hybrid mode
*/
constructor(
readonly env: string | undefined,
readonly options: SystemOptions = {},
) {
2022-03-23 22:41:12 +08:00
super();
if (!options.manifestCache) {
options.manifestCache = new InMemoryManifestCache();
}
2022-03-23 22:41:12 +08:00
}
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
}
}
}
/**
* Invokes a function named using the "plug.functionName" pattern, for convenience
* @param name name of the function (e.g. plug.doSomething)
* @param args an array of arguments to pass to the function
*/
invokeFunction(name: string, args: any[]): Promise<any> {
2024-02-08 20:01:32 +08:00
// Some sanity type checks
if (typeof name !== "string") {
throw new Error(
`invokeFunction: function name should be a string, got ${typeof name}`,
);
}
if (!Array.isArray(args)) {
throw new Error(
`invokeFunction: args should be an array, got ${typeof args}`,
);
}
const [plugName, functionName] = name.split(".");
if (!functionName) {
// Sanity check
throw new Error(`Missing function name: ${name}`);
}
const plug = this.loadedPlugs.get(plugName);
if (!plug) {
throw new Error(`Plug ${plugName} not found invoking ${name}`);
}
return plug.invoke(functionName, args);
}
localSyscall(name: string, args: any[]): Promise<any> {
return this.syscall({}, name, args);
}
syscall(
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}`);
}
if (ctx.plug) {
// Only when running in a plug context do we check permissions
const plug = this.loadedPlugs.get(ctx.plug);
if (!plug) {
throw new Error(
`Plug ${ctx.plug} not found while attempting to invoke ${name}}`,
);
2022-03-25 19:03:06 +08:00
}
for (const permission of syscall.requiredPermissions) {
if (!plug.grantedPermissions.includes(permission)) {
throw Error(`Missing permission '${permission}' for syscall ${name}`);
}
2022-03-25 19:03:06 +08:00
}
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
}
async load(
name: string,
sandboxFactory: SandboxFactory<HookT>,
hash = -1,
2022-03-23 22:41:12 +08:00
): Promise<Plug<HookT>> {
const plug = new Plug(this, name, hash, sandboxFactory);
// Wait for worker to boot, and pass back its manifest
await plug.ready;
2023-08-21 01:54:31 +08:00
// and there it is!
const manifest = plug.manifest!;
// Validate the manifest
2022-03-23 22:41:12 +08:00
let errors: string[] = [];
for (const feature of this.enabledHooks) {
errors = [...errors, ...feature.validateManifest(plug.manifest!)];
2022-03-23 22:41:12 +08:00
}
if (errors.length > 0) {
throw new Error(`Invalid manifest: ${errors.join(", ")}`);
}
if (this.plugs.has(manifest.name)) {
this.unload(manifest.name);
}
console.log("Activated plug", manifest.name);
this.plugs.set(manifest.name, plug);
await this.emit("plugLoaded", plug);
2022-03-23 22:41:12 +08:00
return plug;
}
unload(name: string) {
2022-03-23 22:41:12 +08:00
const plug = this.plugs.get(name);
if (!plug) {
return;
2022-03-23 22:41:12 +08:00
}
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);
}
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
);
}
}