silverbullet/plugos/hooks/event.ts

105 lines
2.9 KiB
TypeScript
Raw Normal View History

import type { Hook, Manifest } from "../types.ts";
import { System } from "../system.ts";
import { safeRun } from "../util.ts";
2022-03-25 19:03:06 +08:00
2022-03-28 14:51:24 +08:00
// System events:
// - plug:load (plugName: string)
export type EventHookT = {
2022-03-27 17:26:13 +08:00
events?: string[];
2022-03-25 19:03:06 +08:00
};
export class EventHook implements Hook<EventHookT> {
private system?: System<EventHookT>;
2022-04-27 01:04:36 +08:00
public localListeners: Map<string, ((data: any) => any)[]> = new Map();
addLocalListener(eventName: string, callback: (data: any) => any) {
if (!this.localListeners.has(eventName)) {
this.localListeners.set(eventName, []);
}
this.localListeners.get(eventName)!.push(callback);
}
2022-03-25 19:03:06 +08:00
// Pull all events listened to
listEvents(): string[] {
if (!this.system) {
throw new Error("Event hook is not initialized");
}
const eventNames = new Set<string>();
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, data?: any): Promise<any[]> {
2022-03-25 19:03:06 +08:00
if (!this.system) {
throw new Error("Event hook is not initialized");
2022-03-25 19:03:06 +08:00
}
const responses: any[] = [];
2022-03-25 19:03:06 +08:00
for (const plug of this.system.loadedPlugs.values()) {
for (
const [name, functionDef] of Object.entries(
plug.manifest!.functions,
)
) {
2022-03-27 17:26:13 +08:00
if (functionDef.events && functionDef.events.includes(eventName)) {
2022-03-28 14:51:24 +08:00
// Only dispatch functions that can run in this environment
if (plug.canInvoke(name)) {
const result = await plug.invoke(name, [data]);
if (result !== undefined) {
responses.push(result);
}
2022-03-28 14:51:24 +08:00
}
2022-03-27 17:26:13 +08:00
}
2022-03-25 19:03:06 +08:00
}
}
const localListeners = this.localListeners.get(eventName);
2022-04-27 01:04:36 +08:00
if (localListeners) {
for (const localListener of localListeners) {
const result = await Promise.resolve(localListener(data));
2022-04-27 01:04:36 +08:00
if (result) {
responses.push(result);
}
}
}
return responses;
2022-03-25 19:03:06 +08:00
}
apply(system: System<EventHookT>): void {
2022-03-25 19:03:06 +08:00
this.system = system;
2022-03-28 14:51:24 +08:00
this.system.on({
2022-04-27 01:04:36 +08:00
plugLoaded: (plug) => {
2022-03-29 18:13:46 +08:00
safeRun(async () => {
2022-04-27 01:04:36 +08:00
await this.dispatchEvent("plug:load", plug.name);
2022-03-29 18:13:46 +08:00
});
2022-03-28 14:51:24 +08:00
},
});
2022-03-25 19:03:06 +08:00
}
validateManifest(manifest: Manifest<EventHookT>): string[] {
const errors = [];
for (
const [_, functionDef] of Object.entries(
manifest.functions || {},
)
) {
2022-03-27 17:26:13 +08:00
if (functionDef.events && !Array.isArray(functionDef.events)) {
errors.push("'events' key must be an array of strings");
}
}
return errors;
2022-03-25 19:03:06 +08:00
}
}