104 lines
2.9 KiB
TypeScript
104 lines
2.9 KiB
TypeScript
import type { Hook, Manifest } from "../types.ts";
|
|
import type { System } from "../system.ts";
|
|
import type { MQMessage } from "../../../plug-api/types.ts";
|
|
import type { MessageQueue } from "../../data/mq.ts";
|
|
import { throttle } from "../../async.ts";
|
|
import type { MQHookT } from "$lib/manifest.ts";
|
|
|
|
export class MQHook implements Hook<MQHookT> {
|
|
subscriptions: (() => void)[] = [];
|
|
|
|
constructor(private system: System<MQHookT>, readonly mq: MessageQueue) {
|
|
}
|
|
|
|
apply(system: System<MQHookT>): void {
|
|
this.system = system;
|
|
system.on({
|
|
plugLoaded: () => {
|
|
this.throttledReloadQueues();
|
|
},
|
|
plugUnloaded: () => {
|
|
this.throttledReloadQueues();
|
|
},
|
|
});
|
|
|
|
this.throttledReloadQueues();
|
|
}
|
|
|
|
stop() {
|
|
// console.log("Unsubscribing from all queues");
|
|
this.subscriptions.forEach((sub) => sub());
|
|
this.subscriptions = [];
|
|
}
|
|
|
|
throttledReloadQueues = throttle(() => {
|
|
this.reloadQueues();
|
|
}, 1000);
|
|
|
|
reloadQueues() {
|
|
this.stop();
|
|
for (const plug of this.system.loadedPlugs.values()) {
|
|
if (!plug.manifest) {
|
|
continue;
|
|
}
|
|
for (
|
|
const [name, functionDef] of Object.entries(
|
|
plug.manifest.functions,
|
|
)
|
|
) {
|
|
if (!functionDef.mqSubscriptions) {
|
|
continue;
|
|
}
|
|
const subscriptions = functionDef.mqSubscriptions;
|
|
for (const subscriptionDef of subscriptions) {
|
|
const queue = subscriptionDef.queue;
|
|
// console.log("Subscribing to queue", queue, subscriptionDef);
|
|
this.subscriptions.push(
|
|
this.mq.subscribe(
|
|
queue,
|
|
{
|
|
batchSize: subscriptionDef.batchSize,
|
|
pollInterval: subscriptionDef.pollInterval,
|
|
},
|
|
async (messages: MQMessage[]) => {
|
|
try {
|
|
await plug.invoke(name, [messages]);
|
|
if (subscriptionDef.autoAck) {
|
|
await this.mq.batchAck(queue, messages.map((m) => m.id));
|
|
}
|
|
} catch (e: any) {
|
|
console.error(
|
|
"Execution of mqSubscription for queue",
|
|
queue,
|
|
"invoking",
|
|
name,
|
|
"with messages",
|
|
messages,
|
|
"failed:",
|
|
e,
|
|
);
|
|
}
|
|
},
|
|
),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
validateManifest(manifest: Manifest<MQHookT>): string[] {
|
|
const errors: string[] = [];
|
|
for (const functionDef of Object.values(manifest.functions)) {
|
|
if (!functionDef.mqSubscriptions) {
|
|
continue;
|
|
}
|
|
for (const subscriptionDef of functionDef.mqSubscriptions) {
|
|
if (!subscriptionDef.queue) {
|
|
errors.push("Missing queue name for mqSubscription");
|
|
}
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
}
|