@blokjs/runner 0.2.2 → 0.4.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/dist/Configuration.d.ts +18 -0
- package/dist/Configuration.js +151 -4
- package/dist/Configuration.js.map +1 -1
- package/dist/PayloadTooLargeError.d.ts +19 -0
- package/dist/PayloadTooLargeError.js +29 -0
- package/dist/PayloadTooLargeError.js.map +1 -0
- package/dist/RunCancelledError.d.ts +17 -0
- package/dist/RunCancelledError.js +25 -0
- package/dist/RunCancelledError.js.map +1 -0
- package/dist/RunnerSteps.js +330 -33
- package/dist/RunnerSteps.js.map +1 -1
- package/dist/SubworkflowNode.d.ts +75 -0
- package/dist/SubworkflowNode.js +221 -0
- package/dist/SubworkflowNode.js.map +1 -0
- package/dist/TriggerBase.d.ts +128 -0
- package/dist/TriggerBase.js +773 -4
- package/dist/TriggerBase.js.map +1 -1
- package/dist/WaitDispatchRequest.d.ts +38 -0
- package/dist/WaitDispatchRequest.js +13 -0
- package/dist/WaitDispatchRequest.js.map +1 -0
- package/dist/WaitNode.d.ts +23 -0
- package/dist/WaitNode.js +26 -0
- package/dist/WaitNode.js.map +1 -0
- package/dist/concurrency/ConcurrencyBackend.d.ts +61 -0
- package/dist/concurrency/ConcurrencyBackend.js +20 -0
- package/dist/concurrency/ConcurrencyBackend.js.map +1 -0
- package/dist/concurrency/ConcurrencyLimitError.d.ts +37 -0
- package/dist/concurrency/ConcurrencyLimitError.js +16 -0
- package/dist/concurrency/ConcurrencyLimitError.js.map +1 -0
- package/dist/concurrency/NatsKvConcurrencyBackend.d.ts +64 -0
- package/dist/concurrency/NatsKvConcurrencyBackend.js +297 -0
- package/dist/concurrency/NatsKvConcurrencyBackend.js.map +1 -0
- package/dist/concurrency/QueueExpiredError.d.ts +40 -0
- package/dist/concurrency/QueueExpiredError.js +15 -0
- package/dist/concurrency/QueueExpiredError.js.map +1 -0
- package/dist/concurrency/createConcurrencyBackend.d.ts +23 -0
- package/dist/concurrency/createConcurrencyBackend.js +34 -0
- package/dist/concurrency/createConcurrencyBackend.js.map +1 -0
- package/dist/concurrency/readConcurrencyConfig.d.ts +60 -0
- package/dist/concurrency/readConcurrencyConfig.js +60 -0
- package/dist/concurrency/readConcurrencyConfig.js.map +1 -0
- package/dist/idempotency/resolveIdempotencyKey.d.ts +20 -0
- package/dist/idempotency/resolveIdempotencyKey.js +37 -0
- package/dist/idempotency/resolveIdempotencyKey.js.map +1 -0
- package/dist/index.d.ts +23 -3
- package/dist/index.js +47 -2
- package/dist/index.js.map +1 -1
- package/dist/monitoring/ConcurrencyMetrics.d.ts +56 -0
- package/dist/monitoring/ConcurrencyMetrics.js +107 -0
- package/dist/monitoring/ConcurrencyMetrics.js.map +1 -0
- package/dist/monitoring/JanitorMetrics.d.ts +27 -0
- package/dist/monitoring/JanitorMetrics.js +48 -0
- package/dist/monitoring/JanitorMetrics.js.map +1 -0
- package/dist/scheduling/DebounceCoordinator.d.ts +88 -0
- package/dist/scheduling/DebounceCoordinator.js +141 -0
- package/dist/scheduling/DebounceCoordinator.js.map +1 -0
- package/dist/scheduling/DeferredDispatchSignal.d.ts +50 -0
- package/dist/scheduling/DeferredDispatchSignal.js +14 -0
- package/dist/scheduling/DeferredDispatchSignal.js.map +1 -0
- package/dist/scheduling/DeferredRunScheduler.d.ts +68 -0
- package/dist/scheduling/DeferredRunScheduler.js +154 -0
- package/dist/scheduling/DeferredRunScheduler.js.map +1 -0
- package/dist/scheduling/readSchedulingConfig.d.ts +24 -0
- package/dist/scheduling/readSchedulingConfig.js +52 -0
- package/dist/scheduling/readSchedulingConfig.js.map +1 -0
- package/dist/timeouts/StepTimeoutError.d.ts +22 -0
- package/dist/timeouts/StepTimeoutError.js +31 -0
- package/dist/timeouts/StepTimeoutError.js.map +1 -0
- package/dist/tracing/InMemoryRunStore.d.ts +28 -1
- package/dist/tracing/InMemoryRunStore.js +150 -0
- package/dist/tracing/InMemoryRunStore.js.map +1 -1
- package/dist/tracing/Janitor.d.ts +70 -0
- package/dist/tracing/Janitor.js +150 -0
- package/dist/tracing/Janitor.js.map +1 -0
- package/dist/tracing/PostgresRunStore.d.ts +30 -0
- package/dist/tracing/PostgresRunStore.js +435 -3
- package/dist/tracing/PostgresRunStore.js.map +1 -1
- package/dist/tracing/RunStore.d.ts +100 -1
- package/dist/tracing/RunTracker.d.ts +238 -9
- package/dist/tracing/RunTracker.js +571 -1
- package/dist/tracing/RunTracker.js.map +1 -1
- package/dist/tracing/SqliteRunStore.d.ts +23 -1
- package/dist/tracing/SqliteRunStore.js +405 -6
- package/dist/tracing/SqliteRunStore.js.map +1 -1
- package/dist/tracing/TraceRouter.d.ts +20 -2
- package/dist/tracing/TraceRouter.js +249 -5
- package/dist/tracing/TraceRouter.js.map +1 -1
- package/dist/tracing/sanitize.d.ts +11 -0
- package/dist/tracing/sanitize.js +29 -0
- package/dist/tracing/sanitize.js.map +1 -1
- package/dist/tracing/types.d.ts +348 -2
- package/dist/utils/createChildContext.d.ts +32 -0
- package/dist/utils/createChildContext.js +113 -0
- package/dist/utils/createChildContext.js.map +1 -0
- package/dist/workflow/WorkflowNormalizer.d.ts +29 -41
- package/dist/workflow/WorkflowNormalizer.js +182 -0
- package/dist/workflow/WorkflowNormalizer.js.map +1 -1
- package/dist/workflow/WorkflowRegistry.d.ts +64 -0
- package/dist/workflow/WorkflowRegistry.js +81 -0
- package/dist/workflow/WorkflowRegistry.js.map +1 -0
- package/package.json +3 -3
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tier 2 #5 + #7 — thrown by `TriggerBase.run()` when a workflow's
|
|
3
|
+
* scheduling gates (debounce / delay) defer the run to a future
|
|
4
|
+
* dispatch. Tier 2 #6 follow-up extends this to cover `onLimit:"queue"`
|
|
5
|
+
* (concurrency-gate-deferred runs); same transport translation.
|
|
6
|
+
*
|
|
7
|
+
* NOT a true error — a control-flow exception. Trigger transports
|
|
8
|
+
* catch it and translate it into the appropriate transport-level
|
|
9
|
+
* response:
|
|
10
|
+
*
|
|
11
|
+
* - HTTP trigger → `202 Accepted` with `Location: /__blok/runs/:id`
|
|
12
|
+
* header and structured JSON body.
|
|
13
|
+
* - Worker trigger → ACK without retry (the deferred coordinator owns
|
|
14
|
+
* the eventual dispatch).
|
|
15
|
+
*
|
|
16
|
+
* Carries the run-detail summary the transport needs to populate the
|
|
17
|
+
* response.
|
|
18
|
+
*/
|
|
19
|
+
export interface DeferredDispatchInfo {
|
|
20
|
+
/** Run id allocated by the tracer when the run was created. */
|
|
21
|
+
runId: string;
|
|
22
|
+
/** Workflow name. */
|
|
23
|
+
workflowName: string;
|
|
24
|
+
/**
|
|
25
|
+
* Resolved status the run was placed in:
|
|
26
|
+
* - `"delayed"` — Tier 2 #5 (`trigger.delay`).
|
|
27
|
+
* - `"debounced"` — Tier 2 #7 (`trigger.debounce`).
|
|
28
|
+
* - `"queued"` — Tier 2 #6 follow-up (`trigger.onLimit:"queue"`).
|
|
29
|
+
*/
|
|
30
|
+
status: "delayed" | "debounced" | "queued";
|
|
31
|
+
/** ms since epoch when the run is scheduled to dispatch. */
|
|
32
|
+
scheduledAt: number;
|
|
33
|
+
/** ms since epoch when the run will expire if not dispatched. Undefined when no TTL. */
|
|
34
|
+
expiresAt?: number;
|
|
35
|
+
/** True when the run was placed in `"debounced"` status (Tier 2 #7). */
|
|
36
|
+
debounced: boolean;
|
|
37
|
+
/** Pings absorbed by the run so far (always 1+). */
|
|
38
|
+
pingCount: number;
|
|
39
|
+
/**
|
|
40
|
+
* For leading-mode debounce, the runId of the sibling that fired
|
|
41
|
+
* immediately. Lets transports return `Location: /__blok/runs/<sibling>`
|
|
42
|
+
* so the caller can poll the actually-running run.
|
|
43
|
+
*/
|
|
44
|
+
intoRunId?: string;
|
|
45
|
+
}
|
|
46
|
+
export declare class DeferredDispatchSignal extends Error {
|
|
47
|
+
readonly info: DeferredDispatchInfo;
|
|
48
|
+
constructor(info: DeferredDispatchInfo);
|
|
49
|
+
}
|
|
50
|
+
export declare function isDeferredDispatchSignal(err: unknown): err is DeferredDispatchSignal;
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export class DeferredDispatchSignal extends Error {
|
|
2
|
+
info;
|
|
3
|
+
constructor(info) {
|
|
4
|
+
super(`Run ${info.runId} for workflow '${info.workflowName}' was deferred ` +
|
|
5
|
+
`(${info.status}; scheduledAt=${info.scheduledAt}; pingCount=${info.pingCount}).`);
|
|
6
|
+
this.name = "DeferredDispatchSignal";
|
|
7
|
+
this.info = info;
|
|
8
|
+
Object.setPrototypeOf(this, DeferredDispatchSignal.prototype);
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
export function isDeferredDispatchSignal(err) {
|
|
12
|
+
return err instanceof DeferredDispatchSignal;
|
|
13
|
+
}
|
|
14
|
+
//# sourceMappingURL=DeferredDispatchSignal.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DeferredDispatchSignal.js","sourceRoot":"","sources":["../../src/scheduling/DeferredDispatchSignal.ts"],"names":[],"mappings":"AA8CA,MAAM,OAAO,sBAAuB,SAAQ,KAAK;IAChC,IAAI,CAAuB;IAE3C,YAAY,IAA0B;QACrC,KAAK,CACJ,OAAO,IAAI,CAAC,KAAK,kBAAkB,IAAI,CAAC,YAAY,iBAAiB;YACpE,IAAI,IAAI,CAAC,MAAM,iBAAiB,IAAI,CAAC,WAAW,eAAe,IAAI,CAAC,SAAS,IAAI,CAClF,CAAC;QACF,IAAI,CAAC,IAAI,GAAG,wBAAwB,CAAC;QACrC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,MAAM,CAAC,cAAc,CAAC,IAAI,EAAE,sBAAsB,CAAC,SAAS,CAAC,CAAC;IAC/D,CAAC;CACD;AAED,MAAM,UAAU,wBAAwB,CAAC,GAAY;IACpD,OAAO,GAAG,YAAY,sBAAsB,CAAC;AAC9C,CAAC"}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import type { ScheduledDispatchRow } from "../tracing/types";
|
|
2
|
+
export type DeferredDispatchFn = () => Promise<void>;
|
|
3
|
+
/**
|
|
4
|
+
* Optional persistence payload — when supplied to `schedule()`, the
|
|
5
|
+
* scheduler writes a `scheduled_dispatches` row before registering the
|
|
6
|
+
* timer, and deletes it on cancel or fire. Trigger boot recovery
|
|
7
|
+
* (e.g. `HttpTrigger.recoverDispatches`) reads these rows to re-register
|
|
8
|
+
* timers across process restarts.
|
|
9
|
+
*/
|
|
10
|
+
export interface DeferredScheduleOptions {
|
|
11
|
+
workflowName: string;
|
|
12
|
+
/** Trigger type — `"http"` for v1; future triggers can opt in. */
|
|
13
|
+
triggerType: string;
|
|
14
|
+
/** TTL deadline (ms since epoch). When set, expired rows get marked `expired` on boot recovery. */
|
|
15
|
+
expiresAt?: number;
|
|
16
|
+
/** Mirrors the run record's status. */
|
|
17
|
+
dispatchStatus: ScheduledDispatchRow["dispatchStatus"];
|
|
18
|
+
/**
|
|
19
|
+
* JSON-serializable trigger-defined payload sufficient to reconstruct
|
|
20
|
+
* dispatch on boot. Trigger packages choose what to put here.
|
|
21
|
+
*/
|
|
22
|
+
payload: unknown;
|
|
23
|
+
}
|
|
24
|
+
export declare class DeferredRunScheduler {
|
|
25
|
+
private static instance;
|
|
26
|
+
private entries;
|
|
27
|
+
static getInstance(): DeferredRunScheduler;
|
|
28
|
+
/** Test-only — reset the singleton. Cancels all pending timers. */
|
|
29
|
+
static resetInstance(): void;
|
|
30
|
+
/**
|
|
31
|
+
* Register a deferred dispatch. The timer fires at `dispatchAt`
|
|
32
|
+
* (clamped to ≥ now); when it fires, the entry is removed from the
|
|
33
|
+
* map and `dispatchFn` is invoked. Errors thrown by `dispatchFn` are
|
|
34
|
+
* swallowed and logged — the scheduler is fire-and-forget by design.
|
|
35
|
+
*
|
|
36
|
+
* Re-scheduling the same `runId` cancels the previous timer and
|
|
37
|
+
* replaces it (used by the debounce coordinator's "reset on ping").
|
|
38
|
+
*
|
|
39
|
+
* When `persist` is provided, the scheduler also writes a
|
|
40
|
+
* `scheduled_dispatches` row before registering the timer (so a
|
|
41
|
+
* crash leaves the dispatch recoverable), and deletes the row on
|
|
42
|
+
* cancel or fire.
|
|
43
|
+
*/
|
|
44
|
+
schedule(runId: string, dispatchAt: number, dispatchFn: DeferredDispatchFn, persist?: DeferredScheduleOptions): void;
|
|
45
|
+
/**
|
|
46
|
+
* Cancel a pending dispatch. Returns true if the entry existed and
|
|
47
|
+
* was cancelled; false otherwise. Idempotent. When the entry was
|
|
48
|
+
* persisted, also deletes the `scheduled_dispatches` row.
|
|
49
|
+
*
|
|
50
|
+
* `cancelPersistedOnly` (default false) lets callers force the
|
|
51
|
+
* persistence-row delete even when the in-memory timer is gone (e.g.
|
|
52
|
+
* recovery cleanup that knows about a row but never had a timer).
|
|
53
|
+
*/
|
|
54
|
+
cancel(runId: string, cancelPersistedOnly?: boolean): boolean;
|
|
55
|
+
private deletePersistedRow;
|
|
56
|
+
/** True if `runId` has a pending timer. */
|
|
57
|
+
has(runId: string): boolean;
|
|
58
|
+
/** Number of pending timers. Useful for tests + observability. */
|
|
59
|
+
size(): number;
|
|
60
|
+
/**
|
|
61
|
+
* Fire ALL pending dispatches immediately and clear the queue.
|
|
62
|
+
* Awaits each `dispatchFn` so the caller knows when the queue is
|
|
63
|
+
* drained. Useful for graceful shutdown.
|
|
64
|
+
*/
|
|
65
|
+
drainAll(): Promise<void>;
|
|
66
|
+
/** Cancel everything without dispatching. */
|
|
67
|
+
clear(): void;
|
|
68
|
+
}
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tier 2 #5 — in-memory scheduler for deferred workflow runs. Tier 2
|
|
3
|
+
* #5+#7 follow-up adds optional sqlite-backed durability via the
|
|
4
|
+
* {@link DeferredScheduleOptions.persist} parameter.
|
|
5
|
+
*
|
|
6
|
+
* Process-wide singleton; obtained via {@link DeferredRunScheduler.getInstance}.
|
|
7
|
+
* Reset between tests via {@link DeferredRunScheduler.resetInstance}.
|
|
8
|
+
*/
|
|
9
|
+
import { RunTracker } from "../tracing/RunTracker";
|
|
10
|
+
export class DeferredRunScheduler {
|
|
11
|
+
static instance = null;
|
|
12
|
+
entries = new Map();
|
|
13
|
+
static getInstance() {
|
|
14
|
+
if (!DeferredRunScheduler.instance) {
|
|
15
|
+
DeferredRunScheduler.instance = new DeferredRunScheduler();
|
|
16
|
+
}
|
|
17
|
+
return DeferredRunScheduler.instance;
|
|
18
|
+
}
|
|
19
|
+
/** Test-only — reset the singleton. Cancels all pending timers. */
|
|
20
|
+
static resetInstance() {
|
|
21
|
+
DeferredRunScheduler.instance?.clear();
|
|
22
|
+
DeferredRunScheduler.instance = null;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Register a deferred dispatch. The timer fires at `dispatchAt`
|
|
26
|
+
* (clamped to ≥ now); when it fires, the entry is removed from the
|
|
27
|
+
* map and `dispatchFn` is invoked. Errors thrown by `dispatchFn` are
|
|
28
|
+
* swallowed and logged — the scheduler is fire-and-forget by design.
|
|
29
|
+
*
|
|
30
|
+
* Re-scheduling the same `runId` cancels the previous timer and
|
|
31
|
+
* replaces it (used by the debounce coordinator's "reset on ping").
|
|
32
|
+
*
|
|
33
|
+
* When `persist` is provided, the scheduler also writes a
|
|
34
|
+
* `scheduled_dispatches` row before registering the timer (so a
|
|
35
|
+
* crash leaves the dispatch recoverable), and deletes the row on
|
|
36
|
+
* cancel or fire.
|
|
37
|
+
*/
|
|
38
|
+
schedule(runId, dispatchAt, dispatchFn, persist) {
|
|
39
|
+
// Persist BEFORE the timer so a crash between persist + setTimeout
|
|
40
|
+
// still leaves the row recoverable.
|
|
41
|
+
const persisted = persist !== undefined;
|
|
42
|
+
if (persisted) {
|
|
43
|
+
const tracker = RunTracker.getInstance();
|
|
44
|
+
if (tracker.active) {
|
|
45
|
+
try {
|
|
46
|
+
tracker.getStore().upsertScheduledDispatch({
|
|
47
|
+
runId,
|
|
48
|
+
workflowName: persist.workflowName,
|
|
49
|
+
triggerType: persist.triggerType,
|
|
50
|
+
scheduledAt: dispatchAt,
|
|
51
|
+
expiresAt: persist.expiresAt,
|
|
52
|
+
dispatchStatus: persist.dispatchStatus,
|
|
53
|
+
payload: persist.payload,
|
|
54
|
+
createdAt: Date.now(),
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
catch (err) {
|
|
58
|
+
// Don't block the dispatch on persistence failure — log and continue.
|
|
59
|
+
console.error(`[blok][scheduling] persist failed for run ${runId}; continuing in-memory only:`, err instanceof Error ? err.stack || err.message : err);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
// Replace any existing entry for this runId.
|
|
64
|
+
const existing = this.entries.get(runId);
|
|
65
|
+
if (existing)
|
|
66
|
+
clearTimeout(existing.timer);
|
|
67
|
+
const delay = Math.max(0, dispatchAt - Date.now());
|
|
68
|
+
const timer = setTimeout(() => {
|
|
69
|
+
this.entries.delete(runId);
|
|
70
|
+
// Best-effort delete the persisted row before invoking dispatchFn —
|
|
71
|
+
// dispatch will write the run's terminal status separately.
|
|
72
|
+
if (persisted)
|
|
73
|
+
this.deletePersistedRow(runId);
|
|
74
|
+
void dispatchFn().catch((err) => {
|
|
75
|
+
console.error(`[blok][scheduling] DeferredRunScheduler dispatch failed for run ${runId}:`, err instanceof Error ? err.stack || err.message : err);
|
|
76
|
+
});
|
|
77
|
+
}, delay);
|
|
78
|
+
// `Node` will keep the event loop alive for pending timers — that's
|
|
79
|
+
// the desired behavior for delayed runs in long-running services.
|
|
80
|
+
// `unref()` would be wrong here.
|
|
81
|
+
this.entries.set(runId, { runId, dispatchAt, timer, dispatchFn, persisted });
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Cancel a pending dispatch. Returns true if the entry existed and
|
|
85
|
+
* was cancelled; false otherwise. Idempotent. When the entry was
|
|
86
|
+
* persisted, also deletes the `scheduled_dispatches` row.
|
|
87
|
+
*
|
|
88
|
+
* `cancelPersistedOnly` (default false) lets callers force the
|
|
89
|
+
* persistence-row delete even when the in-memory timer is gone (e.g.
|
|
90
|
+
* recovery cleanup that knows about a row but never had a timer).
|
|
91
|
+
*/
|
|
92
|
+
cancel(runId, cancelPersistedOnly = false) {
|
|
93
|
+
const entry = this.entries.get(runId);
|
|
94
|
+
if (!entry) {
|
|
95
|
+
if (cancelPersistedOnly) {
|
|
96
|
+
return this.deletePersistedRow(runId);
|
|
97
|
+
}
|
|
98
|
+
return false;
|
|
99
|
+
}
|
|
100
|
+
clearTimeout(entry.timer);
|
|
101
|
+
this.entries.delete(runId);
|
|
102
|
+
if (entry.persisted)
|
|
103
|
+
this.deletePersistedRow(runId);
|
|
104
|
+
return true;
|
|
105
|
+
}
|
|
106
|
+
deletePersistedRow(runId) {
|
|
107
|
+
const tracker = RunTracker.getInstance();
|
|
108
|
+
if (!tracker.active)
|
|
109
|
+
return false;
|
|
110
|
+
try {
|
|
111
|
+
return tracker.getStore().deleteScheduledDispatch(runId);
|
|
112
|
+
}
|
|
113
|
+
catch (err) {
|
|
114
|
+
console.error(`[blok][scheduling] persist-cleanup failed for run ${runId}:`, err instanceof Error ? err.stack || err.message : err);
|
|
115
|
+
return false;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
/** True if `runId` has a pending timer. */
|
|
119
|
+
has(runId) {
|
|
120
|
+
return this.entries.has(runId);
|
|
121
|
+
}
|
|
122
|
+
/** Number of pending timers. Useful for tests + observability. */
|
|
123
|
+
size() {
|
|
124
|
+
return this.entries.size;
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Fire ALL pending dispatches immediately and clear the queue.
|
|
128
|
+
* Awaits each `dispatchFn` so the caller knows when the queue is
|
|
129
|
+
* drained. Useful for graceful shutdown.
|
|
130
|
+
*/
|
|
131
|
+
async drainAll() {
|
|
132
|
+
const toFire = Array.from(this.entries.values());
|
|
133
|
+
// Cancel all timers first so we don't double-dispatch.
|
|
134
|
+
for (const entry of toFire)
|
|
135
|
+
clearTimeout(entry.timer);
|
|
136
|
+
this.entries.clear();
|
|
137
|
+
// Sequential dispatch — preserves intended order if it matters.
|
|
138
|
+
for (const entry of toFire) {
|
|
139
|
+
try {
|
|
140
|
+
await entry.dispatchFn();
|
|
141
|
+
}
|
|
142
|
+
catch (err) {
|
|
143
|
+
console.error(`[blok][scheduling] drainAll dispatch failed for run ${entry.runId}:`, err instanceof Error ? err.stack || err.message : err);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
/** Cancel everything without dispatching. */
|
|
148
|
+
clear() {
|
|
149
|
+
for (const entry of this.entries.values())
|
|
150
|
+
clearTimeout(entry.timer);
|
|
151
|
+
this.entries.clear();
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
//# sourceMappingURL=DeferredRunScheduler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"DeferredRunScheduler.js","sourceRoot":"","sources":["../../src/scheduling/DeferredRunScheduler.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
/** Parsed, normalized debounce config ready for the coordinator. */
|
|
2
|
+
export interface NormalizedDebounceConfig {
|
|
3
|
+
keyExpression: string;
|
|
4
|
+
mode: "leading" | "trailing";
|
|
5
|
+
delayMs: number;
|
|
6
|
+
maxDelayMs?: number;
|
|
7
|
+
}
|
|
8
|
+
/** Parsed, normalized scheduling config ready for the gates. */
|
|
9
|
+
export interface NormalizedSchedulingConfig {
|
|
10
|
+
/** Delay in ms; undefined when no delay configured. */
|
|
11
|
+
delayMs?: number;
|
|
12
|
+
/** TTL in ms; undefined when no TTL configured. */
|
|
13
|
+
ttlMs?: number;
|
|
14
|
+
/** Debounce config; undefined when no debounce configured. */
|
|
15
|
+
debounce?: NormalizedDebounceConfig;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Read a workflow's trigger config and return the normalized scheduling
|
|
19
|
+
* config, or null when the workflow has no scheduling gates.
|
|
20
|
+
*/
|
|
21
|
+
export declare function readSchedulingConfig(trigger: Record<string, unknown> | undefined | null): NormalizedSchedulingConfig | null;
|
|
22
|
+
export declare const SCHEDULING_DEFAULTS: {
|
|
23
|
+
readonly debounceMode: "trailing";
|
|
24
|
+
};
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tier 2 #5 + #7 — extract scheduling config from a workflow's trigger
|
|
3
|
+
* block, regardless of which trigger type owns it (HTTP, Worker, …).
|
|
4
|
+
*
|
|
5
|
+
* Returns `null` when the trigger has NO scheduling fields set. The
|
|
6
|
+
* caller (`TriggerBase.run()`) treats null as "skip the gates, run the
|
|
7
|
+
* workflow synchronously" — zero-overhead default.
|
|
8
|
+
*
|
|
9
|
+
* Upfront duration parsing converts `"1h"` → `3600000` so downstream
|
|
10
|
+
* code only deals in milliseconds. Invalid duration values are dropped
|
|
11
|
+
* (treated as if the field were unset) — this is fail-open by design,
|
|
12
|
+
* matching `idempotencyKey` semantics. Use `BLOK_MAPPER_MODE=strict`
|
|
13
|
+
* for fail-fast behavior in production (handled at the trigger config
|
|
14
|
+
* validation layer, not here).
|
|
15
|
+
*/
|
|
16
|
+
import { tryParseDuration } from "@blokjs/helper";
|
|
17
|
+
const SCHEDULING_TRIGGER_KEYS = ["http", "worker"];
|
|
18
|
+
/**
|
|
19
|
+
* Read a workflow's trigger config and return the normalized scheduling
|
|
20
|
+
* config, or null when the workflow has no scheduling gates.
|
|
21
|
+
*/
|
|
22
|
+
export function readSchedulingConfig(trigger) {
|
|
23
|
+
if (!trigger)
|
|
24
|
+
return null;
|
|
25
|
+
for (const triggerKey of SCHEDULING_TRIGGER_KEYS) {
|
|
26
|
+
const cfg = trigger[triggerKey];
|
|
27
|
+
if (!cfg)
|
|
28
|
+
continue;
|
|
29
|
+
const delayMs = cfg.delay !== undefined ? (tryParseDuration(cfg.delay) ?? undefined) : undefined;
|
|
30
|
+
const ttlMs = cfg.ttl !== undefined ? (tryParseDuration(cfg.ttl) ?? undefined) : undefined;
|
|
31
|
+
let debounce;
|
|
32
|
+
if (cfg.debounce && typeof cfg.debounce === "object") {
|
|
33
|
+
const d = cfg.debounce;
|
|
34
|
+
const keyExpression = typeof d.key === "string" ? d.key.trim() : "";
|
|
35
|
+
const dDelayMs = d.delay !== undefined ? (tryParseDuration(d.delay) ?? undefined) : undefined;
|
|
36
|
+
if (keyExpression && dDelayMs !== undefined && dDelayMs >= 0) {
|
|
37
|
+
const mode = d.mode === "leading" ? "leading" : "trailing";
|
|
38
|
+
const maxDelayMs = d.maxDelay !== undefined ? (tryParseDuration(d.maxDelay) ?? undefined) : undefined;
|
|
39
|
+
debounce = { keyExpression, mode, delayMs: dDelayMs, maxDelayMs };
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
const hasAny = delayMs !== undefined || ttlMs !== undefined || debounce !== undefined;
|
|
43
|
+
if (hasAny) {
|
|
44
|
+
return { delayMs, ttlMs, debounce };
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
return null;
|
|
48
|
+
}
|
|
49
|
+
export const SCHEDULING_DEFAULTS = {
|
|
50
|
+
debounceMode: "trailing",
|
|
51
|
+
};
|
|
52
|
+
//# sourceMappingURL=readSchedulingConfig.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"readSchedulingConfig.js","sourceRoot":"","sources":["../../src/scheduling/readSchedulingConfig.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AACH,OAAO,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAElD,MAAM,uBAAuB,GAAG,CAAC,MAAM,EAAE,QAAQ,CAAU,CAAC;AAiC5D;;;GAGG;AACH,MAAM,UAAU,oBAAoB,CACnC,OAAmD;IAEnD,IAAI,CAAC,OAAO;QAAE,OAAO,IAAI,CAAC;IAE1B,KAAK,MAAM,UAAU,IAAI,uBAAuB,EAAE,CAAC;QAClD,MAAM,GAAG,GAAG,OAAO,CAAC,UAAU,CAA8B,CAAC;QAC7D,IAAI,CAAC,GAAG;YAAE,SAAS;QAEnB,MAAM,OAAO,GAAG,GAAG,CAAC,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,CAAC,gBAAgB,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QACjG,MAAM,KAAK,GAAG,GAAG,CAAC,GAAG,KAAK,SAAS,CAAC,CAAC,CAAC,CAAC,gBAAgB,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAE3F,IAAI,QAA8C,CAAC;QACnD,IAAI,GAAG,CAAC,QAAQ,IAAI,OAAO,GAAG,CAAC,QAAQ,KAAK,QAAQ,EAAE,CAAC;YACtD,MAAM,CAAC,GAAG,GAAG,CAAC,QAAQ,CAAC;YACvB,MAAM,aAAa,GAAG,OAAO,CAAC,CAAC,GAAG,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACpE,MAAM,QAAQ,GAAG,CAAC,CAAC,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;YAC9F,IAAI,aAAa,IAAI,QAAQ,KAAK,SAAS,IAAI,QAAQ,IAAI,CAAC,EAAE,CAAC;gBAC9D,MAAM,IAAI,GAA2B,CAAC,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,CAAC;gBACnF,MAAM,UAAU,GAAG,CAAC,CAAC,QAAQ,KAAK,SAAS,CAAC,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,SAAS,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;gBACtG,QAAQ,GAAG,EAAE,aAAa,EAAE,IAAI,EAAE,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC;YACnE,CAAC;QACF,CAAC;QAED,MAAM,MAAM,GAAG,OAAO,KAAK,SAAS,IAAI,KAAK,KAAK,SAAS,IAAI,QAAQ,KAAK,SAAS,CAAC;QACtF,IAAI,MAAM,EAAE,CAAC;YACZ,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC;QACrC,CAAC;IACF,CAAC;IAED,OAAO,IAAI,CAAC;AACb,CAAC;AAED,MAAM,CAAC,MAAM,mBAAmB,GAAG;IAClC,YAAY,EAAE,UAAmB;CACxB,CAAC"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tier 2 quick-wins — thrown by `RunnerSteps` when a step's
|
|
3
|
+
* `step.process()` exceeds its `maxDuration` cap.
|
|
4
|
+
*
|
|
5
|
+
* The error participates in the existing retry loop: each retry
|
|
6
|
+
* attempt wraps its own setTimeout-based timeout, so a step that
|
|
7
|
+
* times out N times will trigger N retries (each capped at
|
|
8
|
+
* `maxDuration`) before the run flips to `"timedOut"`.
|
|
9
|
+
*
|
|
10
|
+
* Note: setTimeout-based timeout REJECTS the wrapper promise but
|
|
11
|
+
* doesn't truly abort the underlying async work — the
|
|
12
|
+
* `step.process()` continues running until natural completion.
|
|
13
|
+
* The parent runner has already moved on; the orphaned promise
|
|
14
|
+
* resolves harmlessly into the void. Proper cooperative
|
|
15
|
+
* cancellation via `AbortSignal` is a deferred follow-up.
|
|
16
|
+
*/
|
|
17
|
+
export declare class StepTimeoutError extends Error {
|
|
18
|
+
readonly stepName: string;
|
|
19
|
+
readonly maxDurationMs: number;
|
|
20
|
+
constructor(stepName: string, maxDurationMs: number);
|
|
21
|
+
}
|
|
22
|
+
export declare function isStepTimeoutError(err: unknown): err is StepTimeoutError;
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tier 2 quick-wins — thrown by `RunnerSteps` when a step's
|
|
3
|
+
* `step.process()` exceeds its `maxDuration` cap.
|
|
4
|
+
*
|
|
5
|
+
* The error participates in the existing retry loop: each retry
|
|
6
|
+
* attempt wraps its own setTimeout-based timeout, so a step that
|
|
7
|
+
* times out N times will trigger N retries (each capped at
|
|
8
|
+
* `maxDuration`) before the run flips to `"timedOut"`.
|
|
9
|
+
*
|
|
10
|
+
* Note: setTimeout-based timeout REJECTS the wrapper promise but
|
|
11
|
+
* doesn't truly abort the underlying async work — the
|
|
12
|
+
* `step.process()` continues running until natural completion.
|
|
13
|
+
* The parent runner has already moved on; the orphaned promise
|
|
14
|
+
* resolves harmlessly into the void. Proper cooperative
|
|
15
|
+
* cancellation via `AbortSignal` is a deferred follow-up.
|
|
16
|
+
*/
|
|
17
|
+
export class StepTimeoutError extends Error {
|
|
18
|
+
stepName;
|
|
19
|
+
maxDurationMs;
|
|
20
|
+
constructor(stepName, maxDurationMs) {
|
|
21
|
+
super(`Step '${stepName}' exceeded maxDuration of ${maxDurationMs}ms`);
|
|
22
|
+
this.name = "StepTimeoutError";
|
|
23
|
+
this.stepName = stepName;
|
|
24
|
+
this.maxDurationMs = maxDurationMs;
|
|
25
|
+
Object.setPrototypeOf(this, StepTimeoutError.prototype);
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
export function isStepTimeoutError(err) {
|
|
29
|
+
return err instanceof StepTimeoutError;
|
|
30
|
+
}
|
|
31
|
+
//# sourceMappingURL=StepTimeoutError.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"StepTimeoutError.js","sourceRoot":"","sources":["../../src/timeouts/StepTimeoutError.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AACH,MAAM,OAAO,gBAAiB,SAAQ,KAAK;IAC1B,QAAQ,CAAS;IACjB,aAAa,CAAS;IAEtC,YAAY,QAAgB,EAAE,aAAqB;QAClD,KAAK,CAAC,SAAS,QAAQ,6BAA6B,aAAa,IAAI,CAAC,CAAC;QACvE,IAAI,CAAC,IAAI,GAAG,kBAAkB,CAAC;QAC/B,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACnC,MAAM,CAAC,cAAc,CAAC,IAAI,EAAE,gBAAgB,CAAC,SAAS,CAAC,CAAC;IACzD,CAAC;CACD;AAED,MAAM,UAAU,kBAAkB,CAAC,GAAY;IAC9C,OAAO,GAAG,YAAY,gBAAgB,CAAC;AACxC,CAAC"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import type { RunStore } from "./RunStore";
|
|
2
|
-
import type { Dashboard, MetricsResult, NodeRun, RunEvent, RunQuery, TraceLogEntry, WorkflowRun, WorkflowSummary } from "./types";
|
|
2
|
+
import type { CachedStepResult, ConcurrencySlotResult, Dashboard, MetricsResult, NodeRun, RunEvent, RunQuery, ScheduledDispatchRow, TraceLogEntry, WorkflowRun, WorkflowSummary } from "./types";
|
|
3
3
|
/**
|
|
4
4
|
* In-memory implementation of RunStore using Maps.
|
|
5
5
|
* Zero dependencies, fastest possible reads/writes.
|
|
@@ -12,6 +12,11 @@ export declare class InMemoryRunStore implements RunStore {
|
|
|
12
12
|
private events;
|
|
13
13
|
private logs;
|
|
14
14
|
private dashboards;
|
|
15
|
+
private idempotencyCache;
|
|
16
|
+
private concurrencyLocks;
|
|
17
|
+
private scheduledDispatches;
|
|
18
|
+
private idemKey;
|
|
19
|
+
private concurrencyBucketKey;
|
|
15
20
|
saveRun(run: WorkflowRun): void;
|
|
16
21
|
updateRun(runId: string, updates: Partial<WorkflowRun>): void;
|
|
17
22
|
saveNodeRun(nodeRun: NodeRun): void;
|
|
@@ -25,6 +30,7 @@ export declare class InMemoryRunStore implements RunStore {
|
|
|
25
30
|
};
|
|
26
31
|
getNodeRuns(runId: string): NodeRun[];
|
|
27
32
|
getNodeRun(nodeRunId: string): NodeRun | undefined;
|
|
33
|
+
getRunsByParent(parentRunId: string): WorkflowRun[];
|
|
28
34
|
getEvents(runId: string, since?: number): RunEvent[];
|
|
29
35
|
getLogs(runId: string, nodeId?: string): TraceLogEntry[];
|
|
30
36
|
getWorkflowSummaries(): WorkflowSummary[];
|
|
@@ -39,6 +45,27 @@ export declare class InMemoryRunStore implements RunStore {
|
|
|
39
45
|
clearAll(): number;
|
|
40
46
|
deleteRunsBefore(timestamp: number): number;
|
|
41
47
|
evictOldRuns(maxRuns: number): void;
|
|
48
|
+
getIdempotencyCache(workflowName: string, stepId: string, key: string): CachedStepResult | null;
|
|
49
|
+
setIdempotencyCache(workflowName: string, stepId: string, key: string, entry: CachedStepResult): void;
|
|
50
|
+
purgeExpiredIdempotencyCache(now: number): number;
|
|
51
|
+
acquireConcurrencySlot(workflowName: string, concurrencyKey: string, concurrencyLimit: number, runId: string, leaseExpiresAt: number): ConcurrencySlotResult;
|
|
52
|
+
releaseConcurrencySlot(workflowName: string, concurrencyKey: string, runId: string): void;
|
|
53
|
+
purgeExpiredConcurrencySlots(now: number): number;
|
|
54
|
+
getConcurrencySnapshot(now: number): Array<{
|
|
55
|
+
workflowName: string;
|
|
56
|
+
concurrencyKey: string;
|
|
57
|
+
leases: Array<{
|
|
58
|
+
runId: string;
|
|
59
|
+
expiresAt: number;
|
|
60
|
+
}>;
|
|
61
|
+
}>;
|
|
62
|
+
upsertScheduledDispatch(row: ScheduledDispatchRow): void;
|
|
63
|
+
deleteScheduledDispatch(runId: string): boolean;
|
|
64
|
+
getScheduledDispatches(opts?: {
|
|
65
|
+
triggerType?: string;
|
|
66
|
+
status?: string;
|
|
67
|
+
}): ScheduledDispatchRow[];
|
|
68
|
+
purgeExpiredScheduledDispatches(now: number): number;
|
|
42
69
|
close(): void;
|
|
43
70
|
private deleteRun;
|
|
44
71
|
}
|