@blokjs/trigger-cron 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md ADDED
@@ -0,0 +1,22 @@
1
+ # @blokjs/trigger-cron
2
+
3
+ ## 0.2.0
4
+
5
+ ### Minor Changes
6
+
7
+ - Initial public release of Blok packages.
8
+
9
+ This release includes:
10
+
11
+ - Core packages: @blokjs/shared, @blokjs/helper, @blokjs/runner
12
+ - Node packages: @blokjs/api-call, @blokjs/if-else, @blokjs/react
13
+ - Trigger packages: pubsub, queue, webhook, websocket, worker, cron, grpc
14
+ - CLI tool: blokctl
15
+ - Editor support: @blokjs/lsp-server, @blokjs/syntax
16
+
17
+ ### Patch Changes
18
+
19
+ - Updated dependencies
20
+ - @blokjs/shared@0.2.0
21
+ - @blokjs/helper@0.2.0
22
+ - @blokjs/runner@0.2.0
@@ -0,0 +1,124 @@
1
+ /**
2
+ * CronTrigger - Scheduled workflow execution based on cron expressions
3
+ *
4
+ * Extends TriggerBase to support scheduled triggers:
5
+ * - Cron expressions (e.g., "0 * * * *" for hourly)
6
+ * - Timezone-aware scheduling
7
+ * - Overlap prevention
8
+ * - Manual trigger support
9
+ *
10
+ * Uses the 'cron' package for cron parsing and scheduling.
11
+ */
12
+ import type { CronTriggerOpts, HelperResponse } from "@blok/helper";
13
+ import { DefaultLogger, type GlobalOptions, type BlokService, TriggerBase, type TriggerResponse } from "@blok/runner";
14
+ import { CronJob } from "cron";
15
+ /**
16
+ * Scheduled job information
17
+ */
18
+ export interface ScheduledJob {
19
+ /** Unique job ID */
20
+ id: string;
21
+ /** Workflow path */
22
+ workflowPath: string;
23
+ /** Cron expression */
24
+ schedule: string;
25
+ /** Timezone */
26
+ timezone: string;
27
+ /** Allow overlapping executions */
28
+ overlap: boolean;
29
+ /** Whether the job is currently running */
30
+ running: boolean;
31
+ /** Last execution time */
32
+ lastRun?: Date;
33
+ /** Next scheduled time */
34
+ nextRun?: Date;
35
+ /** Internal CronJob instance */
36
+ job: CronJob;
37
+ }
38
+ /**
39
+ * Execution context passed to the workflow
40
+ */
41
+ export interface CronExecutionContext {
42
+ /** Job ID */
43
+ jobId: string;
44
+ /** Scheduled time (when it was supposed to run) */
45
+ scheduledTime: Date;
46
+ /** Actual execution time */
47
+ executionTime: Date;
48
+ /** Cron expression */
49
+ schedule: string;
50
+ /** Timezone */
51
+ timezone: string;
52
+ /** Whether this is a manual trigger */
53
+ manual: boolean;
54
+ }
55
+ /**
56
+ * Workflow model with cron trigger configuration
57
+ */
58
+ interface CronWorkflowModel {
59
+ path: string;
60
+ config: {
61
+ name: string;
62
+ version: string;
63
+ trigger?: {
64
+ cron?: CronTriggerOpts;
65
+ [key: string]: unknown;
66
+ };
67
+ [key: string]: unknown;
68
+ };
69
+ }
70
+ /**
71
+ * CronTrigger - Scheduled workflow execution
72
+ */
73
+ export declare abstract class CronTrigger extends TriggerBase {
74
+ protected nodeMap: GlobalOptions;
75
+ protected readonly tracer: import("@opentelemetry/api").Tracer;
76
+ protected readonly logger: DefaultLogger;
77
+ protected jobs: Map<string, ScheduledJob>;
78
+ protected abstract nodes: Record<string, BlokService<unknown>>;
79
+ protected abstract workflows: Record<string, HelperResponse>;
80
+ constructor();
81
+ /**
82
+ * Load nodes into the node map
83
+ */
84
+ loadNodes(): void;
85
+ /**
86
+ * Load workflows into the workflow map
87
+ */
88
+ loadWorkflows(): void;
89
+ /**
90
+ * Convert cron DateTime to Date
91
+ * The cron package uses luxon DateTime which has toJSDate()
92
+ */
93
+ protected toDate(dateTime: unknown): Date;
94
+ /**
95
+ * Start the cron scheduler - main entry point
96
+ */
97
+ listen(): Promise<number>;
98
+ /**
99
+ * Stop all cron jobs
100
+ */
101
+ stop(): Promise<void>;
102
+ protected onHmrWorkflowChange(): Promise<void>;
103
+ /**
104
+ * Manually trigger a specific job
105
+ */
106
+ triggerJob(jobId: string): Promise<TriggerResponse | null>;
107
+ /**
108
+ * Get all scheduled jobs
109
+ */
110
+ getJobs(): ScheduledJob[];
111
+ /**
112
+ * Get all workflows that have cron triggers
113
+ */
114
+ protected getCronWorkflows(): CronWorkflowModel[];
115
+ /**
116
+ * Get workflow by path
117
+ */
118
+ protected getWorkflowByPath(path: string): CronWorkflowModel | null;
119
+ /**
120
+ * Execute a workflow
121
+ */
122
+ protected executeWorkflow(jobId: string, workflow: CronWorkflowModel, config: CronTriggerOpts, manual: boolean): Promise<TriggerResponse>;
123
+ }
124
+ export default CronTrigger;
@@ -0,0 +1,305 @@
1
+ "use strict";
2
+ /**
3
+ * CronTrigger - Scheduled workflow execution based on cron expressions
4
+ *
5
+ * Extends TriggerBase to support scheduled triggers:
6
+ * - Cron expressions (e.g., "0 * * * *" for hourly)
7
+ * - Timezone-aware scheduling
8
+ * - Overlap prevention
9
+ * - Manual trigger support
10
+ *
11
+ * Uses the 'cron' package for cron parsing and scheduling.
12
+ */
13
+ Object.defineProperty(exports, "__esModule", { value: true });
14
+ exports.CronTrigger = void 0;
15
+ const runner_1 = require("@blok/runner");
16
+ const api_1 = require("@opentelemetry/api");
17
+ const cron_1 = require("cron");
18
+ const uuid_1 = require("uuid");
19
+ /**
20
+ * CronTrigger - Scheduled workflow execution
21
+ */
22
+ class CronTrigger extends runner_1.TriggerBase {
23
+ nodeMap = {};
24
+ tracer = api_1.trace.getTracer(process.env.PROJECT_NAME || "trigger-cron-workflow", process.env.PROJECT_VERSION || "0.0.1");
25
+ logger = new runner_1.DefaultLogger();
26
+ jobs = new Map();
27
+ constructor() {
28
+ super();
29
+ this.loadNodes();
30
+ this.loadWorkflows();
31
+ }
32
+ /**
33
+ * Load nodes into the node map
34
+ */
35
+ loadNodes() {
36
+ this.nodeMap.nodes = new runner_1.NodeMap();
37
+ const nodeKeys = Object.keys(this.nodes);
38
+ for (const key of nodeKeys) {
39
+ this.nodeMap.nodes.addNode(key, this.nodes[key]);
40
+ }
41
+ }
42
+ /**
43
+ * Load workflows into the workflow map
44
+ */
45
+ loadWorkflows() {
46
+ this.nodeMap.workflows = this.workflows;
47
+ }
48
+ /**
49
+ * Convert cron DateTime to Date
50
+ * The cron package uses luxon DateTime which has toJSDate()
51
+ */
52
+ toDate(dateTime) {
53
+ if (dateTime && typeof dateTime === "object" && "toJSDate" in dateTime) {
54
+ return dateTime.toJSDate();
55
+ }
56
+ return dateTime instanceof Date ? dateTime : new Date(dateTime);
57
+ }
58
+ /**
59
+ * Start the cron scheduler - main entry point
60
+ */
61
+ async listen() {
62
+ const startTime = this.startCounter();
63
+ try {
64
+ // Find all workflows with cron triggers
65
+ const cronWorkflows = this.getCronWorkflows();
66
+ if (cronWorkflows.length === 0) {
67
+ this.logger.log("No workflows with cron triggers found");
68
+ return this.endCounter(startTime);
69
+ }
70
+ // Create and start cron jobs for each workflow
71
+ for (const workflow of cronWorkflows) {
72
+ const config = workflow.config.trigger?.cron;
73
+ const jobId = `cron-${workflow.path}-${(0, uuid_1.v4)().slice(0, 8)}`;
74
+ this.logger.log(`Scheduling workflow: ${workflow.path} with schedule: ${config.schedule} (${config.timezone})`);
75
+ const job = new cron_1.CronJob(config.schedule, async () => {
76
+ await this.executeWorkflow(jobId, workflow, config, false);
77
+ }, null, // onComplete
78
+ false, // start
79
+ config.timezone);
80
+ const scheduledJob = {
81
+ id: jobId,
82
+ workflowPath: workflow.path,
83
+ schedule: config.schedule,
84
+ timezone: config.timezone,
85
+ overlap: config.overlap ?? false,
86
+ running: false,
87
+ nextRun: this.toDate(job.nextDate()),
88
+ job,
89
+ };
90
+ this.jobs.set(jobId, scheduledJob);
91
+ // Start the job
92
+ job.start();
93
+ this.logger.log(`Job ${jobId} started. Next run: ${scheduledJob.nextRun}`);
94
+ }
95
+ this.logger.log(`Cron trigger started. ${this.jobs.size} job(s) scheduled`);
96
+ // Enable HMR in development mode
97
+ if (process.env.BLOK_HMR === "true" || process.env.NODE_ENV === "development") {
98
+ await this.enableHotReload();
99
+ }
100
+ return this.endCounter(startTime);
101
+ }
102
+ catch (error) {
103
+ this.logger.error(`Failed to start cron trigger: ${error.message}`);
104
+ throw error;
105
+ }
106
+ }
107
+ /**
108
+ * Stop all cron jobs
109
+ */
110
+ async stop() {
111
+ for (const [jobId, scheduledJob] of this.jobs) {
112
+ scheduledJob.job.stop();
113
+ this.logger.log(`Job ${jobId} stopped`);
114
+ }
115
+ this.jobs.clear();
116
+ this.logger.log("Cron trigger stopped");
117
+ }
118
+ async onHmrWorkflowChange() {
119
+ this.logger.log("[HMR] Cron workflow changed, reloading...");
120
+ await this.waitForInFlightRequests();
121
+ await this.stop();
122
+ this.loadWorkflows();
123
+ await this.listen();
124
+ }
125
+ /**
126
+ * Manually trigger a specific job
127
+ */
128
+ async triggerJob(jobId) {
129
+ const scheduledJob = this.jobs.get(jobId);
130
+ if (!scheduledJob) {
131
+ this.logger.error(`Job not found: ${jobId}`);
132
+ return null;
133
+ }
134
+ // Get the workflow
135
+ const workflow = this.getWorkflowByPath(scheduledJob.workflowPath);
136
+ if (!workflow) {
137
+ this.logger.error(`Workflow not found: ${scheduledJob.workflowPath}`);
138
+ return null;
139
+ }
140
+ const config = workflow.config.trigger?.cron;
141
+ return this.executeWorkflow(jobId, workflow, config, true);
142
+ }
143
+ /**
144
+ * Get all scheduled jobs
145
+ */
146
+ getJobs() {
147
+ return Array.from(this.jobs.values()).map((job) => ({
148
+ ...job,
149
+ nextRun: this.toDate(job.job.nextDate()),
150
+ }));
151
+ }
152
+ /**
153
+ * Get all workflows that have cron triggers
154
+ */
155
+ getCronWorkflows() {
156
+ const workflows = [];
157
+ for (const [path, workflow] of Object.entries(this.nodeMap.workflows || {})) {
158
+ const workflowConfig = workflow._config;
159
+ if (workflowConfig?.trigger) {
160
+ const triggerType = Object.keys(workflowConfig.trigger)[0];
161
+ if (triggerType === "cron" && workflowConfig.trigger.cron) {
162
+ workflows.push({
163
+ path,
164
+ config: workflowConfig,
165
+ });
166
+ }
167
+ }
168
+ }
169
+ return workflows;
170
+ }
171
+ /**
172
+ * Get workflow by path
173
+ */
174
+ getWorkflowByPath(path) {
175
+ const workflow = this.nodeMap.workflows?.[path];
176
+ if (!workflow)
177
+ return null;
178
+ const workflowConfig = workflow._config;
179
+ return {
180
+ path,
181
+ config: workflowConfig,
182
+ };
183
+ }
184
+ /**
185
+ * Execute a workflow
186
+ */
187
+ async executeWorkflow(jobId, workflow, config, manual) {
188
+ const scheduledJob = this.jobs.get(jobId);
189
+ if (!scheduledJob) {
190
+ throw new Error(`Job not found: ${jobId}`);
191
+ }
192
+ // Check for overlap
193
+ if (scheduledJob.running && !scheduledJob.overlap) {
194
+ this.logger.log(`Skipping ${jobId}: previous execution still running (overlap disabled)`);
195
+ return { ctx: {}, metrics: {} };
196
+ }
197
+ const executionId = (0, uuid_1.v4)();
198
+ const lastDate = scheduledJob.job.lastDate();
199
+ const scheduledTime = lastDate ? new Date(lastDate) : new Date();
200
+ const executionTime = new Date();
201
+ const defaultMeter = api_1.metrics.getMeter("default");
202
+ const cronExecutions = defaultMeter.createCounter("cron_executions", {
203
+ description: "Cron job executions",
204
+ });
205
+ const cronErrors = defaultMeter.createCounter("cron_errors", {
206
+ description: "Cron job execution errors",
207
+ });
208
+ return new Promise((resolve) => {
209
+ this.tracer.startActiveSpan(`cron:${workflow.path}`, async (span) => {
210
+ scheduledJob.running = true;
211
+ try {
212
+ const start = performance.now();
213
+ // Initialize configuration for this workflow
214
+ await this.configuration.init(workflow.path, this.nodeMap);
215
+ // Create context
216
+ const ctx = this.createContext(undefined, workflow.path, executionId);
217
+ // Create execution context
218
+ const cronContext = {
219
+ jobId,
220
+ scheduledTime,
221
+ executionTime,
222
+ schedule: config.schedule,
223
+ timezone: config.timezone,
224
+ manual,
225
+ };
226
+ // Populate request with cron context
227
+ ctx.request = {
228
+ body: cronContext,
229
+ headers: {
230
+ "x-cron-job-id": jobId,
231
+ "x-cron-schedule": config.schedule,
232
+ "x-cron-timezone": config.timezone,
233
+ "x-cron-manual": String(manual),
234
+ },
235
+ query: {},
236
+ params: {
237
+ jobId,
238
+ schedule: config.schedule,
239
+ },
240
+ };
241
+ // Store cron context in vars
242
+ if (!ctx.vars)
243
+ ctx.vars = {};
244
+ ctx.vars._cron_context = {
245
+ jobId,
246
+ scheduledTime: scheduledTime.toISOString(),
247
+ executionTime: executionTime.toISOString(),
248
+ schedule: config.schedule,
249
+ timezone: config.timezone,
250
+ manual: String(manual),
251
+ };
252
+ ctx.logger.log(`Executing cron job: ${jobId} (${manual ? "manual" : "scheduled"})`);
253
+ // Execute workflow
254
+ const response = await this.run(ctx);
255
+ const end = performance.now();
256
+ // Update job state
257
+ scheduledJob.lastRun = executionTime;
258
+ scheduledJob.nextRun = this.toDate(scheduledJob.job.nextDate());
259
+ // Set span attributes
260
+ span.setAttribute("success", true);
261
+ span.setAttribute("job_id", jobId);
262
+ span.setAttribute("workflow_path", workflow.path);
263
+ span.setAttribute("schedule", config.schedule);
264
+ span.setAttribute("timezone", config.timezone);
265
+ span.setAttribute("manual", manual);
266
+ span.setAttribute("elapsed_ms", end - start);
267
+ span.setStatus({ code: api_1.SpanStatusCode.OK });
268
+ // Record metrics
269
+ cronExecutions.add(1, {
270
+ env: process.env.NODE_ENV,
271
+ job_id: jobId,
272
+ workflow_name: this.configuration.name,
273
+ manual: String(manual),
274
+ success: "true",
275
+ });
276
+ ctx.logger.log(`Cron job completed in ${(end - start).toFixed(2)}ms: ${jobId}`);
277
+ resolve(response);
278
+ }
279
+ catch (error) {
280
+ const errorMessage = error.message;
281
+ // Set span error
282
+ span.setAttribute("success", false);
283
+ span.recordException(error);
284
+ span.setStatus({ code: api_1.SpanStatusCode.ERROR, message: errorMessage });
285
+ // Record error metrics
286
+ cronErrors.add(1, {
287
+ env: process.env.NODE_ENV,
288
+ job_id: jobId,
289
+ workflow_name: this.configuration?.name || "unknown",
290
+ manual: String(manual),
291
+ });
292
+ this.logger.error(`Cron job failed ${jobId}: ${errorMessage}`, error.stack);
293
+ resolve({ ctx: {}, metrics: {} });
294
+ }
295
+ finally {
296
+ scheduledJob.running = false;
297
+ span.end();
298
+ }
299
+ });
300
+ });
301
+ }
302
+ }
303
+ exports.CronTrigger = CronTrigger;
304
+ exports.default = CronTrigger;
305
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CronTrigger.js","sourceRoot":"","sources":["../src/CronTrigger.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CronTrigger - Scheduled workflow execution based on cron expressions\n *\n * Extends TriggerBase to support scheduled triggers:\n * - Cron expressions (e.g., \"0 * * * *\" for hourly)\n * - Timezone-aware scheduling\n * - Overlap prevention\n * - Manual trigger support\n *\n * Uses the 'cron' package for cron parsing and scheduling.\n */\n\nimport type { CronTriggerOpts, HelperResponse } from \"@blok/helper\";\nimport {\n\tDefaultLogger,\n\ttype GlobalOptions,\n\ttype BlokService,\n\tNodeMap,\n\tTriggerBase,\n\ttype TriggerResponse,\n} from \"@blok/runner\";\nimport type { Context, MetricsType, RequestContext } from \"@blok/shared\";\nimport { type Span, SpanStatusCode, metrics, trace } from \"@opentelemetry/api\";\nimport { CronJob } from \"cron\";\nimport { v4 as uuid } from \"uuid\";\n\n/**\n * Scheduled job information\n */\nexport interface ScheduledJob {\n\t/** Unique job ID */\n\tid: string;\n\t/** Workflow path */\n\tworkflowPath: string;\n\t/** Cron expression */\n\tschedule: string;\n\t/** Timezone */\n\ttimezone: string;\n\t/** Allow overlapping executions */\n\toverlap: boolean;\n\t/** Whether the job is currently running */\n\trunning: boolean;\n\t/** Last execution time */\n\tlastRun?: Date;\n\t/** Next scheduled time */\n\tnextRun?: Date;\n\t/** Internal CronJob instance */\n\tjob: CronJob;\n}\n\n/**\n * Execution context passed to the workflow\n */\nexport interface CronExecutionContext {\n\t/** Job ID */\n\tjobId: string;\n\t/** Scheduled time (when it was supposed to run) */\n\tscheduledTime: Date;\n\t/** Actual execution time */\n\texecutionTime: Date;\n\t/** Cron expression */\n\tschedule: string;\n\t/** Timezone */\n\ttimezone: string;\n\t/** Whether this is a manual trigger */\n\tmanual: boolean;\n}\n\n/**\n * Workflow model with cron trigger configuration\n */\ninterface CronWorkflowModel {\n\tpath: string;\n\tconfig: {\n\t\tname: string;\n\t\tversion: string;\n\t\ttrigger?: {\n\t\t\tcron?: CronTriggerOpts;\n\t\t\t[key: string]: unknown;\n\t\t};\n\t\t[key: string]: unknown;\n\t};\n}\n\n/**\n * CronTrigger - Scheduled workflow execution\n */\nexport abstract class CronTrigger extends TriggerBase {\n\tprotected nodeMap: GlobalOptions = {} as GlobalOptions;\n\tprotected readonly tracer = trace.getTracer(\n\t\tprocess.env.PROJECT_NAME || \"trigger-cron-workflow\",\n\t\tprocess.env.PROJECT_VERSION || \"0.0.1\",\n\t);\n\tprotected readonly logger = new DefaultLogger();\n\tprotected jobs: Map<string, ScheduledJob> = new Map();\n\n\t// Subclasses provide these\n\tprotected abstract nodes: Record<string, BlokService<unknown>>;\n\tprotected abstract workflows: Record<string, HelperResponse>;\n\n\tconstructor() {\n\t\tsuper();\n\t\tthis.loadNodes();\n\t\tthis.loadWorkflows();\n\t}\n\n\t/**\n\t * Load nodes into the node map\n\t */\n\tloadNodes(): void {\n\t\tthis.nodeMap.nodes = new NodeMap();\n\t\tconst nodeKeys = Object.keys(this.nodes);\n\t\tfor (const key of nodeKeys) {\n\t\t\tthis.nodeMap.nodes.addNode(key, this.nodes[key]);\n\t\t}\n\t}\n\n\t/**\n\t * Load workflows into the workflow map\n\t */\n\tloadWorkflows(): void {\n\t\tthis.nodeMap.workflows = this.workflows;\n\t}\n\n\t/**\n\t * Convert cron DateTime to Date\n\t * The cron package uses luxon DateTime which has toJSDate()\n\t */\n\tprotected toDate(dateTime: unknown): Date {\n\t\tif (dateTime && typeof dateTime === \"object\" && \"toJSDate\" in dateTime) {\n\t\t\treturn (dateTime as { toJSDate: () => Date }).toJSDate();\n\t\t}\n\t\treturn dateTime instanceof Date ? dateTime : new Date(dateTime as string | number);\n\t}\n\n\t/**\n\t * Start the cron scheduler - main entry point\n\t */\n\tasync listen(): Promise<number> {\n\t\tconst startTime = this.startCounter();\n\n\t\ttry {\n\t\t\t// Find all workflows with cron triggers\n\t\t\tconst cronWorkflows = this.getCronWorkflows();\n\n\t\t\tif (cronWorkflows.length === 0) {\n\t\t\t\tthis.logger.log(\"No workflows with cron triggers found\");\n\t\t\t\treturn this.endCounter(startTime);\n\t\t\t}\n\n\t\t\t// Create and start cron jobs for each workflow\n\t\t\tfor (const workflow of cronWorkflows) {\n\t\t\t\tconst config = workflow.config.trigger?.cron as CronTriggerOpts;\n\t\t\t\tconst jobId = `cron-${workflow.path}-${uuid().slice(0, 8)}`;\n\n\t\t\t\tthis.logger.log(`Scheduling workflow: ${workflow.path} with schedule: ${config.schedule} (${config.timezone})`);\n\n\t\t\t\tconst job = new CronJob(\n\t\t\t\t\tconfig.schedule,\n\t\t\t\t\tasync () => {\n\t\t\t\t\t\tawait this.executeWorkflow(jobId, workflow, config, false);\n\t\t\t\t\t},\n\t\t\t\t\tnull, // onComplete\n\t\t\t\t\tfalse, // start\n\t\t\t\t\tconfig.timezone,\n\t\t\t\t);\n\n\t\t\t\tconst scheduledJob: ScheduledJob = {\n\t\t\t\t\tid: jobId,\n\t\t\t\t\tworkflowPath: workflow.path,\n\t\t\t\t\tschedule: config.schedule,\n\t\t\t\t\ttimezone: config.timezone,\n\t\t\t\t\toverlap: config.overlap ?? false,\n\t\t\t\t\trunning: false,\n\t\t\t\t\tnextRun: this.toDate(job.nextDate()),\n\t\t\t\t\tjob,\n\t\t\t\t};\n\n\t\t\t\tthis.jobs.set(jobId, scheduledJob);\n\n\t\t\t\t// Start the job\n\t\t\t\tjob.start();\n\t\t\t\tthis.logger.log(`Job ${jobId} started. Next run: ${scheduledJob.nextRun}`);\n\t\t\t}\n\n\t\t\tthis.logger.log(`Cron trigger started. ${this.jobs.size} job(s) scheduled`);\n\n\t\t\t// Enable HMR in development mode\n\t\t\tif (process.env.BLOK_HMR === \"true\" || process.env.NODE_ENV === \"development\") {\n\t\t\t\tawait this.enableHotReload();\n\t\t\t}\n\n\t\t\treturn this.endCounter(startTime);\n\t\t} catch (error) {\n\t\t\tthis.logger.error(`Failed to start cron trigger: ${(error as Error).message}`);\n\t\t\tthrow error;\n\t\t}\n\t}\n\n\t/**\n\t * Stop all cron jobs\n\t */\n\tasync stop(): Promise<void> {\n\t\tfor (const [jobId, scheduledJob] of this.jobs) {\n\t\t\tscheduledJob.job.stop();\n\t\t\tthis.logger.log(`Job ${jobId} stopped`);\n\t\t}\n\t\tthis.jobs.clear();\n\t\tthis.logger.log(\"Cron trigger stopped\");\n\t}\n\n\tprotected override async onHmrWorkflowChange(): Promise<void> {\n\t\tthis.logger.log(\"[HMR] Cron workflow changed, reloading...\");\n\t\tawait this.waitForInFlightRequests();\n\t\tawait this.stop();\n\t\tthis.loadWorkflows();\n\t\tawait this.listen();\n\t}\n\n\t/**\n\t * Manually trigger a specific job\n\t */\n\tasync triggerJob(jobId: string): Promise<TriggerResponse | null> {\n\t\tconst scheduledJob = this.jobs.get(jobId);\n\t\tif (!scheduledJob) {\n\t\t\tthis.logger.error(`Job not found: ${jobId}`);\n\t\t\treturn null;\n\t\t}\n\n\t\t// Get the workflow\n\t\tconst workflow = this.getWorkflowByPath(scheduledJob.workflowPath);\n\t\tif (!workflow) {\n\t\t\tthis.logger.error(`Workflow not found: ${scheduledJob.workflowPath}`);\n\t\t\treturn null;\n\t\t}\n\n\t\tconst config = workflow.config.trigger?.cron as CronTriggerOpts;\n\t\treturn this.executeWorkflow(jobId, workflow, config, true);\n\t}\n\n\t/**\n\t * Get all scheduled jobs\n\t */\n\tgetJobs(): ScheduledJob[] {\n\t\treturn Array.from(this.jobs.values()).map((job) => ({\n\t\t\t...job,\n\t\t\tnextRun: this.toDate(job.job.nextDate()),\n\t\t}));\n\t}\n\n\t/**\n\t * Get all workflows that have cron triggers\n\t */\n\tprotected getCronWorkflows(): CronWorkflowModel[] {\n\t\tconst workflows: CronWorkflowModel[] = [];\n\n\t\tfor (const [path, workflow] of Object.entries(this.nodeMap.workflows || {})) {\n\t\t\tconst workflowConfig = (workflow as unknown as { _config: CronWorkflowModel[\"config\"] })._config;\n\n\t\t\tif (workflowConfig?.trigger) {\n\t\t\t\tconst triggerType = Object.keys(workflowConfig.trigger)[0];\n\n\t\t\t\tif (triggerType === \"cron\" && workflowConfig.trigger.cron) {\n\t\t\t\t\tworkflows.push({\n\t\t\t\t\t\tpath,\n\t\t\t\t\t\tconfig: workflowConfig,\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn workflows;\n\t}\n\n\t/**\n\t * Get workflow by path\n\t */\n\tprotected getWorkflowByPath(path: string): CronWorkflowModel | null {\n\t\tconst workflow = this.nodeMap.workflows?.[path];\n\t\tif (!workflow) return null;\n\n\t\tconst workflowConfig = (workflow as unknown as { _config: CronWorkflowModel[\"config\"] })._config;\n\t\treturn {\n\t\t\tpath,\n\t\t\tconfig: workflowConfig,\n\t\t};\n\t}\n\n\t/**\n\t * Execute a workflow\n\t */\n\tprotected async executeWorkflow(\n\t\tjobId: string,\n\t\tworkflow: CronWorkflowModel,\n\t\tconfig: CronTriggerOpts,\n\t\tmanual: boolean,\n\t): Promise<TriggerResponse> {\n\t\tconst scheduledJob = this.jobs.get(jobId);\n\t\tif (!scheduledJob) {\n\t\t\tthrow new Error(`Job not found: ${jobId}`);\n\t\t}\n\n\t\t// Check for overlap\n\t\tif (scheduledJob.running && !scheduledJob.overlap) {\n\t\t\tthis.logger.log(`Skipping ${jobId}: previous execution still running (overlap disabled)`);\n\t\t\treturn { ctx: {} as Context, metrics: {} as MetricsType };\n\t\t}\n\n\t\tconst executionId = uuid();\n\t\tconst lastDate = scheduledJob.job.lastDate();\n\t\tconst scheduledTime = lastDate ? new Date(lastDate as unknown as string | number) : new Date();\n\t\tconst executionTime = new Date();\n\n\t\tconst defaultMeter = metrics.getMeter(\"default\");\n\t\tconst cronExecutions = defaultMeter.createCounter(\"cron_executions\", {\n\t\t\tdescription: \"Cron job executions\",\n\t\t});\n\t\tconst cronErrors = defaultMeter.createCounter(\"cron_errors\", {\n\t\t\tdescription: \"Cron job execution errors\",\n\t\t});\n\n\t\treturn new Promise((resolve) => {\n\t\t\tthis.tracer.startActiveSpan(`cron:${workflow.path}`, async (span: Span) => {\n\t\t\t\tscheduledJob.running = true;\n\n\t\t\t\ttry {\n\t\t\t\t\tconst start = performance.now();\n\n\t\t\t\t\t// Initialize configuration for this workflow\n\t\t\t\t\tawait this.configuration.init(workflow.path, this.nodeMap);\n\n\t\t\t\t\t// Create context\n\t\t\t\t\tconst ctx: Context = this.createContext(undefined, workflow.path, executionId);\n\n\t\t\t\t\t// Create execution context\n\t\t\t\t\tconst cronContext: CronExecutionContext = {\n\t\t\t\t\t\tjobId,\n\t\t\t\t\t\tscheduledTime,\n\t\t\t\t\t\texecutionTime,\n\t\t\t\t\t\tschedule: config.schedule,\n\t\t\t\t\t\ttimezone: config.timezone,\n\t\t\t\t\t\tmanual,\n\t\t\t\t\t};\n\n\t\t\t\t\t// Populate request with cron context\n\t\t\t\t\tctx.request = {\n\t\t\t\t\t\tbody: cronContext,\n\t\t\t\t\t\theaders: {\n\t\t\t\t\t\t\t\"x-cron-job-id\": jobId,\n\t\t\t\t\t\t\t\"x-cron-schedule\": config.schedule,\n\t\t\t\t\t\t\t\"x-cron-timezone\": config.timezone,\n\t\t\t\t\t\t\t\"x-cron-manual\": String(manual),\n\t\t\t\t\t\t},\n\t\t\t\t\t\tquery: {},\n\t\t\t\t\t\tparams: {\n\t\t\t\t\t\t\tjobId,\n\t\t\t\t\t\t\tschedule: config.schedule,\n\t\t\t\t\t\t},\n\t\t\t\t\t} as unknown as RequestContext;\n\n\t\t\t\t\t// Store cron context in vars\n\t\t\t\t\tif (!ctx.vars) ctx.vars = {};\n\t\t\t\t\tctx.vars._cron_context = {\n\t\t\t\t\t\tjobId,\n\t\t\t\t\t\tscheduledTime: scheduledTime.toISOString(),\n\t\t\t\t\t\texecutionTime: executionTime.toISOString(),\n\t\t\t\t\t\tschedule: config.schedule,\n\t\t\t\t\t\ttimezone: config.timezone,\n\t\t\t\t\t\tmanual: String(manual),\n\t\t\t\t\t};\n\n\t\t\t\t\tctx.logger.log(`Executing cron job: ${jobId} (${manual ? \"manual\" : \"scheduled\"})`);\n\n\t\t\t\t\t// Execute workflow\n\t\t\t\t\tconst response: TriggerResponse = await this.run(ctx);\n\t\t\t\t\tconst end = performance.now();\n\n\t\t\t\t\t// Update job state\n\t\t\t\t\tscheduledJob.lastRun = executionTime;\n\t\t\t\t\tscheduledJob.nextRun = this.toDate(scheduledJob.job.nextDate());\n\n\t\t\t\t\t// Set span attributes\n\t\t\t\t\tspan.setAttribute(\"success\", true);\n\t\t\t\t\tspan.setAttribute(\"job_id\", jobId);\n\t\t\t\t\tspan.setAttribute(\"workflow_path\", workflow.path);\n\t\t\t\t\tspan.setAttribute(\"schedule\", config.schedule);\n\t\t\t\t\tspan.setAttribute(\"timezone\", config.timezone);\n\t\t\t\t\tspan.setAttribute(\"manual\", manual);\n\t\t\t\t\tspan.setAttribute(\"elapsed_ms\", end - start);\n\t\t\t\t\tspan.setStatus({ code: SpanStatusCode.OK });\n\n\t\t\t\t\t// Record metrics\n\t\t\t\t\tcronExecutions.add(1, {\n\t\t\t\t\t\tenv: process.env.NODE_ENV,\n\t\t\t\t\t\tjob_id: jobId,\n\t\t\t\t\t\tworkflow_name: this.configuration.name,\n\t\t\t\t\t\tmanual: String(manual),\n\t\t\t\t\t\tsuccess: \"true\",\n\t\t\t\t\t});\n\n\t\t\t\t\tctx.logger.log(`Cron job completed in ${(end - start).toFixed(2)}ms: ${jobId}`);\n\n\t\t\t\t\tresolve(response);\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconst errorMessage = (error as Error).message;\n\n\t\t\t\t\t// Set span error\n\t\t\t\t\tspan.setAttribute(\"success\", false);\n\t\t\t\t\tspan.recordException(error as Error);\n\t\t\t\t\tspan.setStatus({ code: SpanStatusCode.ERROR, message: errorMessage });\n\n\t\t\t\t\t// Record error metrics\n\t\t\t\t\tcronErrors.add(1, {\n\t\t\t\t\t\tenv: process.env.NODE_ENV,\n\t\t\t\t\t\tjob_id: jobId,\n\t\t\t\t\t\tworkflow_name: this.configuration?.name || \"unknown\",\n\t\t\t\t\t\tmanual: String(manual),\n\t\t\t\t\t});\n\n\t\t\t\t\tthis.logger.error(`Cron job failed ${jobId}: ${errorMessage}`, (error as Error).stack);\n\n\t\t\t\t\tresolve({ ctx: {} as Context, metrics: {} as MetricsType });\n\t\t\t\t} finally {\n\t\t\t\t\tscheduledJob.running = false;\n\t\t\t\t\tspan.end();\n\t\t\t\t}\n\t\t\t});\n\t\t});\n\t}\n}\n\nexport default CronTrigger;\n"]}
@@ -0,0 +1,64 @@
1
+ /**
2
+ * @blok/trigger-cron
3
+ *
4
+ * Cron/scheduled trigger for Blok workflows.
5
+ * Execute workflows on a schedule using cron expressions.
6
+ *
7
+ * Features:
8
+ * - Standard cron expression syntax
9
+ * - Timezone-aware scheduling
10
+ * - Overlap prevention
11
+ * - Manual trigger support
12
+ * - Job management API
13
+ *
14
+ * @example
15
+ * ```typescript
16
+ * import { CronTrigger } from "@blok/trigger-cron";
17
+ *
18
+ * class MyCronTrigger extends CronTrigger {
19
+ * protected nodes = myNodes;
20
+ * protected workflows = myWorkflows;
21
+ * }
22
+ *
23
+ * const trigger = new MyCronTrigger();
24
+ * await trigger.listen();
25
+ *
26
+ * // List all scheduled jobs
27
+ * const jobs = trigger.getJobs();
28
+ *
29
+ * // Manually trigger a job
30
+ * await trigger.triggerJob("cron-my-workflow-abc123");
31
+ * ```
32
+ *
33
+ * Workflow Definition:
34
+ * ```typescript
35
+ * Workflow({ name: "daily-cleanup", version: "1.0.0" })
36
+ * .addTrigger("cron", {
37
+ * schedule: "0 2 * * *", // Run at 2 AM daily
38
+ * timezone: "America/New_York",
39
+ * overlap: false,
40
+ * })
41
+ * .addStep({ ... });
42
+ * ```
43
+ *
44
+ * Cron Expression Format:
45
+ * ```
46
+ * * * * * * *
47
+ * │ │ │ │ │ │
48
+ * │ │ │ │ │ └── Day of week (0-7, Sun=0,7)
49
+ * │ │ │ │ └──── Month (1-12)
50
+ * │ │ │ └────── Day of month (1-31)
51
+ * │ │ └──────── Hour (0-23)
52
+ * │ └────────── Minute (0-59)
53
+ * └──────────── Second (0-59, optional)
54
+ * ```
55
+ *
56
+ * Common schedules:
57
+ * - "* * * * *" - Every minute
58
+ * - "0 * * * *" - Every hour
59
+ * - "0 0 * * *" - Every day at midnight
60
+ * - "0 0 * * 0" - Every Sunday at midnight
61
+ * - "0 0 1 * *" - First day of every month
62
+ */
63
+ export { CronTrigger, type ScheduledJob, type CronExecutionContext, } from "./CronTrigger";
64
+ export type { CronTriggerOpts } from "@blok/helper";
package/dist/index.js ADDED
@@ -0,0 +1,69 @@
1
+ "use strict";
2
+ /**
3
+ * @blok/trigger-cron
4
+ *
5
+ * Cron/scheduled trigger for Blok workflows.
6
+ * Execute workflows on a schedule using cron expressions.
7
+ *
8
+ * Features:
9
+ * - Standard cron expression syntax
10
+ * - Timezone-aware scheduling
11
+ * - Overlap prevention
12
+ * - Manual trigger support
13
+ * - Job management API
14
+ *
15
+ * @example
16
+ * ```typescript
17
+ * import { CronTrigger } from "@blok/trigger-cron";
18
+ *
19
+ * class MyCronTrigger extends CronTrigger {
20
+ * protected nodes = myNodes;
21
+ * protected workflows = myWorkflows;
22
+ * }
23
+ *
24
+ * const trigger = new MyCronTrigger();
25
+ * await trigger.listen();
26
+ *
27
+ * // List all scheduled jobs
28
+ * const jobs = trigger.getJobs();
29
+ *
30
+ * // Manually trigger a job
31
+ * await trigger.triggerJob("cron-my-workflow-abc123");
32
+ * ```
33
+ *
34
+ * Workflow Definition:
35
+ * ```typescript
36
+ * Workflow({ name: "daily-cleanup", version: "1.0.0" })
37
+ * .addTrigger("cron", {
38
+ * schedule: "0 2 * * *", // Run at 2 AM daily
39
+ * timezone: "America/New_York",
40
+ * overlap: false,
41
+ * })
42
+ * .addStep({ ... });
43
+ * ```
44
+ *
45
+ * Cron Expression Format:
46
+ * ```
47
+ * * * * * * *
48
+ * │ │ │ │ │ │
49
+ * │ │ │ │ │ └── Day of week (0-7, Sun=0,7)
50
+ * │ │ │ │ └──── Month (1-12)
51
+ * │ │ │ └────── Day of month (1-31)
52
+ * │ │ └──────── Hour (0-23)
53
+ * │ └────────── Minute (0-59)
54
+ * └──────────── Second (0-59, optional)
55
+ * ```
56
+ *
57
+ * Common schedules:
58
+ * - "* * * * *" - Every minute
59
+ * - "0 * * * *" - Every hour
60
+ * - "0 0 * * *" - Every day at midnight
61
+ * - "0 0 * * 0" - Every Sunday at midnight
62
+ * - "0 0 1 * *" - First day of every month
63
+ */
64
+ Object.defineProperty(exports, "__esModule", { value: true });
65
+ exports.CronTrigger = void 0;
66
+ // Core exports
67
+ var CronTrigger_1 = require("./CronTrigger");
68
+ Object.defineProperty(exports, "CronTrigger", { enumerable: true, get: function () { return CronTrigger_1.CronTrigger; } });
69
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNkRHOzs7QUFFSCxlQUFlO0FBQ2YsNkNBSXVCO0FBSHRCLDBHQUFBLFdBQVcsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGJsb2svdHJpZ2dlci1jcm9uXG4gKlxuICogQ3Jvbi9zY2hlZHVsZWQgdHJpZ2dlciBmb3IgQmxvayB3b3JrZmxvd3MuXG4gKiBFeGVjdXRlIHdvcmtmbG93cyBvbiBhIHNjaGVkdWxlIHVzaW5nIGNyb24gZXhwcmVzc2lvbnMuXG4gKlxuICogRmVhdHVyZXM6XG4gKiAtIFN0YW5kYXJkIGNyb24gZXhwcmVzc2lvbiBzeW50YXhcbiAqIC0gVGltZXpvbmUtYXdhcmUgc2NoZWR1bGluZ1xuICogLSBPdmVybGFwIHByZXZlbnRpb25cbiAqIC0gTWFudWFsIHRyaWdnZXIgc3VwcG9ydFxuICogLSBKb2IgbWFuYWdlbWVudCBBUElcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaW1wb3J0IHsgQ3JvblRyaWdnZXIgfSBmcm9tIFwiQGJsb2svdHJpZ2dlci1jcm9uXCI7XG4gKlxuICogY2xhc3MgTXlDcm9uVHJpZ2dlciBleHRlbmRzIENyb25UcmlnZ2VyIHtcbiAqICAgcHJvdGVjdGVkIG5vZGVzID0gbXlOb2RlcztcbiAqICAgcHJvdGVjdGVkIHdvcmtmbG93cyA9IG15V29ya2Zsb3dzO1xuICogfVxuICpcbiAqIGNvbnN0IHRyaWdnZXIgPSBuZXcgTXlDcm9uVHJpZ2dlcigpO1xuICogYXdhaXQgdHJpZ2dlci5saXN0ZW4oKTtcbiAqXG4gKiAvLyBMaXN0IGFsbCBzY2hlZHVsZWQgam9ic1xuICogY29uc3Qgam9icyA9IHRyaWdnZXIuZ2V0Sm9icygpO1xuICpcbiAqIC8vIE1hbnVhbGx5IHRyaWdnZXIgYSBqb2JcbiAqIGF3YWl0IHRyaWdnZXIudHJpZ2dlckpvYihcImNyb24tbXktd29ya2Zsb3ctYWJjMTIzXCIpO1xuICogYGBgXG4gKlxuICogV29ya2Zsb3cgRGVmaW5pdGlvbjpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIFdvcmtmbG93KHsgbmFtZTogXCJkYWlseS1jbGVhbnVwXCIsIHZlcnNpb246IFwiMS4wLjBcIiB9KVxuICogICAuYWRkVHJpZ2dlcihcImNyb25cIiwge1xuICogICAgIHNjaGVkdWxlOiBcIjAgMiAqICogKlwiLCAgLy8gUnVuIGF0IDIgQU0gZGFpbHlcbiAqICAgICB0aW1lem9uZTogXCJBbWVyaWNhL05ld19Zb3JrXCIsXG4gKiAgICAgb3ZlcmxhcDogZmFsc2UsXG4gKiAgIH0pXG4gKiAgIC5hZGRTdGVwKHsgLi4uIH0pO1xuICogYGBgXG4gKlxuICogQ3JvbiBFeHByZXNzaW9uIEZvcm1hdDpcbiAqIGBgYFxuICogKiAqICogKiAqICpcbiAqIOKUgiDilIIg4pSCIOKUgiDilIIg4pSCXG4gKiDilIIg4pSCIOKUgiDilIIg4pSCIOKUlOKUgOKUgCBEYXkgb2Ygd2VlayAoMC03LCBTdW49MCw3KVxuICog4pSCIOKUgiDilIIg4pSCIOKUlOKUgOKUgOKUgOKUgCBNb250aCAoMS0xMilcbiAqIOKUgiDilIIg4pSCIOKUlOKUgOKUgOKUgOKUgOKUgOKUgCBEYXkgb2YgbW9udGggKDEtMzEpXG4gKiDilIIg4pSCIOKUlOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgCBIb3VyICgwLTIzKVxuICog4pSCIOKUlOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgCBNaW51dGUgKDAtNTkpXG4gKiDilJTilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIAgU2Vjb25kICgwLTU5LCBvcHRpb25hbClcbiAqIGBgYFxuICpcbiAqIENvbW1vbiBzY2hlZHVsZXM6XG4gKiAtIFwiKiAqICogKiAqXCIgLSBFdmVyeSBtaW51dGVcbiAqIC0gXCIwICogKiAqICpcIiAtIEV2ZXJ5IGhvdXJcbiAqIC0gXCIwIDAgKiAqICpcIiAtIEV2ZXJ5IGRheSBhdCBtaWRuaWdodFxuICogLSBcIjAgMCAqICogMFwiIC0gRXZlcnkgU3VuZGF5IGF0IG1pZG5pZ2h0XG4gKiAtIFwiMCAwIDEgKiAqXCIgLSBGaXJzdCBkYXkgb2YgZXZlcnkgbW9udGhcbiAqL1xuXG4vLyBDb3JlIGV4cG9ydHNcbmV4cG9ydCB7XG5cdENyb25UcmlnZ2VyLFxuXHR0eXBlIFNjaGVkdWxlZEpvYixcblx0dHlwZSBDcm9uRXhlY3V0aW9uQ29udGV4dCxcbn0gZnJvbSBcIi4vQ3JvblRyaWdnZXJcIjtcblxuLy8gUmUtZXhwb3J0IHR5cGVzIGZyb20gaGVscGVyIGZvciBjb252ZW5pZW5jZVxuZXhwb3J0IHR5cGUgeyBDcm9uVHJpZ2dlck9wdHMgfSBmcm9tIFwiQGJsb2svaGVscGVyXCI7XG4iXX0=
package/package.json ADDED
@@ -0,0 +1,35 @@
1
+ {
2
+ "name": "@blokjs/trigger-cron",
3
+ "version": "0.2.0",
4
+ "description": "Cron/scheduled trigger for Blok workflows - supports cron expressions and interval-based scheduling",
5
+ "type": "module",
6
+ "main": "dist/index.js",
7
+ "types": "dist/index.d.ts",
8
+ "scripts": {
9
+ "build": "rm -rf dist && bun run tsc",
10
+ "build:dev": "tsc --watch",
11
+ "test": "vitest run",
12
+ "test:dev": "vitest"
13
+ },
14
+ "author": "Deskree Technologies Inc.",
15
+ "license": "Apache-2.0",
16
+ "dependencies": {
17
+ "@blokjs/helper": "workspace:*",
18
+ "@blokjs/runner": "workspace:*",
19
+ "@blokjs/shared": "workspace:*",
20
+ "@opentelemetry/api": "^1.9.0",
21
+ "cron": "^4.4.0",
22
+ "uuid": "^11.1.0"
23
+ },
24
+ "devDependencies": {
25
+ "@types/cron": "^2.4.3",
26
+ "@types/node": "^22.15.21",
27
+ "@types/uuid": "^11.0.0",
28
+ "typescript": "^5.8.3",
29
+ "vitest": "^4.0.18"
30
+ },
31
+ "private": false,
32
+ "publishConfig": {
33
+ "access": "public"
34
+ }
35
+ }
@@ -0,0 +1,119 @@
1
+ /**
2
+ * CronTrigger Tests
3
+ */
4
+
5
+ import { describe, expect, it, vi } from "vitest";
6
+
7
+ describe("CronTrigger", () => {
8
+ describe("ScheduledJob Interface", () => {
9
+ it("should accept valid scheduled job structure", () => {
10
+ const mockJob = {
11
+ nextDate: () => new Date(),
12
+ lastDate: () => new Date(),
13
+ start: vi.fn(),
14
+ stop: vi.fn(),
15
+ };
16
+
17
+ const scheduledJob = {
18
+ id: "cron-test-abc123",
19
+ workflowPath: "daily-cleanup",
20
+ schedule: "0 2 * * *",
21
+ timezone: "America/New_York",
22
+ overlap: false,
23
+ running: false,
24
+ lastRun: new Date(),
25
+ nextRun: new Date(),
26
+ job: mockJob,
27
+ };
28
+
29
+ expect(scheduledJob.id).toBe("cron-test-abc123");
30
+ expect(scheduledJob.schedule).toBe("0 2 * * *");
31
+ expect(scheduledJob.timezone).toBe("America/New_York");
32
+ expect(scheduledJob.overlap).toBe(false);
33
+ });
34
+ });
35
+
36
+ describe("CronExecutionContext Interface", () => {
37
+ it("should create valid execution context", () => {
38
+ const context = {
39
+ jobId: "cron-job-123",
40
+ scheduledTime: new Date("2024-01-01T02:00:00Z"),
41
+ executionTime: new Date("2024-01-01T02:00:01Z"),
42
+ schedule: "0 2 * * *",
43
+ timezone: "UTC",
44
+ manual: false,
45
+ };
46
+
47
+ expect(context.jobId).toBe("cron-job-123");
48
+ expect(context.scheduledTime.toISOString()).toBe("2024-01-01T02:00:00.000Z");
49
+ expect(context.manual).toBe(false);
50
+ });
51
+
52
+ it("should support manual trigger context", () => {
53
+ const context = {
54
+ jobId: "cron-job-456",
55
+ scheduledTime: new Date(),
56
+ executionTime: new Date(),
57
+ schedule: "0 * * * *",
58
+ timezone: "Europe/London",
59
+ manual: true,
60
+ };
61
+
62
+ expect(context.manual).toBe(true);
63
+ });
64
+ });
65
+
66
+ describe("CronTriggerOpts Schema", () => {
67
+ it("should validate cron trigger configuration", () => {
68
+ const validConfig = {
69
+ schedule: "0 * * * *",
70
+ timezone: "America/Los_Angeles",
71
+ overlap: false,
72
+ };
73
+
74
+ expect(validConfig.schedule).toBe("0 * * * *");
75
+ expect(validConfig.timezone).toBe("America/Los_Angeles");
76
+ expect(validConfig.overlap).toBe(false);
77
+ });
78
+
79
+ it("should support common cron expressions", () => {
80
+ const schedules = [
81
+ { expr: "* * * * *", desc: "Every minute" },
82
+ { expr: "0 * * * *", desc: "Every hour" },
83
+ { expr: "0 0 * * *", desc: "Every day at midnight" },
84
+ { expr: "0 0 * * 0", desc: "Every Sunday" },
85
+ { expr: "0 0 1 * *", desc: "First day of month" },
86
+ { expr: "*/5 * * * *", desc: "Every 5 minutes" },
87
+ { expr: "0 9-17 * * 1-5", desc: "Hourly, 9-5 on weekdays" },
88
+ ];
89
+
90
+ for (const schedule of schedules) {
91
+ expect(schedule.expr).toBeDefined();
92
+ expect(schedule.desc).toBeDefined();
93
+ }
94
+ });
95
+ });
96
+
97
+ describe("Timezone Support", () => {
98
+ it("should support common timezones", () => {
99
+ const timezones = [
100
+ "UTC",
101
+ "America/New_York",
102
+ "America/Los_Angeles",
103
+ "Europe/London",
104
+ "Europe/Paris",
105
+ "Asia/Tokyo",
106
+ "Australia/Sydney",
107
+ ];
108
+
109
+ for (const tz of timezones) {
110
+ const config = {
111
+ schedule: "0 0 * * *",
112
+ timezone: tz,
113
+ overlap: false,
114
+ };
115
+ expect(config.timezone).toBe(tz);
116
+ }
117
+ });
118
+ });
119
+ });
@@ -0,0 +1,432 @@
1
+ /**
2
+ * CronTrigger - Scheduled workflow execution based on cron expressions
3
+ *
4
+ * Extends TriggerBase to support scheduled triggers:
5
+ * - Cron expressions (e.g., "0 * * * *" for hourly)
6
+ * - Timezone-aware scheduling
7
+ * - Overlap prevention
8
+ * - Manual trigger support
9
+ *
10
+ * Uses the 'cron' package for cron parsing and scheduling.
11
+ */
12
+
13
+ import type { CronTriggerOpts, HelperResponse } from "@blokjs/helper";
14
+ import {
15
+ DefaultLogger,
16
+ type GlobalOptions,
17
+ type BlokService,
18
+ NodeMap,
19
+ TriggerBase,
20
+ type TriggerResponse,
21
+ } from "@blokjs/runner";
22
+ import type { Context, MetricsType, RequestContext } from "@blokjs/shared";
23
+ import { type Span, SpanStatusCode, metrics, trace } from "@opentelemetry/api";
24
+ import { CronJob } from "cron";
25
+ import { v4 as uuid } from "uuid";
26
+
27
+ /**
28
+ * Scheduled job information
29
+ */
30
+ export interface ScheduledJob {
31
+ /** Unique job ID */
32
+ id: string;
33
+ /** Workflow path */
34
+ workflowPath: string;
35
+ /** Cron expression */
36
+ schedule: string;
37
+ /** Timezone */
38
+ timezone: string;
39
+ /** Allow overlapping executions */
40
+ overlap: boolean;
41
+ /** Whether the job is currently running */
42
+ running: boolean;
43
+ /** Last execution time */
44
+ lastRun?: Date;
45
+ /** Next scheduled time */
46
+ nextRun?: Date;
47
+ /** Internal CronJob instance */
48
+ job: CronJob;
49
+ }
50
+
51
+ /**
52
+ * Execution context passed to the workflow
53
+ */
54
+ export interface CronExecutionContext {
55
+ /** Job ID */
56
+ jobId: string;
57
+ /** Scheduled time (when it was supposed to run) */
58
+ scheduledTime: Date;
59
+ /** Actual execution time */
60
+ executionTime: Date;
61
+ /** Cron expression */
62
+ schedule: string;
63
+ /** Timezone */
64
+ timezone: string;
65
+ /** Whether this is a manual trigger */
66
+ manual: boolean;
67
+ }
68
+
69
+ /**
70
+ * Workflow model with cron trigger configuration
71
+ */
72
+ interface CronWorkflowModel {
73
+ path: string;
74
+ config: {
75
+ name: string;
76
+ version: string;
77
+ trigger?: {
78
+ cron?: CronTriggerOpts;
79
+ [key: string]: unknown;
80
+ };
81
+ [key: string]: unknown;
82
+ };
83
+ }
84
+
85
+ /**
86
+ * CronTrigger - Scheduled workflow execution
87
+ */
88
+ export abstract class CronTrigger extends TriggerBase {
89
+ protected nodeMap: GlobalOptions = {} as GlobalOptions;
90
+ protected readonly tracer = trace.getTracer(
91
+ process.env.PROJECT_NAME || "trigger-cron-workflow",
92
+ process.env.PROJECT_VERSION || "0.0.1",
93
+ );
94
+ protected readonly logger = new DefaultLogger();
95
+ protected jobs: Map<string, ScheduledJob> = new Map();
96
+
97
+ // Subclasses provide these
98
+ protected abstract nodes: Record<string, BlokService<unknown>>;
99
+ protected abstract workflows: Record<string, HelperResponse>;
100
+
101
+ constructor() {
102
+ super();
103
+ this.loadNodes();
104
+ this.loadWorkflows();
105
+ }
106
+
107
+ /**
108
+ * Load nodes into the node map
109
+ */
110
+ loadNodes(): void {
111
+ this.nodeMap.nodes = new NodeMap();
112
+ const nodeKeys = Object.keys(this.nodes);
113
+ for (const key of nodeKeys) {
114
+ this.nodeMap.nodes.addNode(key, this.nodes[key]);
115
+ }
116
+ }
117
+
118
+ /**
119
+ * Load workflows into the workflow map
120
+ */
121
+ loadWorkflows(): void {
122
+ this.nodeMap.workflows = this.workflows;
123
+ }
124
+
125
+ /**
126
+ * Convert cron DateTime to Date
127
+ * The cron package uses luxon DateTime which has toJSDate()
128
+ */
129
+ protected toDate(dateTime: unknown): Date {
130
+ if (dateTime && typeof dateTime === "object" && "toJSDate" in dateTime) {
131
+ return (dateTime as { toJSDate: () => Date }).toJSDate();
132
+ }
133
+ return dateTime instanceof Date ? dateTime : new Date(dateTime as string | number);
134
+ }
135
+
136
+ /**
137
+ * Start the cron scheduler - main entry point
138
+ */
139
+ async listen(): Promise<number> {
140
+ const startTime = this.startCounter();
141
+
142
+ try {
143
+ // Find all workflows with cron triggers
144
+ const cronWorkflows = this.getCronWorkflows();
145
+
146
+ if (cronWorkflows.length === 0) {
147
+ this.logger.log("No workflows with cron triggers found");
148
+ return this.endCounter(startTime);
149
+ }
150
+
151
+ // Create and start cron jobs for each workflow
152
+ for (const workflow of cronWorkflows) {
153
+ const config = workflow.config.trigger?.cron as CronTriggerOpts;
154
+ const jobId = `cron-${workflow.path}-${uuid().slice(0, 8)}`;
155
+
156
+ this.logger.log(`Scheduling workflow: ${workflow.path} with schedule: ${config.schedule} (${config.timezone})`);
157
+
158
+ const job = new CronJob(
159
+ config.schedule,
160
+ async () => {
161
+ await this.executeWorkflow(jobId, workflow, config, false);
162
+ },
163
+ null, // onComplete
164
+ false, // start
165
+ config.timezone,
166
+ );
167
+
168
+ const scheduledJob: ScheduledJob = {
169
+ id: jobId,
170
+ workflowPath: workflow.path,
171
+ schedule: config.schedule,
172
+ timezone: config.timezone,
173
+ overlap: config.overlap ?? false,
174
+ running: false,
175
+ nextRun: this.toDate(job.nextDate()),
176
+ job,
177
+ };
178
+
179
+ this.jobs.set(jobId, scheduledJob);
180
+
181
+ // Start the job
182
+ job.start();
183
+ this.logger.log(`Job ${jobId} started. Next run: ${scheduledJob.nextRun}`);
184
+ }
185
+
186
+ this.logger.log(`Cron trigger started. ${this.jobs.size} job(s) scheduled`);
187
+
188
+ // Enable HMR in development mode
189
+ if (process.env.BLOK_HMR === "true" || process.env.NODE_ENV === "development") {
190
+ await this.enableHotReload();
191
+ }
192
+
193
+ return this.endCounter(startTime);
194
+ } catch (error) {
195
+ this.logger.error(`Failed to start cron trigger: ${(error as Error).message}`);
196
+ throw error;
197
+ }
198
+ }
199
+
200
+ /**
201
+ * Stop all cron jobs
202
+ */
203
+ async stop(): Promise<void> {
204
+ for (const [jobId, scheduledJob] of this.jobs) {
205
+ scheduledJob.job.stop();
206
+ this.logger.log(`Job ${jobId} stopped`);
207
+ }
208
+ this.jobs.clear();
209
+ this.logger.log("Cron trigger stopped");
210
+ }
211
+
212
+ protected override async onHmrWorkflowChange(): Promise<void> {
213
+ this.logger.log("[HMR] Cron workflow changed, reloading...");
214
+ await this.waitForInFlightRequests();
215
+ await this.stop();
216
+ this.loadWorkflows();
217
+ await this.listen();
218
+ }
219
+
220
+ /**
221
+ * Manually trigger a specific job
222
+ */
223
+ async triggerJob(jobId: string): Promise<TriggerResponse | null> {
224
+ const scheduledJob = this.jobs.get(jobId);
225
+ if (!scheduledJob) {
226
+ this.logger.error(`Job not found: ${jobId}`);
227
+ return null;
228
+ }
229
+
230
+ // Get the workflow
231
+ const workflow = this.getWorkflowByPath(scheduledJob.workflowPath);
232
+ if (!workflow) {
233
+ this.logger.error(`Workflow not found: ${scheduledJob.workflowPath}`);
234
+ return null;
235
+ }
236
+
237
+ const config = workflow.config.trigger?.cron as CronTriggerOpts;
238
+ return this.executeWorkflow(jobId, workflow, config, true);
239
+ }
240
+
241
+ /**
242
+ * Get all scheduled jobs
243
+ */
244
+ getJobs(): ScheduledJob[] {
245
+ return Array.from(this.jobs.values()).map((job) => ({
246
+ ...job,
247
+ nextRun: this.toDate(job.job.nextDate()),
248
+ }));
249
+ }
250
+
251
+ /**
252
+ * Get all workflows that have cron triggers
253
+ */
254
+ protected getCronWorkflows(): CronWorkflowModel[] {
255
+ const workflows: CronWorkflowModel[] = [];
256
+
257
+ for (const [path, workflow] of Object.entries(this.nodeMap.workflows || {})) {
258
+ const workflowConfig = (workflow as unknown as { _config: CronWorkflowModel["config"] })._config;
259
+
260
+ if (workflowConfig?.trigger) {
261
+ const triggerType = Object.keys(workflowConfig.trigger)[0];
262
+
263
+ if (triggerType === "cron" && workflowConfig.trigger.cron) {
264
+ workflows.push({
265
+ path,
266
+ config: workflowConfig,
267
+ });
268
+ }
269
+ }
270
+ }
271
+
272
+ return workflows;
273
+ }
274
+
275
+ /**
276
+ * Get workflow by path
277
+ */
278
+ protected getWorkflowByPath(path: string): CronWorkflowModel | null {
279
+ const workflow = this.nodeMap.workflows?.[path];
280
+ if (!workflow) return null;
281
+
282
+ const workflowConfig = (workflow as unknown as { _config: CronWorkflowModel["config"] })._config;
283
+ return {
284
+ path,
285
+ config: workflowConfig,
286
+ };
287
+ }
288
+
289
+ /**
290
+ * Execute a workflow
291
+ */
292
+ protected async executeWorkflow(
293
+ jobId: string,
294
+ workflow: CronWorkflowModel,
295
+ config: CronTriggerOpts,
296
+ manual: boolean,
297
+ ): Promise<TriggerResponse> {
298
+ const scheduledJob = this.jobs.get(jobId);
299
+ if (!scheduledJob) {
300
+ throw new Error(`Job not found: ${jobId}`);
301
+ }
302
+
303
+ // Check for overlap
304
+ if (scheduledJob.running && !scheduledJob.overlap) {
305
+ this.logger.log(`Skipping ${jobId}: previous execution still running (overlap disabled)`);
306
+ return { ctx: {} as Context, metrics: {} as MetricsType };
307
+ }
308
+
309
+ const executionId = uuid();
310
+ const lastDate = scheduledJob.job.lastDate();
311
+ const scheduledTime = lastDate ? new Date(lastDate as unknown as string | number) : new Date();
312
+ const executionTime = new Date();
313
+
314
+ const defaultMeter = metrics.getMeter("default");
315
+ const cronExecutions = defaultMeter.createCounter("cron_executions", {
316
+ description: "Cron job executions",
317
+ });
318
+ const cronErrors = defaultMeter.createCounter("cron_errors", {
319
+ description: "Cron job execution errors",
320
+ });
321
+
322
+ return new Promise((resolve) => {
323
+ this.tracer.startActiveSpan(`cron:${workflow.path}`, async (span: Span) => {
324
+ scheduledJob.running = true;
325
+
326
+ try {
327
+ const start = performance.now();
328
+
329
+ // Initialize configuration for this workflow
330
+ await this.configuration.init(workflow.path, this.nodeMap);
331
+
332
+ // Create context
333
+ const ctx: Context = this.createContext(undefined, workflow.path, executionId);
334
+
335
+ // Create execution context
336
+ const cronContext: CronExecutionContext = {
337
+ jobId,
338
+ scheduledTime,
339
+ executionTime,
340
+ schedule: config.schedule,
341
+ timezone: config.timezone,
342
+ manual,
343
+ };
344
+
345
+ // Populate request with cron context
346
+ ctx.request = {
347
+ body: cronContext,
348
+ headers: {
349
+ "x-cron-job-id": jobId,
350
+ "x-cron-schedule": config.schedule,
351
+ "x-cron-timezone": config.timezone,
352
+ "x-cron-manual": String(manual),
353
+ },
354
+ query: {},
355
+ params: {
356
+ jobId,
357
+ schedule: config.schedule,
358
+ },
359
+ } as unknown as RequestContext;
360
+
361
+ // Store cron context in vars
362
+ if (!ctx.vars) ctx.vars = {};
363
+ ctx.vars._cron_context = {
364
+ jobId,
365
+ scheduledTime: scheduledTime.toISOString(),
366
+ executionTime: executionTime.toISOString(),
367
+ schedule: config.schedule,
368
+ timezone: config.timezone,
369
+ manual: String(manual),
370
+ };
371
+
372
+ ctx.logger.log(`Executing cron job: ${jobId} (${manual ? "manual" : "scheduled"})`);
373
+
374
+ // Execute workflow
375
+ const response: TriggerResponse = await this.run(ctx);
376
+ const end = performance.now();
377
+
378
+ // Update job state
379
+ scheduledJob.lastRun = executionTime;
380
+ scheduledJob.nextRun = this.toDate(scheduledJob.job.nextDate());
381
+
382
+ // Set span attributes
383
+ span.setAttribute("success", true);
384
+ span.setAttribute("job_id", jobId);
385
+ span.setAttribute("workflow_path", workflow.path);
386
+ span.setAttribute("schedule", config.schedule);
387
+ span.setAttribute("timezone", config.timezone);
388
+ span.setAttribute("manual", manual);
389
+ span.setAttribute("elapsed_ms", end - start);
390
+ span.setStatus({ code: SpanStatusCode.OK });
391
+
392
+ // Record metrics
393
+ cronExecutions.add(1, {
394
+ env: process.env.NODE_ENV,
395
+ job_id: jobId,
396
+ workflow_name: this.configuration.name,
397
+ manual: String(manual),
398
+ success: "true",
399
+ });
400
+
401
+ ctx.logger.log(`Cron job completed in ${(end - start).toFixed(2)}ms: ${jobId}`);
402
+
403
+ resolve(response);
404
+ } catch (error) {
405
+ const errorMessage = (error as Error).message;
406
+
407
+ // Set span error
408
+ span.setAttribute("success", false);
409
+ span.recordException(error as Error);
410
+ span.setStatus({ code: SpanStatusCode.ERROR, message: errorMessage });
411
+
412
+ // Record error metrics
413
+ cronErrors.add(1, {
414
+ env: process.env.NODE_ENV,
415
+ job_id: jobId,
416
+ workflow_name: this.configuration?.name || "unknown",
417
+ manual: String(manual),
418
+ });
419
+
420
+ this.logger.error(`Cron job failed ${jobId}: ${errorMessage}`, (error as Error).stack);
421
+
422
+ resolve({ ctx: {} as Context, metrics: {} as MetricsType });
423
+ } finally {
424
+ scheduledJob.running = false;
425
+ span.end();
426
+ }
427
+ });
428
+ });
429
+ }
430
+ }
431
+
432
+ export default CronTrigger;
package/src/index.ts ADDED
@@ -0,0 +1,72 @@
1
+ /**
2
+ * @blokjs/trigger-cron
3
+ *
4
+ * Cron/scheduled trigger for Blok workflows.
5
+ * Execute workflows on a schedule using cron expressions.
6
+ *
7
+ * Features:
8
+ * - Standard cron expression syntax
9
+ * - Timezone-aware scheduling
10
+ * - Overlap prevention
11
+ * - Manual trigger support
12
+ * - Job management API
13
+ *
14
+ * @example
15
+ * ```typescript
16
+ * import { CronTrigger } from "@blokjs/trigger-cron";
17
+ *
18
+ * class MyCronTrigger extends CronTrigger {
19
+ * protected nodes = myNodes;
20
+ * protected workflows = myWorkflows;
21
+ * }
22
+ *
23
+ * const trigger = new MyCronTrigger();
24
+ * await trigger.listen();
25
+ *
26
+ * // List all scheduled jobs
27
+ * const jobs = trigger.getJobs();
28
+ *
29
+ * // Manually trigger a job
30
+ * await trigger.triggerJob("cron-my-workflow-abc123");
31
+ * ```
32
+ *
33
+ * Workflow Definition:
34
+ * ```typescript
35
+ * Workflow({ name: "daily-cleanup", version: "1.0.0" })
36
+ * .addTrigger("cron", {
37
+ * schedule: "0 2 * * *", // Run at 2 AM daily
38
+ * timezone: "America/New_York",
39
+ * overlap: false,
40
+ * })
41
+ * .addStep({ ... });
42
+ * ```
43
+ *
44
+ * Cron Expression Format:
45
+ * ```
46
+ * * * * * * *
47
+ * │ │ │ │ │ │
48
+ * │ │ │ │ │ └── Day of week (0-7, Sun=0,7)
49
+ * │ │ │ │ └──── Month (1-12)
50
+ * │ │ │ └────── Day of month (1-31)
51
+ * │ │ └──────── Hour (0-23)
52
+ * │ └────────── Minute (0-59)
53
+ * └──────────── Second (0-59, optional)
54
+ * ```
55
+ *
56
+ * Common schedules:
57
+ * - "* * * * *" - Every minute
58
+ * - "0 * * * *" - Every hour
59
+ * - "0 0 * * *" - Every day at midnight
60
+ * - "0 0 * * 0" - Every Sunday at midnight
61
+ * - "0 0 1 * *" - First day of every month
62
+ */
63
+
64
+ // Core exports
65
+ export {
66
+ CronTrigger,
67
+ type ScheduledJob,
68
+ type CronExecutionContext,
69
+ } from "./CronTrigger";
70
+
71
+ // Re-export types from helper for convenience
72
+ export type { CronTriggerOpts } from "@blokjs/helper";
package/tsconfig.json ADDED
@@ -0,0 +1,32 @@
1
+ {
2
+ "ts-node": {
3
+ "transpileOnly": true
4
+ },
5
+ "compilerOptions": {
6
+ "target": "ES2022",
7
+ "module": "es2022",
8
+ "lib": ["ES2022"],
9
+ "declaration": true,
10
+ "strict": true,
11
+ "noImplicitAny": true,
12
+ "strictNullChecks": true,
13
+ "noImplicitThis": true,
14
+ "alwaysStrict": true,
15
+ "noUnusedLocals": false,
16
+ "noUnusedParameters": false,
17
+ "noImplicitReturns": true,
18
+ "noFallthroughCasesInSwitch": false,
19
+ "inlineSourceMap": true,
20
+ "inlineSources": true,
21
+ "experimentalDecorators": true,
22
+ "emitDecoratorMetadata": true,
23
+ "skipLibCheck": true,
24
+ "esModuleInterop": true,
25
+ "resolveJsonModule": true,
26
+ "outDir": "./dist",
27
+ "rootDir": "./src",
28
+ "moduleResolution": "bundler"
29
+ },
30
+ "include": ["src/**/*"],
31
+ "exclude": ["node_modules", "dist", "**/*.test.ts"]
32
+ }