footprintjs 4.2.0 → 4.3.1

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 (73) hide show
  1. package/CLAUDE.md +43 -1
  2. package/dist/esm/index.js +1 -1
  3. package/dist/esm/lib/builder/FlowChartBuilder.js +53 -1
  4. package/dist/esm/lib/builder/types.js +1 -1
  5. package/dist/esm/lib/engine/graph/StageNode.js +1 -1
  6. package/dist/esm/lib/engine/handlers/ChildrenExecutor.js +14 -1
  7. package/dist/esm/lib/engine/handlers/DeciderHandler.js +7 -1
  8. package/dist/esm/lib/engine/handlers/SelectorHandler.js +7 -1
  9. package/dist/esm/lib/engine/handlers/StageRunner.js +13 -1
  10. package/dist/esm/lib/engine/handlers/SubflowExecutor.js +8 -1
  11. package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +39 -1
  12. package/dist/esm/lib/engine/narrative/FlowRecorderDispatcher.js +31 -1
  13. package/dist/esm/lib/engine/narrative/NarrativeFlowRecorder.js +10 -1
  14. package/dist/esm/lib/engine/narrative/NullControlFlowNarrativeGenerator.js +3 -1
  15. package/dist/esm/lib/engine/narrative/narrativeTypes.js +1 -1
  16. package/dist/esm/lib/engine/narrative/types.js +1 -1
  17. package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +8 -1
  18. package/dist/esm/lib/engine/types.js +2 -1
  19. package/dist/esm/lib/pause/index.js +2 -0
  20. package/dist/esm/lib/pause/types.js +58 -0
  21. package/dist/esm/lib/reactive/types.js +2 -1
  22. package/dist/esm/lib/runner/ExecutionRuntime.js +11 -2
  23. package/dist/esm/lib/runner/FlowChartExecutor.js +232 -8
  24. package/dist/esm/lib/scope/ScopeFacade.js +21 -1
  25. package/dist/esm/lib/scope/recorders/DebugRecorder.js +19 -1
  26. package/dist/esm/lib/scope/recorders/MetricRecorder.js +10 -1
  27. package/dist/esm/lib/scope/types.js +1 -1
  28. package/dist/index.js +1 -1
  29. package/dist/lib/builder/FlowChartBuilder.js +53 -1
  30. package/dist/lib/builder/types.js +1 -1
  31. package/dist/lib/engine/graph/StageNode.js +1 -1
  32. package/dist/lib/engine/handlers/ChildrenExecutor.js +14 -1
  33. package/dist/lib/engine/handlers/DeciderHandler.js +7 -1
  34. package/dist/lib/engine/handlers/SelectorHandler.js +7 -1
  35. package/dist/lib/engine/handlers/StageRunner.js +17 -5
  36. package/dist/lib/engine/handlers/SubflowExecutor.js +8 -1
  37. package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +39 -1
  38. package/dist/lib/engine/narrative/FlowRecorderDispatcher.js +31 -1
  39. package/dist/lib/engine/narrative/NarrativeFlowRecorder.js +10 -1
  40. package/dist/lib/engine/narrative/NullControlFlowNarrativeGenerator.js +3 -1
  41. package/dist/lib/engine/narrative/narrativeTypes.js +1 -1
  42. package/dist/lib/engine/narrative/types.js +1 -1
  43. package/dist/lib/engine/traversal/FlowchartTraverser.js +8 -1
  44. package/dist/lib/engine/types.js +6 -2
  45. package/dist/lib/pause/index.js +8 -0
  46. package/dist/lib/pause/types.js +64 -0
  47. package/dist/lib/reactive/types.js +2 -1
  48. package/dist/lib/runner/ExecutionRuntime.js +11 -2
  49. package/dist/lib/runner/FlowChartExecutor.js +232 -8
  50. package/dist/lib/scope/ScopeFacade.js +21 -1
  51. package/dist/lib/scope/recorders/DebugRecorder.js +19 -1
  52. package/dist/lib/scope/recorders/MetricRecorder.js +10 -1
  53. package/dist/lib/scope/types.js +1 -1
  54. package/dist/types/index.d.ts +4 -1
  55. package/dist/types/lib/builder/FlowChartBuilder.d.ts +22 -0
  56. package/dist/types/lib/builder/types.d.ts +3 -1
  57. package/dist/types/lib/engine/graph/StageNode.d.ts +9 -0
  58. package/dist/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +9 -1
  59. package/dist/types/lib/engine/narrative/FlowRecorderDispatcher.d.ts +2 -0
  60. package/dist/types/lib/engine/narrative/NarrativeFlowRecorder.d.ts +3 -1
  61. package/dist/types/lib/engine/narrative/NullControlFlowNarrativeGenerator.d.ts +2 -0
  62. package/dist/types/lib/engine/narrative/narrativeTypes.d.ts +1 -1
  63. package/dist/types/lib/engine/narrative/types.d.ts +24 -0
  64. package/dist/types/lib/engine/types.d.ts +11 -0
  65. package/dist/types/lib/pause/index.d.ts +2 -0
  66. package/dist/types/lib/pause/types.d.ts +175 -0
  67. package/dist/types/lib/runner/ExecutionRuntime.d.ts +4 -0
  68. package/dist/types/lib/runner/FlowChartExecutor.d.ts +54 -2
  69. package/dist/types/lib/scope/ScopeFacade.d.ts +4 -0
  70. package/dist/types/lib/scope/recorders/DebugRecorder.d.ts +4 -2
  71. package/dist/types/lib/scope/recorders/MetricRecorder.d.ts +4 -1
  72. package/dist/types/lib/scope/types.d.ts +10 -0
  73. package/package.json +1 -1
