2022-04-05 23:02:17 +08:00
|
|
|
import { Hook, Manifest } from "../types";
|
|
|
|
import { System } from "../system";
|
|
|
|
import { safeRun } from "../util";
|
2022-04-27 01:04:36 +08:00
|
|
|
import { EventEmitter } from "events";
|
2022-03-25 19:03:06 +08:00
|
|
|
|
2022-03-28 14:51:24 +08:00
|
|
|
// System events:
|
|
|
|
// - plug:load (plugName: string)
|
|
|
|
|
2022-03-29 17:21:32 +08:00
|
|
|
export type EventHookT = {
|
2022-03-27 17:26:13 +08:00
|
|
|
events?: string[];
|
2022-03-25 19:03:06 +08:00
|
|
|
};
|
|
|
|
|
2022-03-29 17:21:32 +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
|
|
|
|
2022-04-19 22:54:47 +08:00
|
|
|
async dispatchEvent(eventName: string, data?: any): Promise<any[]> {
|
2022-03-25 19:03:06 +08:00
|
|
|
if (!this.system) {
|
2022-03-29 17:21:32 +08:00
|
|
|
throw new Error("Event hook is not initialized");
|
2022-03-25 19:03:06 +08:00
|
|
|
}
|
2022-04-19 22:54:47 +08:00
|
|
|
let responses: any[] = [];
|
2022-03-25 19:03:06 +08:00
|
|
|
for (const plug of this.system.loadedPlugs.values()) {
|
2022-03-27 17:26:13 +08:00
|
|
|
for (const [name, functionDef] of Object.entries(
|
|
|
|
plug.manifest!.functions
|
|
|
|
)) {
|
|
|
|
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)) {
|
2022-04-19 22:54:47 +08:00
|
|
|
let 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
|
|
|
}
|
|
|
|
}
|
2022-04-27 01:04:36 +08:00
|
|
|
let localListeners = this.localListeners.get(eventName);
|
|
|
|
if (localListeners) {
|
|
|
|
for (let localListener of localListeners) {
|
|
|
|
let result = await Promise.resolve(localListener(data));
|
|
|
|
if (result) {
|
|
|
|
responses.push(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-19 22:54:47 +08:00
|
|
|
return responses;
|
2022-03-25 19:03:06 +08:00
|
|
|
}
|
|
|
|
|
2022-03-29 17:21:32 +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
|
|
|
}
|
|
|
|
|
2022-03-29 17:21:32 +08:00
|
|
|
validateManifest(manifest: Manifest<EventHookT>): string[] {
|
2022-03-27 17:26:13 +08:00
|
|
|
let errors = [];
|
|
|
|
for (const [name, 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;
|
2022-03-25 19:03:06 +08:00
|
|
|
}
|
|
|
|
}
|