silverbullet/lib/plugos/plug.ts

116 lines
3.3 KiB
TypeScript
Raw Permalink Normal View History

2024-07-30 23:33:33 +08:00
import type { Manifest } from "./types.ts";
import type { System } from "./system.ts";
import type { AssetBundle } from "../asset_bundle/bundle.ts";
import type { Sandbox, SandboxFactory } from "./sandboxes/sandbox.ts";
2022-03-23 22:41:12 +08:00
export class Plug<HookT> {
2023-08-05 00:56:55 +08:00
readonly runtimeEnv?: string;
public grantedPermissions: string[] = [];
public sandbox: Sandbox<HookT>;
// Resolves once the plug's manifest is available
ready: Promise<void>;
// Only available after ready resolves
2022-03-23 22:41:12 +08:00
public manifest?: Manifest<HookT>;
2022-10-13 21:16:18 +08:00
public assets?: AssetBundle;
2022-03-23 22:41:12 +08:00
// Time of last function invocation
unloadTimeout?: number;
2022-03-25 19:03:06 +08:00
constructor(
private system: System<HookT>,
readonly name: string,
private hash: number,
2024-01-14 20:38:39 +08:00
private sandboxFactory: SandboxFactory<HookT>,
2022-03-25 19:03:06 +08:00
) {
this.runtimeEnv = system.env;
2022-03-23 22:41:12 +08:00
this.scheduleUnloadTimeout();
this.sandbox = this.sandboxFactory(this);
// Retrieve the manifest asynchonously, which may either come from a cache or be loaded from the worker
this.ready = system.options.manifestCache!.getManifest(this, this.hash)
.then(
(manifest) => {
this.manifest = manifest;
// TODO: These need to be explicitly granted, not just taken
this.grantedPermissions = manifest.requiredPermissions || [];
},
);
2022-03-25 19:03:06 +08:00
}
// Invoke a syscall
2022-03-25 19:03:06 +08:00
syscall(name: string, args: any[]): Promise<any> {
return this.system.syscall({ plug: this.name }, name, args);
2022-03-23 22:41:12 +08:00
}
2023-12-17 22:10:00 +08:00
/**
* Checks if a function can be invoked (it may be restricted on its execution environment)
*/
async canInvoke(name: string) {
await this.ready;
const funDef = this.manifest!.functions[name];
2022-03-23 22:41:12 +08:00
if (!funDef) {
throw new Error(`Function ${name} not found in manifest`);
}
return !funDef.env || !this.runtimeEnv || funDef.env === this.runtimeEnv;
2022-03-23 22:41:12 +08:00
}
scheduleUnloadTimeout() {
if (!this.system.options.plugFlushTimeout) {
return;
}
// Reset the unload timeout, if set
if (this.unloadTimeout) {
clearTimeout(this.unloadTimeout);
}
this.unloadTimeout = setTimeout(() => {
this.stop();
}, this.system.options.plugFlushTimeout);
}
// Invoke a function
async invoke(name: string, args: any[]): Promise<any> {
// Ensure the worker is fully up and running
await this.ready;
this.scheduleUnloadTimeout();
// Before we access the manifest
const funDef = this.manifest!.functions[name];
if (!funDef) {
throw new Error(`Function ${name} not found in manifest`);
}
const sandbox = this.sandbox!;
if (funDef.redirect) {
// Function redirect, look up
// deno-lint-ignore no-this-alias
let plug: Plug<HookT> | undefined = this;
if (funDef.redirect.indexOf(".") !== -1) {
const [plugName, functionName] = funDef.redirect.split(".");
plug = this.system.loadedPlugs.get(plugName);
if (!plug) {
throw Error(`Plug ${plugName} redirected to not found`);
}
name = functionName;
} else {
name = funDef.redirect;
2022-03-23 22:41:12 +08:00
}
return plug.invoke(name, args);
}
if (!await this.canInvoke(name)) {
throw new Error(
`Function ${name} is not available in ${this.runtimeEnv}`,
);
2022-03-23 22:41:12 +08:00
}
return sandbox.invoke(name, args);
2022-03-23 22:41:12 +08:00
}
2022-10-16 01:02:56 +08:00
stop() {
console.log("Stopping sandbox for", this.name);
this.sandbox.stop();
2022-03-23 22:41:12 +08:00
}
}