@@ -21,6 +21,7 @@ import { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder
21
21
  import { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';
22
22
  import { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';
23
23
  import { defaultLogger, } from '../engine/types.js';
24
+ import { isPauseSignal } from '../pause/types.js';
24
25
  import { ScopeFacade } from '../scope/ScopeFacade.js';
25
26
  import { ExecutionRuntime } from './ExecutionRuntime.js';
26
27
  import { validateInput } from './validateInput.js';
@@ -88,8 +89,8 @@ export class FlowChartExecutor {
88
89
  };
89
90
  this.traverser = this.createTraverser();
90
91
  }
91
- createTraverser(signal, readOnlyContextOverride, env, maxDepth) {
92
- var _a, _b, _c;
92
+ createTraverser(signal, readOnlyContextOverride, env, maxDepth, overrides) {
93
+ var _a, _b, _c, _d, _e;
93
94
  const args = this.flowChartArgs;
94
95
  const fc = args.flowChart;
95
96
  const narrativeFlag = this.narrativeEnabled || ((_a = fc.enableNarrative) !== null && _a !== void 0 ? _a : false);
@@ -97,7 +98,10 @@ export class FlowChartExecutor {
97
98
  // Collect all scope modifiers (recorders, redaction) into a single list,
98
99
  // then create ONE factory that applies them in a loop. Replaces the
99
100
  // previous 4-deep closure nesting with a flat, debuggable composition.
100
- if (narrativeFlag) {
101
+ if (overrides === null || overrides === void 0 ? void 0 : overrides.preserveRecorders) {
102
+ // Resume mode: keep existing combinedRecorder so narrative accumulates
103
+ }
104
+ else if (narrativeFlag) {
101
105
  this.combinedRecorder = new CombinedNarrativeRecorder(this.narrativeOptions);
102
106
  }
103
107
  else {
@@ -157,9 +161,25 @@ export class FlowChartExecutor {
157
161
  mod(scope);
158
162
  return scope;
159
163
  });
160
- const runtime = new ExecutionRuntime(fc.root.name, fc.root.id, args.defaultValuesForContext, args.initialContext);
164
+ const effectiveRoot = (_b = overrides === null || overrides === void 0 ? void 0 : overrides.root) !== null && _b !== void 0 ? _b : fc.root;
165
+ const effectiveInitialContext = (_c = overrides === null || overrides === void 0 ? void 0 : overrides.initialContext) !== null && _c !== void 0 ? _c : args.initialContext;
166
+ let runtime;
167
+ if (overrides === null || overrides === void 0 ? void 0 : overrides.existingRuntime) {
168
+ // Resume mode: reuse existing runtime so execution tree continues from pause point.
169
+ // Preserve the original root for getSnapshot() (full tree), then advance
170
+ // rootStageContext to a continuation from the leaf (for traversal).
171
+ runtime = overrides.existingRuntime;
172
+ runtime.preserveSnapshotRoot();
173
+ let leaf = runtime.rootStageContext;
174
+ while (leaf.next)
175
+ leaf = leaf.next;
176
+ runtime.rootStageContext = leaf.createNext('', effectiveRoot.name, effectiveRoot.id);
177
+ }
178
+ else {
179
+ runtime = new ExecutionRuntime(effectiveRoot.name, effectiveRoot.id, args.defaultValuesForContext, effectiveInitialContext);
180
+ }
161
181
  return new FlowchartTraverser({
162
- root: fc.root,
182
+ root: effectiveRoot,
163
183
  stageMap: fc.stageMap,
164
184
  scopeFactory,
165
185
  executionRuntime: runtime,
@@ -169,10 +189,10 @@ export class FlowChartExecutor {
169
189
  extractor: fc.extractor,
170
190
  scopeProtectionMode: args.scopeProtectionMode,
171
191
  subflows: fc.subflows,
172
- enrichSnapshots: (_b = args.enrichSnapshots) !== null && _b !== void 0 ? _b : fc.enrichSnapshots,
192
+ enrichSnapshots: (_d = args.enrichSnapshots) !== null && _d !== void 0 ? _d : fc.enrichSnapshots,
173
193
  narrativeEnabled: narrativeFlag,
174
194
  buildTimeStructure: fc.buildTimeStructure,
175
- logger: (_c = fc.logger) !== null && _c !== void 0 ? _c : defaultLogger,
195
+ logger: (_e = fc.logger) !== null && _e !== void 0 ? _e : defaultLogger,
176
196
  signal,
177
197
  executionEnv: env,
178
198
  flowRecorders: this.buildFlowRecordersList(),
@@ -207,6 +227,190 @@ export class FlowChartExecutor {
207
227
  patterns: ((_b = (_a = this.redactionPolicy) === null || _a === void 0 ? void 0 : _a.patterns) !== null && _b !== void 0 ? _b : []).map((p) => p.source),
208
228
  };
209
229
  }
230
+ // ─── Pause/Resume ───
231
+ /**
232
+ * Returns the checkpoint from the most recent paused execution, or `undefined`
233
+ * if the last run completed without pausing.
234
+ *
235
+ * The checkpoint is JSON-serializable — store it in Redis, Postgres, localStorage, etc.
236
+ *
237
+ * @example
238
+ * ```typescript
239
+ * const result = await executor.run({ input });
240
+ * if (executor.isPaused()) {
241
+ * const checkpoint = executor.getCheckpoint()!;
242
+ * await redis.set(`session:${id}`, JSON.stringify(checkpoint));
243
+ * }
244
+ * ```
245
+ */
246
+ getCheckpoint() {
247
+ return this.lastCheckpoint;
248
+ }
249
+ /** Returns `true` if the most recent run() was paused (checkpoint available). */
250
+ isPaused() {
251
+ return this.lastCheckpoint !== undefined;
252
+ }
253
+ /**
254
+ * Resume a paused flowchart from a checkpoint.
255
+ *
256
+ * Restores the scope state, calls the paused stage's `resumeFn` with the
257
+ * provided input, then continues traversal from the next stage.
258
+ *
259
+ * The checkpoint can come from `getCheckpoint()` on a previous run, or from
260
+ * a serialized checkpoint stored in Redis/Postgres/localStorage.
261
+ *
262
+ * **Narrative/recorder state is reset on resume.** To keep a unified narrative
263
+ * across pause/resume cycles, collect it before calling resume.
264
+ *
265
+ * @example
266
+ * ```typescript
267
+ * // After a pause...
268
+ * const checkpoint = executor.getCheckpoint()!;
269
+ * await redis.set(`session:${id}`, JSON.stringify(checkpoint));
270
+ *
271
+ * // Later (possibly different server, same chart)
272
+ * const checkpoint = JSON.parse(await redis.get(`session:${id}`));
273
+ * const executor = new FlowChartExecutor(chart);
274
+ * const result = await executor.resume(checkpoint, { approved: true });
275
+ * ```
276
+ */
277
+ async resume(checkpoint, resumeInput, options) {
278
+ var _a, _b;
279
+ this.lastCheckpoint = undefined;
280
+ // ── Validate checkpoint structure (may come from untrusted external storage) ──
281
+ if (!checkpoint ||
282
+ typeof checkpoint !== 'object' ||
283
+ typeof checkpoint.sharedState !== 'object' ||
284
+ checkpoint.sharedState === null ||
285
+ Array.isArray(checkpoint.sharedState)) {
286
+ throw new Error('Invalid checkpoint: sharedState must be a plain object.');
287
+ }
288
+ if (typeof checkpoint.pausedStageId !== 'string' || checkpoint.pausedStageId === '') {
289
+ throw new Error('Invalid checkpoint: pausedStageId must be a non-empty string.');
290
+ }
291
+ if (!Array.isArray(checkpoint.subflowPath) ||
292
+ !checkpoint.subflowPath.every((s) => typeof s === 'string')) {
293
+ throw new Error('Invalid checkpoint: subflowPath must be an array of strings.');
294
+ }
295
+ // Find the paused node in the graph
296
+ const pausedNode = this.findNodeInGraph(checkpoint.pausedStageId, checkpoint.subflowPath);
297
+ if (!pausedNode) {
298
+ throw new Error(`Cannot resume: stage '${checkpoint.pausedStageId}' not found in flowchart. ` +
299
+ 'The chart may have changed since the checkpoint was created.');
300
+ }
301
+ if (!pausedNode.resumeFn) {
302
+ throw new Error(`Cannot resume: stage '${pausedNode.name}' (${pausedNode.id}) has no resumeFn. ` +
303
+ 'Only stages created with addPausableFunction() can be resumed.');
304
+ }
305
+ // Build a synthetic resume node: calls resumeFn with resumeInput, then continues to original next.
306
+ // resumeFn signature is (scope, input) per PausableHandler — wrap to match StageFunction(scope, breakFn).
307
+ const resumeFn = pausedNode.resumeFn;
308
+ const resumeStageFn = (scope) => {
309
+ return resumeFn(scope, resumeInput);
310
+ };
311
+ const resumeNode = {
312
+ name: pausedNode.name,
313
+ id: pausedNode.id,
314
+ description: pausedNode.description,
315
+ fn: resumeStageFn,
316
+ next: pausedNode.next,
317
+ };
318
+ // Don't clear recorders — resume continues from previous state.
319
+ // Narrative, metrics, debug entries accumulate across pause/resume.
320
+ // Reuse the existing runtime so the execution tree continues from the pause point.
321
+ // preserveRecorders keeps the CombinedNarrativeRecorder so narrative accumulates.
322
+ const existingRuntime = this.traverser.getRuntime();
323
+ this.traverser = this.createTraverser(options === null || options === void 0 ? void 0 : options.signal, undefined, options === null || options === void 0 ? void 0 : options.env, options === null || options === void 0 ? void 0 : options.maxDepth, {
324
+ root: resumeNode,
325
+ initialContext: checkpoint.sharedState,
326
+ preserveRecorders: true,
327
+ existingRuntime,
328
+ });
329
+ // Fire onResume event on all recorders (flow + scope)
330
+ const hasInput = resumeInput !== undefined;
331
+ const flowResumeEvent = {
332
+ stageName: pausedNode.name,
333
+ stageId: pausedNode.id,
334
+ hasInput,
335
+ };
336
+ if (this.combinedRecorder)
337
+ this.combinedRecorder.onResume(flowResumeEvent);
338
+ for (const r of this.flowRecorders)
339
+ (_a = r.onResume) === null || _a === void 0 ? void 0 : _a.call(r, flowResumeEvent);
340
+ const scopeResumeEvent = {
341
+ stageName: pausedNode.name,
342
+ stageId: pausedNode.id,
343
+ hasInput,
344
+ pipelineId: '',
345
+ timestamp: Date.now(),
346
+ };
347
+ for (const r of this.scopeRecorders)
348
+ (_b = r.onResume) === null || _b === void 0 ? void 0 : _b.call(r, scopeResumeEvent);
349
+ try {
350
+ return await this.traverser.execute();
351
+ }
352
+ catch (error) {
353
+ if (isPauseSignal(error)) {
354
+ const snapshot = this.traverser.getSnapshot();
355
+ const sfResults = this.traverser.getSubflowResults();
356
+ this.lastCheckpoint = {
357
+ sharedState: snapshot.sharedState,
358
+ executionTree: snapshot.executionTree,
359
+ pausedStageId: error.stageId,
360
+ subflowPath: error.subflowPath,
361
+ pauseData: error.pauseData,
362
+ ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),
363
+ pausedAt: Date.now(),
364
+ };
365
+ return { paused: true, checkpoint: this.lastCheckpoint };
366
+ }
367
+ throw error;
368
+ }
369
+ }
370
+ /**
371
+ * Find a StageNode in the compiled graph by ID.
372
+ * Handles subflow paths by drilling into registered subflows.
373
+ */
374
+ findNodeInGraph(stageId, subflowPath) {
375
+ var _a;
376
+ const fc = this.flowChartArgs.flowChart;
377
+ if (subflowPath.length === 0) {
378
+ // Top-level: DFS from root
379
+ return this.dfsFind(fc.root, stageId);
380
+ }
381
+ // Subflow: drill into the subflow chain, then search from the last subflow's root
382
+ let subflowRoot;
383
+ for (const sfId of subflowPath) {
384
+ const subflow = (_a = fc.subflows) === null || _a === void 0 ? void 0 : _a[sfId];
385
+ if (!subflow)
386
+ return undefined;
387
+ subflowRoot = subflow.root;
388
+ }
389
+ if (!subflowRoot)
390
+ return undefined;
391
+ return this.dfsFind(subflowRoot, stageId);
392
+ }
393
+ /** DFS search for a node by ID in the StageNode graph. Cycle-safe via visited set. */
394
+ dfsFind(node, targetId, visited = new Set()) {
395
+ // Skip loop back-edge references (they share the target's ID but have no fn/resumeFn)
396
+ if (node.isLoopRef)
397
+ return undefined;
398
+ if (visited.has(node.id))
399
+ return undefined;
400
+ visited.add(node.id);
401
+ if (node.id === targetId)
402
+ return node;
403
+ if (node.children) {
404
+ for (const child of node.children) {
405
+ const found = this.dfsFind(child, targetId, visited);
406
+ if (found)
407
+ return found;
408
+ }
409
+ }
410
+ if (node.next)
411
+ return this.dfsFind(node.next, targetId, visited);
412
+ return undefined;
413
+ }
210
414
  // ─── Recorder Management ───
211
415
  /**
212
416
  * Attach a scope Recorder to observe data operations (reads, writes, commits).
@@ -348,10 +552,30 @@ export class FlowChartExecutor {
348
552
  for (const r of this.scopeRecorders) {
349
553
  (_b = r.clear) === null || _b === void 0 ? void 0 : _b.call(r);
350
554
  }
555
+ this.lastCheckpoint = undefined;
351
556
  this.traverser = this.createTraverser(signal, validatedInput, options === null || options === void 0 ? void 0 : options.env, options === null || options === void 0 ? void 0 : options.maxDepth);
352
557
  try {
353
558
  return await this.traverser.execute();
354
559
  }
560
+ catch (error) {
561
+ if (isPauseSignal(error)) {
562
+ // Build checkpoint from current execution state
563
+ const snapshot = this.traverser.getSnapshot();
564
+ const sfResults = this.traverser.getSubflowResults();
565
+ this.lastCheckpoint = {
566
+ sharedState: snapshot.sharedState,
567
+ executionTree: snapshot.executionTree,
568
+ pausedStageId: error.stageId,
569
+ subflowPath: error.subflowPath,
570
+ pauseData: error.pauseData,
571
+ ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),
572
+ pausedAt: Date.now(),
573
+ };
574
+ // Return a PauseResult-shaped value so callers can check without try/catch
575
+ return { paused: true, checkpoint: this.lastCheckpoint };
576
+ }
577
+ throw error;
578
+ }
355
579
  finally {
356
580
  if (timeoutId !== undefined)
357
581
  clearTimeout(timeoutId);
@@ -433,4 +657,4 @@ export class FlowChartExecutor {
433
657
  return recorder === null || recorder === void 0 ? void 0 : recorder.getSpec(subflowId);
434
658
  }
435
659
  }
436
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Build a chart with flowChart() and pass the result here:\n *\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *\n *   // No-options form (uses auto-detected TypedScope factory from the chart):\n *   const executor = new FlowChartExecutor(chart);\n *\n *   // Options-object form (preferred when you need to customize behavior):\n *   const executor = new FlowChartExecutor(chart, { scopeFactory: myFactory, enrichSnapshots: true });\n *\n *   // 2-param form (accepts a ScopeFactory directly, for backward compatibility):\n *   const executor = new FlowChartExecutor(chart, myFactory);\n *\n *   const result = await executor.run({ input: data, env: { traceId: 'req-123' } });\n */\n\nimport type { CombinedNarrativeRecorderOptions } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { CombinedNarrativeEntry } from '../engine/narrative/narrativeTypes.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExtractorError,\n  type FlowChart,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { Recorder, RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\n/**\n * Options object for `FlowChartExecutor` — preferred over positional params.\n *\n * ```typescript\n * const ex = new FlowChartExecutor(chart, {\n *   scopeFactory: myFactory,\n *   enrichSnapshots: true,\n * });\n * ```\n *\n * **Sync note for maintainers:** Every field added here must also appear in the\n * `flowChartArgs` private field type and in the constructor's options-resolution\n * block (the `else if` branch that reads from `opts`). Missing any one of the\n * three causes silent omission — the option is accepted but never applied.\n *\n * **TScope inference note:** When using the options-object form with a custom scope,\n * TypeScript cannot infer `TScope` through the options object. Pass the type\n * explicitly: `new FlowChartExecutor<TOut, MyScope>(chart, { scopeFactory })`.\n */\nexport interface FlowChartExecutorOptions<TScope = any> {\n  // ── Common options (most callers need only these) ────────────────────────\n\n  /** Custom scope factory. Defaults to TypedScope or ScopeFacade auto-detection. */\n  scopeFactory?: ScopeFactory<TScope>;\n  /**\n   * Attach a per-stage scope snapshot to each extractor result. When `true`, the\n   * extraction callback receives the full shared state at the point that stage\n   * committed — useful for debugging multi-stage state transitions. Defaults to\n   * `false` (no scope snapshot attached). Can also be set on the chart via\n   * `flowChart(...).enrichSnapshots(true)`.\n   */\n  enrichSnapshots?: boolean;\n\n  // ── Context options ──────────────────────────────────────────────────────\n\n  /**\n   * Default values pre-populated into the shared context before **each** stage\n   * (re-applied every stage, acting as baseline defaults).\n   */\n  defaultValuesForContext?: unknown;\n  /**\n   * Initial context values merged into the shared context **once** at startup\n   * (applied before the first stage, not repeated on subsequent stages).\n   * Distinct from `defaultValuesForContext`, which is re-applied every stage.\n   */\n  initialContext?: unknown;\n  /** Read-only input accessible via `scope.getArgs()` — never tracked or written. */\n  readOnlyContext?: unknown;\n\n  // ── Advanced / escape-hatch options (most callers do not need these) ─────\n\n  /**\n   * Custom error classifier for throttling detection. Return `true` if the\n   * error represents a rate-limit or backpressure condition (the executor will\n   * treat it differently from hard failures). Defaults to no throttling classification.\n   */\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  /** Handlers for streaming stage lifecycle events (see `addStreamingFunction`). */\n  streamHandlers?: StreamHandlers;\n  /** Scope protection mode for TypedScope direct-assignment detection. */\n  scopeProtectionMode?: ScopeProtectionMode;\n}\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private narrativeOptions?: CombinedNarrativeRecorderOptions;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private scopeRecorders: Recorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n  // SYNC REQUIRED: every optional field here must mirror FlowChartExecutorOptions\n  // AND be assigned in the constructor's options-resolution block (the `else if` branch).\n  // Adding a field to only one of the three places causes silent omission.\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  /**\n   * Create a FlowChartExecutor.\n   *\n   * **Options object form** (preferred):\n   * ```typescript\n   * new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots: true })\n   * ```\n   *\n   * **2-param form** (also supported):\n   * ```typescript\n   * new FlowChartExecutor(chart, scopeFactory)\n   * ```\n   *\n   * @param flowChart - The compiled FlowChart returned by `flowChart(...).build()`\n   * @param factoryOrOptions - A `ScopeFactory<TScope>` OR a `FlowChartExecutorOptions<TScope>` options object.\n   */\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    factoryOrOptions?: ScopeFactory<TScope> | FlowChartExecutorOptions<TScope>,\n  ) {\n    // Detect options-object form vs factory form\n    let scopeFactory: ScopeFactory<TScope> | undefined;\n    let defaultValuesForContext: unknown;\n    let initialContext: unknown;\n    let readOnlyContext: unknown;\n    let throttlingErrorChecker: ((error: unknown) => boolean) | undefined;\n    let streamHandlers: StreamHandlers | undefined;\n    let scopeProtectionMode: ScopeProtectionMode | undefined;\n    let enrichSnapshots: boolean | undefined;\n\n    if (typeof factoryOrOptions === 'function') {\n      // 2-param form: new FlowChartExecutor(chart, scopeFactory)\n      scopeFactory = factoryOrOptions;\n    } else if (factoryOrOptions !== undefined) {\n      // Options object form: new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots, ... })\n      const opts = factoryOrOptions;\n      scopeFactory = opts.scopeFactory;\n      defaultValuesForContext = opts.defaultValuesForContext;\n      initialContext = opts.initialContext;\n      readOnlyContext = opts.readOnlyContext;\n      throttlingErrorChecker = opts.throttlingErrorChecker;\n      streamHandlers = opts.streamHandlers;\n      scopeProtectionMode = opts.scopeProtectionMode;\n      enrichSnapshots = opts.enrichSnapshots;\n    }\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? flowChart.scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n    maxDepth?: number,\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // ── Composed scope factory ─────────────────────────────────────────\n    // Collect all scope modifiers (recorders, redaction) into a single list,\n    // then create ONE factory that applies them in a loop. Replaces the\n    // previous 4-deep closure nesting with a flat, debuggable composition.\n\n    if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder(this.narrativeOptions);\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    this.sharedRedactedKeys = new Set<string>();\n    this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n    // Build modifier list — each modifier receives the scope after creation\n    type ScopeModifier = (scope: any) => void;\n    const modifiers: ScopeModifier[] = [];\n\n    // 1. Narrative recorder (if enabled)\n    if (this.combinedRecorder) {\n      const recorder = this.combinedRecorder;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') scope.attachRecorder(recorder);\n      });\n    }\n\n    // 2. User-provided scope recorders\n    if (this.scopeRecorders.length > 0) {\n      const recorders = this.scopeRecorders;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') {\n          for (const r of recorders) scope.attachRecorder(r);\n        }\n      });\n    }\n\n    // 3. Redaction policy (conditional — only when policy is set)\n    if (this.redactionPolicy) {\n      const policy = this.redactionPolicy;\n      modifiers.push((scope) => {\n        if (typeof scope.useRedactionPolicy === 'function') {\n          scope.useRedactionPolicy(policy);\n        }\n      });\n      // Pre-populate executor-level field redaction map from policy\n      // so getRedactionReport() includes field-level redactions.\n      if (policy.fields) {\n        for (const [key, fields] of Object.entries(policy.fields)) {\n          this.sharedRedactedFieldsByKey.set(key, new Set(fields));\n        }\n      }\n    }\n\n    // Compose: base factory + modifiers in a single pass.\n    // Shared redacted keys are ALWAYS wired up (unconditional — ensures cross-stage\n    // propagation even without a policy, because stages can call setValue(key, val, true)\n    // for per-call redaction). Optional modifiers (recorders, policy) are in the list.\n    const baseFactory = args.scopeFactory;\n    const sharedRedactedKeys = this.sharedRedactedKeys;\n    const scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n      const scope = baseFactory(ctx, stageName, readOnly, envArg);\n      // Always wire shared redaction state\n      if (typeof (scope as any).useSharedRedactedKeys === 'function') {\n        (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n      }\n      // Apply optional modifiers\n      for (const mod of modifiers) mod(scope);\n      return scope;\n    }) as ScopeFactory<TScope>;\n\n    const runtime = new ExecutionRuntime(fc.root.name, fc.root.id, args.defaultValuesForContext, args.initialContext);\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: fc.root,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n      ...(maxDepth !== undefined && { maxDepth }),\n    });\n  }\n\n  enableNarrative(options?: CombinedNarrativeRecorderOptions): void {\n    this.narrativeEnabled = true;\n    if (options) this.narrativeOptions = options;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── Recorder Management ───\n\n  /**\n   * Attach a scope Recorder to observe data operations (reads, writes, commits).\n   * Automatically attached to every ScopeFacade created during traversal.\n   * Must be called before run().\n   *\n   * **Idempotent by ID:** If a recorder with the same `id` is already attached,\n   * it is replaced (not duplicated). This prevents double-counting when both\n   * a framework and the user attach the same recorder type.\n   *\n   * Built-in recorders use auto-increment IDs (`metrics-1`, `debug-1`, ...) by\n   * default, so multiple instances with different configs coexist. To override\n   * a framework-attached recorder, pass the same well-known ID.\n   *\n   * @example\n   * ```typescript\n   * // Multiple recorders with different configs — each gets a unique ID\n   * executor.attachRecorder(new MetricRecorder());\n   * executor.attachRecorder(new DebugRecorder({ verbosity: 'minimal' }));\n   *\n   * // Override a framework-attached recorder by passing its well-known ID\n   * executor.attachRecorder(new MetricRecorder('metrics'));\n   *\n   * // Attaching twice with same ID replaces (no double-counting)\n   * executor.attachRecorder(new MetricRecorder('my-metrics'));\n   * executor.attachRecorder(new MetricRecorder('my-metrics')); // replaces previous\n   * ```\n   */\n  attachRecorder(recorder: Recorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== recorder.id);\n    this.scopeRecorders.push(recorder);\n  }\n\n  /** Detach all scope Recorders with the given ID. */\n  detachRecorder(id: string): void {\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached scope Recorders. */\n  getRecorders(): Recorder[] {\n    return [...this.scopeRecorders];\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   *\n   * **Idempotent by ID:** replaces existing recorder with same `id`.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== recorder.id);\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   *\n   * Sentences come from `NarrativeFlowRecorder` (a dedicated flow-only recorder automatically\n   * attached when narrative is enabled). It emits both `onStageExecuted` sentences (one per\n   * stage) AND `onNext` transition sentences (one per stage-to-stage transition), so for a\n   * chart with N stages you will typically get more entries here than from `getNarrative()`.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<TraversalResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // User-attached recorders (flowRecorders + scopeRecorders) are cleared via clear() to prevent\n    // cross-run accumulation. The combinedRecorder is NOT cleared here — createTraverser() always\n    // creates a fresh CombinedNarrativeRecorder instance on each run, so stale state is never an issue.\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n    for (const r of this.scopeRecorders) {\n      r.clear?.();\n    }\n\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env, options?.maxDepth);\n    try {\n      return await this.traverser.execute();\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.scopeRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  /** @internal */\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  /** @internal */\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  /** @internal */\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  /** @internal */\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  /** @internal */\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  /** @internal */\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  /** @internal */\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  /** @internal */\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}
660
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Build a chart with flowChart() and pass the result here:\n *\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *\n *   // No-options form (uses auto-detected TypedScope factory from the chart):\n *   const executor = new FlowChartExecutor(chart);\n *\n *   // Options-object form (preferred when you need to customize behavior):\n *   const executor = new FlowChartExecutor(chart, { scopeFactory: myFactory, enrichSnapshots: true });\n *\n *   // 2-param form (accepts a ScopeFactory directly, for backward compatibility):\n *   const executor = new FlowChartExecutor(chart, myFactory);\n *\n *   const result = await executor.run({ input: data, env: { traceId: 'req-123' } });\n */\n\nimport type { CombinedNarrativeRecorderOptions } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { CombinedNarrativeEntry } from '../engine/narrative/narrativeTypes.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExecutorResult,\n  type ExtractorError,\n  type FlowChart,\n  type PausedResult,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { FlowchartCheckpoint } from '../pause/types.js';\nimport { isPauseSignal } from '../pause/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { Recorder, RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\n/**\n * Options object for `FlowChartExecutor` — preferred over positional params.\n *\n * ```typescript\n * const ex = new FlowChartExecutor(chart, {\n *   scopeFactory: myFactory,\n *   enrichSnapshots: true,\n * });\n * ```\n *\n * **Sync note for maintainers:** Every field added here must also appear in the\n * `flowChartArgs` private field type and in the constructor's options-resolution\n * block (the `else if` branch that reads from `opts`). Missing any one of the\n * three causes silent omission — the option is accepted but never applied.\n *\n * **TScope inference note:** When using the options-object form with a custom scope,\n * TypeScript cannot infer `TScope` through the options object. Pass the type\n * explicitly: `new FlowChartExecutor<TOut, MyScope>(chart, { scopeFactory })`.\n */\nexport interface FlowChartExecutorOptions<TScope = any> {\n  // ── Common options (most callers need only these) ────────────────────────\n\n  /** Custom scope factory. Defaults to TypedScope or ScopeFacade auto-detection. */\n  scopeFactory?: ScopeFactory<TScope>;\n  /**\n   * Attach a per-stage scope snapshot to each extractor result. When `true`, the\n   * extraction callback receives the full shared state at the point that stage\n   * committed — useful for debugging multi-stage state transitions. Defaults to\n   * `false` (no scope snapshot attached). Can also be set on the chart via\n   * `flowChart(...).enrichSnapshots(true)`.\n   */\n  enrichSnapshots?: boolean;\n\n  // ── Context options ──────────────────────────────────────────────────────\n\n  /**\n   * Default values pre-populated into the shared context before **each** stage\n   * (re-applied every stage, acting as baseline defaults).\n   */\n  defaultValuesForContext?: unknown;\n  /**\n   * Initial context values merged into the shared context **once** at startup\n   * (applied before the first stage, not repeated on subsequent stages).\n   * Distinct from `defaultValuesForContext`, which is re-applied every stage.\n   */\n  initialContext?: unknown;\n  /** Read-only input accessible via `scope.getArgs()` — never tracked or written. */\n  readOnlyContext?: unknown;\n\n  // ── Advanced / escape-hatch options (most callers do not need these) ─────\n\n  /**\n   * Custom error classifier for throttling detection. Return `true` if the\n   * error represents a rate-limit or backpressure condition (the executor will\n   * treat it differently from hard failures). Defaults to no throttling classification.\n   */\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  /** Handlers for streaming stage lifecycle events (see `addStreamingFunction`). */\n  streamHandlers?: StreamHandlers;\n  /** Scope protection mode for TypedScope direct-assignment detection. */\n  scopeProtectionMode?: ScopeProtectionMode;\n}\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private narrativeOptions?: CombinedNarrativeRecorderOptions;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private scopeRecorders: Recorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n  private lastCheckpoint: FlowchartCheckpoint | undefined;\n\n  // SYNC REQUIRED: every optional field here must mirror FlowChartExecutorOptions\n  // AND be assigned in the constructor's options-resolution block (the `else if` branch).\n  // Adding a field to only one of the three places causes silent omission.\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  /**\n   * Create a FlowChartExecutor.\n   *\n   * **Options object form** (preferred):\n   * ```typescript\n   * new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots: true })\n   * ```\n   *\n   * **2-param form** (also supported):\n   * ```typescript\n   * new FlowChartExecutor(chart, scopeFactory)\n   * ```\n   *\n   * @param flowChart - The compiled FlowChart returned by `flowChart(...).build()`\n   * @param factoryOrOptions - A `ScopeFactory<TScope>` OR a `FlowChartExecutorOptions<TScope>` options object.\n   */\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    factoryOrOptions?: ScopeFactory<TScope> | FlowChartExecutorOptions<TScope>,\n  ) {\n    // Detect options-object form vs factory form\n    let scopeFactory: ScopeFactory<TScope> | undefined;\n    let defaultValuesForContext: unknown;\n    let initialContext: unknown;\n    let readOnlyContext: unknown;\n    let throttlingErrorChecker: ((error: unknown) => boolean) | undefined;\n    let streamHandlers: StreamHandlers | undefined;\n    let scopeProtectionMode: ScopeProtectionMode | undefined;\n    let enrichSnapshots: boolean | undefined;\n\n    if (typeof factoryOrOptions === 'function') {\n      // 2-param form: new FlowChartExecutor(chart, scopeFactory)\n      scopeFactory = factoryOrOptions;\n    } else if (factoryOrOptions !== undefined) {\n      // Options object form: new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots, ... })\n      const opts = factoryOrOptions;\n      scopeFactory = opts.scopeFactory;\n      defaultValuesForContext = opts.defaultValuesForContext;\n      initialContext = opts.initialContext;\n      readOnlyContext = opts.readOnlyContext;\n      throttlingErrorChecker = opts.throttlingErrorChecker;\n      streamHandlers = opts.streamHandlers;\n      scopeProtectionMode = opts.scopeProtectionMode;\n      enrichSnapshots = opts.enrichSnapshots;\n    }\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? flowChart.scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n    maxDepth?: number,\n    overrides?: {\n      root?: StageNode<TOut, TScope>;\n      initialContext?: unknown;\n      preserveRecorders?: boolean;\n      existingRuntime?: InstanceType<typeof ExecutionRuntime>;\n    },\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // ── Composed scope factory ─────────────────────────────────────────\n    // Collect all scope modifiers (recorders, redaction) into a single list,\n    // then create ONE factory that applies them in a loop. Replaces the\n    // previous 4-deep closure nesting with a flat, debuggable composition.\n\n    if (overrides?.preserveRecorders) {\n      // Resume mode: keep existing combinedRecorder so narrative accumulates\n    } else if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder(this.narrativeOptions);\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    this.sharedRedactedKeys = new Set<string>();\n    this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n    // Build modifier list — each modifier receives the scope after creation\n    type ScopeModifier = (scope: any) => void;\n    const modifiers: ScopeModifier[] = [];\n\n    // 1. Narrative recorder (if enabled)\n    if (this.combinedRecorder) {\n      const recorder = this.combinedRecorder;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') scope.attachRecorder(recorder);\n      });\n    }\n\n    // 2. User-provided scope recorders\n    if (this.scopeRecorders.length > 0) {\n      const recorders = this.scopeRecorders;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') {\n          for (const r of recorders) scope.attachRecorder(r);\n        }\n      });\n    }\n\n    // 3. Redaction policy (conditional — only when policy is set)\n    if (this.redactionPolicy) {\n      const policy = this.redactionPolicy;\n      modifiers.push((scope) => {\n        if (typeof scope.useRedactionPolicy === 'function') {\n          scope.useRedactionPolicy(policy);\n        }\n      });\n      // Pre-populate executor-level field redaction map from policy\n      // so getRedactionReport() includes field-level redactions.\n      if (policy.fields) {\n        for (const [key, fields] of Object.entries(policy.fields)) {\n          this.sharedRedactedFieldsByKey.set(key, new Set(fields));\n        }\n      }\n    }\n\n    // Compose: base factory + modifiers in a single pass.\n    // Shared redacted keys are ALWAYS wired up (unconditional — ensures cross-stage\n    // propagation even without a policy, because stages can call setValue(key, val, true)\n    // for per-call redaction). Optional modifiers (recorders, policy) are in the list.\n    const baseFactory = args.scopeFactory;\n    const sharedRedactedKeys = this.sharedRedactedKeys;\n    const scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n      const scope = baseFactory(ctx, stageName, readOnly, envArg);\n      // Always wire shared redaction state\n      if (typeof (scope as any).useSharedRedactedKeys === 'function') {\n        (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n      }\n      // Apply optional modifiers\n      for (const mod of modifiers) mod(scope);\n      return scope;\n    }) as ScopeFactory<TScope>;\n\n    const effectiveRoot = overrides?.root ?? fc.root;\n    const effectiveInitialContext = overrides?.initialContext ?? args.initialContext;\n\n    let runtime: ExecutionRuntime;\n    if (overrides?.existingRuntime) {\n      // Resume mode: reuse existing runtime so execution tree continues from pause point.\n      // Preserve the original root for getSnapshot() (full tree), then advance\n      // rootStageContext to a continuation from the leaf (for traversal).\n      runtime = overrides.existingRuntime;\n      runtime.preserveSnapshotRoot();\n      let leaf = runtime.rootStageContext;\n      while (leaf.next) leaf = leaf.next;\n      runtime.rootStageContext = leaf.createNext('', effectiveRoot.name, effectiveRoot.id);\n    } else {\n      runtime = new ExecutionRuntime(\n        effectiveRoot.name,\n        effectiveRoot.id,\n        args.defaultValuesForContext,\n        effectiveInitialContext,\n      );\n    }\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: effectiveRoot,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n      ...(maxDepth !== undefined && { maxDepth }),\n    });\n  }\n\n  enableNarrative(options?: CombinedNarrativeRecorderOptions): void {\n    this.narrativeEnabled = true;\n    if (options) this.narrativeOptions = options;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── Pause/Resume ───\n\n  /**\n   * Returns the checkpoint from the most recent paused execution, or `undefined`\n   * if the last run completed without pausing.\n   *\n   * The checkpoint is JSON-serializable — store it in Redis, Postgres, localStorage, etc.\n   *\n   * @example\n   * ```typescript\n   * const result = await executor.run({ input });\n   * if (executor.isPaused()) {\n   *   const checkpoint = executor.getCheckpoint()!;\n   *   await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n   * }\n   * ```\n   */\n  getCheckpoint(): FlowchartCheckpoint | undefined {\n    return this.lastCheckpoint;\n  }\n\n  /** Returns `true` if the most recent run() was paused (checkpoint available). */\n  isPaused(): boolean {\n    return this.lastCheckpoint !== undefined;\n  }\n\n  /**\n   * Resume a paused flowchart from a checkpoint.\n   *\n   * Restores the scope state, calls the paused stage's `resumeFn` with the\n   * provided input, then continues traversal from the next stage.\n   *\n   * The checkpoint can come from `getCheckpoint()` on a previous run, or from\n   * a serialized checkpoint stored in Redis/Postgres/localStorage.\n   *\n   * **Narrative/recorder state is reset on resume.** To keep a unified narrative\n   * across pause/resume cycles, collect it before calling resume.\n   *\n   * @example\n   * ```typescript\n   * // After a pause...\n   * const checkpoint = executor.getCheckpoint()!;\n   * await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n   *\n   * // Later (possibly different server, same chart)\n   * const checkpoint = JSON.parse(await redis.get(`session:${id}`));\n   * const executor = new FlowChartExecutor(chart);\n   * const result = await executor.resume(checkpoint, { approved: true });\n   * ```\n   */\n  async resume(\n    checkpoint: FlowchartCheckpoint,\n    resumeInput?: unknown,\n    options?: Pick<RunOptions, 'signal' | 'env' | 'maxDepth'>,\n  ): Promise<ExecutorResult> {\n    this.lastCheckpoint = undefined;\n\n    // ── Validate checkpoint structure (may come from untrusted external storage) ──\n    if (\n      !checkpoint ||\n      typeof checkpoint !== 'object' ||\n      typeof checkpoint.sharedState !== 'object' ||\n      checkpoint.sharedState === null ||\n      Array.isArray(checkpoint.sharedState)\n    ) {\n      throw new Error('Invalid checkpoint: sharedState must be a plain object.');\n    }\n    if (typeof checkpoint.pausedStageId !== 'string' || checkpoint.pausedStageId === '') {\n      throw new Error('Invalid checkpoint: pausedStageId must be a non-empty string.');\n    }\n    if (\n      !Array.isArray(checkpoint.subflowPath) ||\n      !checkpoint.subflowPath.every((s: unknown) => typeof s === 'string')\n    ) {\n      throw new Error('Invalid checkpoint: subflowPath must be an array of strings.');\n    }\n\n    // Find the paused node in the graph\n    const pausedNode = this.findNodeInGraph(checkpoint.pausedStageId, checkpoint.subflowPath);\n    if (!pausedNode) {\n      throw new Error(\n        `Cannot resume: stage '${checkpoint.pausedStageId}' not found in flowchart. ` +\n          'The chart may have changed since the checkpoint was created.',\n      );\n    }\n    if (!pausedNode.resumeFn) {\n      throw new Error(\n        `Cannot resume: stage '${pausedNode.name}' (${pausedNode.id}) has no resumeFn. ` +\n          'Only stages created with addPausableFunction() can be resumed.',\n      );\n    }\n\n    // Build a synthetic resume node: calls resumeFn with resumeInput, then continues to original next.\n    // resumeFn signature is (scope, input) per PausableHandler — wrap to match StageFunction(scope, breakFn).\n    const resumeFn = pausedNode.resumeFn;\n    const resumeStageFn = (scope: TScope) => {\n      return resumeFn(scope, resumeInput);\n    };\n\n    const resumeNode: StageNode<TOut, TScope> = {\n      name: pausedNode.name,\n      id: pausedNode.id,\n      description: pausedNode.description,\n      fn: resumeStageFn,\n      next: pausedNode.next,\n    };\n\n    // Don't clear recorders — resume continues from previous state.\n    // Narrative, metrics, debug entries accumulate across pause/resume.\n\n    // Reuse the existing runtime so the execution tree continues from the pause point.\n    // preserveRecorders keeps the CombinedNarrativeRecorder so narrative accumulates.\n    const existingRuntime = this.traverser.getRuntime() as InstanceType<typeof ExecutionRuntime>;\n    this.traverser = this.createTraverser(options?.signal, undefined, options?.env, options?.maxDepth, {\n      root: resumeNode,\n      initialContext: checkpoint.sharedState,\n      preserveRecorders: true,\n      existingRuntime,\n    });\n\n    // Fire onResume event on all recorders (flow + scope)\n    const hasInput = resumeInput !== undefined;\n    const flowResumeEvent = {\n      stageName: pausedNode.name,\n      stageId: pausedNode.id,\n      hasInput,\n    };\n    if (this.combinedRecorder) this.combinedRecorder.onResume(flowResumeEvent);\n    for (const r of this.flowRecorders) r.onResume?.(flowResumeEvent);\n\n    const scopeResumeEvent = {\n      stageName: pausedNode.name,\n      stageId: pausedNode.id,\n      hasInput,\n      pipelineId: '',\n      timestamp: Date.now(),\n    };\n    for (const r of this.scopeRecorders) r.onResume?.(scopeResumeEvent);\n\n    try {\n      return await this.traverser.execute();\n    } catch (error: unknown) {\n      if (isPauseSignal(error)) {\n        const snapshot = this.traverser.getSnapshot();\n        const sfResults = this.traverser.getSubflowResults();\n        this.lastCheckpoint = {\n          sharedState: snapshot.sharedState,\n          executionTree: snapshot.executionTree,\n          pausedStageId: error.stageId,\n          subflowPath: error.subflowPath,\n          pauseData: error.pauseData,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          pausedAt: Date.now(),\n        };\n        return { paused: true, checkpoint: this.lastCheckpoint } satisfies PausedResult;\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Find a StageNode in the compiled graph by ID.\n   * Handles subflow paths by drilling into registered subflows.\n   */\n  private findNodeInGraph(stageId: string, subflowPath: readonly string[]): StageNode<TOut, TScope> | undefined {\n    const fc = this.flowChartArgs.flowChart;\n\n    if (subflowPath.length === 0) {\n      // Top-level: DFS from root\n      return this.dfsFind(fc.root, stageId);\n    }\n\n    // Subflow: drill into the subflow chain, then search from the last subflow's root\n    let subflowRoot: StageNode<TOut, TScope> | undefined;\n    for (const sfId of subflowPath) {\n      const subflow = fc.subflows?.[sfId];\n      if (!subflow) return undefined;\n      subflowRoot = subflow.root;\n    }\n    if (!subflowRoot) return undefined;\n    return this.dfsFind(subflowRoot, stageId);\n  }\n\n  /** DFS search for a node by ID in the StageNode graph. Cycle-safe via visited set. */\n  private dfsFind(\n    node: StageNode<TOut, TScope>,\n    targetId: string,\n    visited = new Set<string>(),\n  ): StageNode<TOut, TScope> | undefined {\n    // Skip loop back-edge references (they share the target's ID but have no fn/resumeFn)\n    if (node.isLoopRef) return undefined;\n    if (visited.has(node.id)) return undefined;\n    visited.add(node.id);\n    if (node.id === targetId) return node;\n    if (node.children) {\n      for (const child of node.children) {\n        const found = this.dfsFind(child, targetId, visited);\n        if (found) return found;\n      }\n    }\n    if (node.next) return this.dfsFind(node.next, targetId, visited);\n    return undefined;\n  }\n\n  // ─── Recorder Management ───\n\n  /**\n   * Attach a scope Recorder to observe data operations (reads, writes, commits).\n   * Automatically attached to every ScopeFacade created during traversal.\n   * Must be called before run().\n   *\n   * **Idempotent by ID:** If a recorder with the same `id` is already attached,\n   * it is replaced (not duplicated). This prevents double-counting when both\n   * a framework and the user attach the same recorder type.\n   *\n   * Built-in recorders use auto-increment IDs (`metrics-1`, `debug-1`, ...) by\n   * default, so multiple instances with different configs coexist. To override\n   * a framework-attached recorder, pass the same well-known ID.\n   *\n   * @example\n   * ```typescript\n   * // Multiple recorders with different configs — each gets a unique ID\n   * executor.attachRecorder(new MetricRecorder());\n   * executor.attachRecorder(new DebugRecorder({ verbosity: 'minimal' }));\n   *\n   * // Override a framework-attached recorder by passing its well-known ID\n   * executor.attachRecorder(new MetricRecorder('metrics'));\n   *\n   * // Attaching twice with same ID replaces (no double-counting)\n   * executor.attachRecorder(new MetricRecorder('my-metrics'));\n   * executor.attachRecorder(new MetricRecorder('my-metrics')); // replaces previous\n   * ```\n   */\n  attachRecorder(recorder: Recorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== recorder.id);\n    this.scopeRecorders.push(recorder);\n  }\n\n  /** Detach all scope Recorders with the given ID. */\n  detachRecorder(id: string): void {\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached scope Recorders. */\n  getRecorders(): Recorder[] {\n    return [...this.scopeRecorders];\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   *\n   * **Idempotent by ID:** replaces existing recorder with same `id`.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== recorder.id);\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   *\n   * Sentences come from `NarrativeFlowRecorder` (a dedicated flow-only recorder automatically\n   * attached when narrative is enabled). It emits both `onStageExecuted` sentences (one per\n   * stage) AND `onNext` transition sentences (one per stage-to-stage transition), so for a\n   * chart with N stages you will typically get more entries here than from `getNarrative()`.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<ExecutorResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // User-attached recorders (flowRecorders + scopeRecorders) are cleared via clear() to prevent\n    // cross-run accumulation. The combinedRecorder is NOT cleared here — createTraverser() always\n    // creates a fresh CombinedNarrativeRecorder instance on each run, so stale state is never an issue.\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n    for (const r of this.scopeRecorders) {\n      r.clear?.();\n    }\n\n    this.lastCheckpoint = undefined;\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env, options?.maxDepth);\n    try {\n      return await this.traverser.execute();\n    } catch (error: unknown) {\n      if (isPauseSignal(error)) {\n        // Build checkpoint from current execution state\n        const snapshot = this.traverser.getSnapshot();\n        const sfResults = this.traverser.getSubflowResults();\n        this.lastCheckpoint = {\n          sharedState: snapshot.sharedState,\n          executionTree: snapshot.executionTree,\n          pausedStageId: error.stageId,\n          subflowPath: error.subflowPath,\n          pauseData: error.pauseData,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          pausedAt: Date.now(),\n        };\n        // Return a PauseResult-shaped value so callers can check without try/catch\n        return { paused: true, checkpoint: this.lastCheckpoint } satisfies PausedResult;\n      }\n      throw error;\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.scopeRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  /** @internal */\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  /** @internal */\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  /** @internal */\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  /** @internal */\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  /** @internal */\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  /** @internal */\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  /** @internal */\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  /** @internal */\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}