import { Manifest, RuntimeEnvironment } from "./types.ts"; import { Sandbox } from "./sandbox.ts"; import { System } from "./system.ts"; import { AssetBundle, AssetJson } from "./asset_bundle/bundle.ts"; export class Plug { system: System; sandbox?: Sandbox; public manifest?: Manifest; public assets?: AssetBundle; private sandboxFactory: (plug: Plug) => Sandbox; readonly runtimeEnv: RuntimeEnvironment; grantedPermissions: string[] = []; name: string; version: number; constructor( system: System, name: string, sandboxFactory: (plug: Plug) => Sandbox, ) { this.system = system; this.name = name; this.sandboxFactory = sandboxFactory; // this.sandbox = sandboxFactory(this); this.runtimeEnv = system.runtimeEnv; this.version = new Date().getTime(); } // Lazy load sandbox, guarantees that the sandbox is loaded async ensureSandbox() { if (!this.sandbox) { console.log("Now starting sandbox for", this.name); // Kick off worker this.sandbox = this.sandboxFactory(this); // Push in any dependencies for ( const [dep, code] of Object.entries(this.manifest!.dependencies || {}) ) { await this.sandbox.loadDependency(dep, code); } await this.system.emit("sandboxInitialized", this.sandbox, this); } } load(manifest: Manifest) { this.manifest = manifest; this.assets = new AssetBundle( manifest.assets ? manifest.assets as AssetJson : {}, ); // TODO: These need to be explicitly granted, not just taken this.grantedPermissions = manifest.requiredPermissions || []; } syscall(name: string, args: any[]): Promise { return this.system.syscallWithContext({ plug: this }, name, args); } canInvoke(name: string) { if (!this.manifest) { return false; } const funDef = this.manifest.functions[name]; if (!funDef) { throw new Error(`Function ${name} not found in manifest`); } return !funDef.env || funDef.env === this.runtimeEnv; } async invoke(name: string, args: any[]): Promise { const funDef = this.manifest!.functions[name]; if (!funDef) { throw new Error(`Function ${name} not found in manifest`); } await this.ensureSandbox(); const sandbox = this.sandbox!; if (funDef.redirect) { // Function redirect, look up // deno-lint-ignore no-this-alias let plug: Plug | 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; } return plug.invoke(name, args); } if (!sandbox.isLoaded(name)) { if (!this.canInvoke(name)) { throw new Error( `Function ${name} is not available in ${this.runtimeEnv}`, ); } await sandbox.load(name, funDef.code!); } return await sandbox.invoke(name, args); } stop() { if (this.sandbox) { this.sandbox.stop(); } } }