import type { Hook, Manifest } from "../../lib/plugos/types.ts"; import { System } from "../../lib/plugos/system.ts"; import { ScriptEnvironment } from "$common/space_script.ts"; // System events: // - plug:load (plugName: string) export type EventHookT = { events?: string[]; }; export class EventHook implements Hook { private system?: System; private localListeners: Map any)[]> = new Map(); public scriptEnvironment?: ScriptEnvironment; addLocalListener(eventName: string, callback: (...args: any[]) => any) { if (!this.localListeners.has(eventName)) { this.localListeners.set(eventName, []); } this.localListeners.get(eventName)!.push(callback); } // Pull all events listened to listEvents(): string[] { if (!this.system) { throw new Error("Event hook is not initialized"); } const eventNames = new Set(); for (const plug of this.system.loadedPlugs.values()) { for (const functionDef of Object.values(plug.manifest!.functions)) { if (functionDef.events) { for (const eventName of functionDef.events) { eventNames.add(eventName); } } } } for (const eventName of this.localListeners.keys()) { eventNames.add(eventName); } return [...eventNames]; } async dispatchEvent(eventName: string, ...args: any[]): Promise { if (!this.system) { throw new Error("Event hook is not initialized"); } const responses: any[] = []; const promises: Promise[] = []; for (const plug of this.system.loadedPlugs.values()) { const manifest = plug.manifest; for ( const [name, functionDef] of Object.entries( manifest!.functions, ) ) { if (functionDef.events) { for (const event of functionDef.events) { if ( event === eventName || eventNameToRegex(event).test(eventName) ) { // Only dispatch functions that can run in this environment if (await plug.canInvoke(name)) { // Queue the promise promises.push((async () => { try { const result = await plug.invoke(name, args); if (result !== undefined) { responses.push(result); } } catch (e: any) { console.error( `Error dispatching event ${eventName} to ${plug.name}.${name}: ${e.message}`, ); } })()); } } } } } } // Local listeners const localListeners = this.localListeners.get(eventName); if (localListeners) { for (const localListener of localListeners) { // Queue the promise promises.push((async () => { const result = await Promise.resolve(localListener(...args)); if (result) { responses.push(result); } })()); } } // Space script listeners if (this.scriptEnvironment) { for ( const [name, listeners] of Object.entries( this.scriptEnvironment.eventHandlers, ) ) { if (eventNameToRegex(name).test(eventName)) { for (const listener of listeners) { promises.push((async () => { const result = await Promise.resolve( listener({ name: eventName, // Most events have a single argument, so let's optimize for that, otherwise pass all arguments as an array data: args.length === 1 ? args[0] : args, }), ); if (result) { responses.push(result); } })()); } } } } // Wait for all promises to resolve await Promise.all(promises); return responses; } apply(system: System): void { this.system = system; this.system.on({ plugLoaded: async (plug) => { await this.dispatchEvent("plug:load", plug.name); }, }); } validateManifest(manifest: Manifest): string[] { const errors = []; for ( const [_, functionDef] of Object.entries( manifest.functions || {}, ) ) { if (functionDef.events && !Array.isArray(functionDef.events)) { errors.push("'events' key must be an array of strings"); } } return errors; } } function eventNameToRegex(eventName: string): RegExp { return new RegExp( `^${eventName.replace(/\*/g, ".*").replace(/\//g, "\\/")}$`, ); }