2022-03-29 17:21:32 +08:00
|
|
|
import { Hook, Manifest } from "../types";
|
2022-03-23 22:41:12 +08:00
|
|
|
import cron, { ScheduledTask } from "node-cron";
|
|
|
|
import { safeRun } from "../util";
|
|
|
|
import { System } from "../system";
|
|
|
|
|
2022-03-29 17:21:32 +08:00
|
|
|
export type CronHookT = {
|
2022-03-27 17:26:13 +08:00
|
|
|
cron?: string | string[];
|
2022-03-23 22:41:12 +08:00
|
|
|
};
|
|
|
|
|
2022-03-29 17:21:32 +08:00
|
|
|
export class NodeCronHook implements Hook<CronHookT> {
|
|
|
|
apply(system: System<CronHookT>): void {
|
2022-03-23 22:41:12 +08:00
|
|
|
let tasks: ScheduledTask[] = [];
|
|
|
|
system.on({
|
2022-04-27 01:04:36 +08:00
|
|
|
plugLoaded: () => {
|
2022-03-23 22:41:12 +08:00
|
|
|
reloadCrons();
|
|
|
|
},
|
2022-04-27 01:04:36 +08:00
|
|
|
plugUnloaded() {
|
2022-03-23 22:41:12 +08:00
|
|
|
reloadCrons();
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
reloadCrons();
|
|
|
|
|
|
|
|
function reloadCrons() {
|
|
|
|
tasks.forEach((task) => task.stop());
|
|
|
|
tasks = [];
|
|
|
|
for (let plug of system.loadedPlugs.values()) {
|
2022-03-27 17:26:13 +08:00
|
|
|
if (!plug.manifest) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
for (const [name, functionDef] of Object.entries(
|
|
|
|
plug.manifest.functions
|
|
|
|
)) {
|
|
|
|
if (!functionDef.cron) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const crons = Array.isArray(functionDef.cron)
|
|
|
|
? functionDef.cron
|
|
|
|
: [functionDef.cron];
|
2022-03-23 22:41:12 +08:00
|
|
|
for (let cronDef of crons) {
|
|
|
|
tasks.push(
|
2022-03-27 17:26:13 +08:00
|
|
|
cron.schedule(cronDef, () => {
|
|
|
|
console.log("Now acting on cron", cronDef);
|
2022-03-23 22:41:12 +08:00
|
|
|
safeRun(async () => {
|
|
|
|
try {
|
2022-03-27 17:26:13 +08:00
|
|
|
await plug.invoke(name, [cronDef]);
|
2022-03-23 22:41:12 +08:00
|
|
|
} catch (e: any) {
|
|
|
|
console.error("Execution of cron function failed", e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-29 17:21:32 +08:00
|
|
|
validateManifest(manifest: Manifest<CronHookT>): string[] {
|
2022-03-23 22:41:12 +08:00
|
|
|
let errors = [];
|
2022-03-27 17:26:13 +08:00
|
|
|
for (const [name, functionDef] of Object.entries(manifest.functions)) {
|
|
|
|
if (!functionDef.cron) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const crons = Array.isArray(functionDef.cron)
|
|
|
|
? functionDef.cron
|
|
|
|
: [functionDef.cron];
|
2022-03-23 22:41:12 +08:00
|
|
|
for (let cronDef of crons) {
|
2022-03-27 17:26:13 +08:00
|
|
|
if (!cron.validate(cronDef)) {
|
|
|
|
errors.push(`Invalid cron expression ${cronDef}`);
|
2022-03-23 22:41:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return errors;
|
|
|
|
}
|
|
|
|
}
|