@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.
Files changed (101) hide show
  1. package/dist/Configuration.d.ts +18 -0
  2. package/dist/Configuration.js +151 -4
  3. package/dist/Configuration.js.map +1 -1
  4. package/dist/PayloadTooLargeError.d.ts +19 -0
  5. package/dist/PayloadTooLargeError.js +29 -0
  6. package/dist/PayloadTooLargeError.js.map +1 -0
  7. package/dist/RunCancelledError.d.ts +17 -0
  8. package/dist/RunCancelledError.js +25 -0
  9. package/dist/RunCancelledError.js.map +1 -0
  10. package/dist/RunnerSteps.js +330 -33
  11. package/dist/RunnerSteps.js.map +1 -1
  12. package/dist/SubworkflowNode.d.ts +75 -0
  13. package/dist/SubworkflowNode.js +221 -0
  14. package/dist/SubworkflowNode.js.map +1 -0
  15. package/dist/TriggerBase.d.ts +128 -0
  16. package/dist/TriggerBase.js +773 -4
  17. package/dist/TriggerBase.js.map +1 -1
  18. package/dist/WaitDispatchRequest.d.ts +38 -0
  19. package/dist/WaitDispatchRequest.js +13 -0
  20. package/dist/WaitDispatchRequest.js.map +1 -0
  21. package/dist/WaitNode.d.ts +23 -0
  22. package/dist/WaitNode.js +26 -0
  23. package/dist/WaitNode.js.map +1 -0
  24. package/dist/concurrency/ConcurrencyBackend.d.ts +61 -0
  25. package/dist/concurrency/ConcurrencyBackend.js +20 -0
  26. package/dist/concurrency/ConcurrencyBackend.js.map +1 -0
  27. package/dist/concurrency/ConcurrencyLimitError.d.ts +37 -0
  28. package/dist/concurrency/ConcurrencyLimitError.js +16 -0
  29. package/dist/concurrency/ConcurrencyLimitError.js.map +1 -0
  30. package/dist/concurrency/NatsKvConcurrencyBackend.d.ts +64 -0
  31. package/dist/concurrency/NatsKvConcurrencyBackend.js +297 -0
  32. package/dist/concurrency/NatsKvConcurrencyBackend.js.map +1 -0
  33. package/dist/concurrency/QueueExpiredError.d.ts +40 -0
  34. package/dist/concurrency/QueueExpiredError.js +15 -0
  35. package/dist/concurrency/QueueExpiredError.js.map +1 -0
  36. package/dist/concurrency/createConcurrencyBackend.d.ts +23 -0
  37. package/dist/concurrency/createConcurrencyBackend.js +34 -0
  38. package/dist/concurrency/createConcurrencyBackend.js.map +1 -0
  39. package/dist/concurrency/readConcurrencyConfig.d.ts +60 -0
  40. package/dist/concurrency/readConcurrencyConfig.js +60 -0
  41. package/dist/concurrency/readConcurrencyConfig.js.map +1 -0
  42. package/dist/idempotency/resolveIdempotencyKey.d.ts +20 -0
  43. package/dist/idempotency/resolveIdempotencyKey.js +37 -0
  44. package/dist/idempotency/resolveIdempotencyKey.js.map +1 -0
  45. package/dist/index.d.ts +23 -3
  46. package/dist/index.js +47 -2
  47. package/dist/index.js.map +1 -1
  48. package/dist/monitoring/ConcurrencyMetrics.d.ts +56 -0
  49. package/dist/monitoring/ConcurrencyMetrics.js +107 -0
  50. package/dist/monitoring/ConcurrencyMetrics.js.map +1 -0
  51. package/dist/monitoring/JanitorMetrics.d.ts +27 -0
  52. package/dist/monitoring/JanitorMetrics.js +48 -0
  53. package/dist/monitoring/JanitorMetrics.js.map +1 -0
  54. package/dist/scheduling/DebounceCoordinator.d.ts +88 -0
  55. package/dist/scheduling/DebounceCoordinator.js +141 -0
  56. package/dist/scheduling/DebounceCoordinator.js.map +1 -0
  57. package/dist/scheduling/DeferredDispatchSignal.d.ts +50 -0
  58. package/dist/scheduling/DeferredDispatchSignal.js +14 -0
  59. package/dist/scheduling/DeferredDispatchSignal.js.map +1 -0
  60. package/dist/scheduling/DeferredRunScheduler.d.ts +68 -0
  61. package/dist/scheduling/DeferredRunScheduler.js +154 -0
  62. package/dist/scheduling/DeferredRunScheduler.js.map +1 -0
  63. package/dist/scheduling/readSchedulingConfig.d.ts +24 -0
  64. package/dist/scheduling/readSchedulingConfig.js +52 -0
  65. package/dist/scheduling/readSchedulingConfig.js.map +1 -0
  66. package/dist/timeouts/StepTimeoutError.d.ts +22 -0
  67. package/dist/timeouts/StepTimeoutError.js +31 -0
  68. package/dist/timeouts/StepTimeoutError.js.map +1 -0
  69. package/dist/tracing/InMemoryRunStore.d.ts +28 -1
  70. package/dist/tracing/InMemoryRunStore.js +150 -0
  71. package/dist/tracing/InMemoryRunStore.js.map +1 -1
  72. package/dist/tracing/Janitor.d.ts +70 -0
  73. package/dist/tracing/Janitor.js +150 -0
  74. package/dist/tracing/Janitor.js.map +1 -0
  75. package/dist/tracing/PostgresRunStore.d.ts +30 -0
  76. package/dist/tracing/PostgresRunStore.js +435 -3
  77. package/dist/tracing/PostgresRunStore.js.map +1 -1
  78. package/dist/tracing/RunStore.d.ts +100 -1
  79. package/dist/tracing/RunTracker.d.ts +238 -9
  80. package/dist/tracing/RunTracker.js +571 -1
  81. package/dist/tracing/RunTracker.js.map +1 -1
  82. package/dist/tracing/SqliteRunStore.d.ts +23 -1
  83. package/dist/tracing/SqliteRunStore.js +405 -6
  84. package/dist/tracing/SqliteRunStore.js.map +1 -1
  85. package/dist/tracing/TraceRouter.d.ts +20 -2
  86. package/dist/tracing/TraceRouter.js +249 -5
  87. package/dist/tracing/TraceRouter.js.map +1 -1
  88. package/dist/tracing/sanitize.d.ts +11 -0
  89. package/dist/tracing/sanitize.js +29 -0
  90. package/dist/tracing/sanitize.js.map +1 -1
  91. package/dist/tracing/types.d.ts +348 -2
  92. package/dist/utils/createChildContext.d.ts +32 -0
  93. package/dist/utils/createChildContext.js +113 -0
  94. package/dist/utils/createChildContext.js.map +1 -0
  95. package/dist/workflow/WorkflowNormalizer.d.ts +29 -41
  96. package/dist/workflow/WorkflowNormalizer.js +182 -0
  97. package/dist/workflow/WorkflowNormalizer.js.map +1 -1
  98. package/dist/workflow/WorkflowRegistry.d.ts +64 -0
  99. package/dist/workflow/WorkflowRegistry.js +81 -0
  100. package/dist/workflow/WorkflowRegistry.js.map +1 -0
  101. 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
  }