footprintjs 4.16.0 → 4.17.2

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 (188) hide show
  1. package/AGENTS.md +567 -82
  2. package/CLAUDE.md +57 -0
  3. package/README.md +2 -0
  4. package/dist/advanced.js +4 -2
  5. package/dist/detach.js +78 -0
  6. package/dist/esm/advanced.js +2 -1
  7. package/dist/esm/detach.js +57 -0
  8. package/dist/esm/lib/builder/FlowChartBuilder.js +171 -61
  9. package/dist/esm/lib/contract/openapi.js +4 -5
  10. package/dist/esm/lib/contract/schema.js +115 -4
  11. package/dist/esm/lib/decide/decide.js +4 -5
  12. package/dist/esm/lib/decide/evidence.js +3 -2
  13. package/dist/esm/lib/detach/drivers/immediate.js +66 -0
  14. package/dist/esm/lib/detach/drivers/microtaskBatch.js +113 -0
  15. package/dist/esm/lib/detach/drivers/sendBeacon.js +78 -0
  16. package/dist/esm/lib/detach/drivers/setImmediate.js +81 -0
  17. package/dist/esm/lib/detach/drivers/setTimeout.js +69 -0
  18. package/dist/esm/lib/detach/drivers/workerThread.js +117 -0
  19. package/dist/esm/lib/detach/flush.js +91 -0
  20. package/dist/esm/lib/detach/handle.js +134 -0
  21. package/dist/esm/lib/detach/registry.js +97 -0
  22. package/dist/esm/lib/detach/runChild.js +40 -0
  23. package/dist/esm/lib/detach/spawn.js +86 -0
  24. package/dist/esm/lib/detach/types.js +37 -0
  25. package/dist/esm/lib/engine/errors/errorInfo.js +4 -4
  26. package/dist/esm/lib/engine/graph/StageNode.js +2 -2
  27. package/dist/esm/lib/engine/handlers/ChildrenExecutor.js +9 -8
  28. package/dist/esm/lib/engine/handlers/ContinuationResolver.js +12 -9
  29. package/dist/esm/lib/engine/handlers/DeciderHandler.js +7 -8
  30. package/dist/esm/lib/engine/handlers/ExtractorRunner.js +14 -8
  31. package/dist/esm/lib/engine/handlers/NodeResolver.js +5 -3
  32. package/dist/esm/lib/engine/handlers/RuntimeStructureManager.js +11 -14
  33. package/dist/esm/lib/engine/handlers/SelectorHandler.js +9 -9
  34. package/dist/esm/lib/engine/handlers/StageRunner.js +9 -11
  35. package/dist/esm/lib/engine/handlers/SubflowExecutor.js +13 -12
  36. package/dist/esm/lib/engine/handlers/SubflowInputMapper.js +4 -4
  37. package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +85 -96
  38. package/dist/esm/lib/engine/narrative/FlowRecorderDispatcher.js +18 -36
  39. package/dist/esm/lib/engine/narrative/NarrativeFlowRecorder.js +6 -5
  40. package/dist/esm/lib/engine/narrative/recorders/AdaptiveNarrativeFlowRecorder.js +7 -6
  41. package/dist/esm/lib/engine/narrative/recorders/ManifestFlowRecorder.js +10 -10
  42. package/dist/esm/lib/engine/narrative/recorders/MilestoneNarrativeFlowRecorder.js +5 -3
  43. package/dist/esm/lib/engine/narrative/recorders/ProgressiveNarrativeFlowRecorder.js +4 -3
  44. package/dist/esm/lib/engine/narrative/recorders/RLENarrativeFlowRecorder.js +4 -4
  45. package/dist/esm/lib/engine/narrative/recorders/SeparateNarrativeFlowRecorder.js +5 -6
  46. package/dist/esm/lib/engine/narrative/recorders/SilentNarrativeFlowRecorder.js +5 -6
  47. package/dist/esm/lib/engine/narrative/recorders/WindowedNarrativeFlowRecorder.js +5 -3
  48. package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +97 -71
  49. package/dist/esm/lib/memory/DiagnosticCollector.js +6 -8
  50. package/dist/esm/lib/memory/EventLog.js +5 -3
  51. package/dist/esm/lib/memory/SharedMemory.js +3 -2
  52. package/dist/esm/lib/memory/StageContext.js +44 -14
  53. package/dist/esm/lib/memory/TransactionBuffer.js +9 -8
  54. package/dist/esm/lib/memory/backtrack.js +3 -4
  55. package/dist/esm/lib/memory/commitLogUtils.js +2 -2
  56. package/dist/esm/lib/memory/utils.js +2 -3
  57. package/dist/esm/lib/pause/types.js +33 -14
  58. package/dist/esm/lib/reactive/createTypedScope.js +10 -8
  59. package/dist/esm/lib/reactive/types.js +3 -1
  60. package/dist/esm/lib/recorder/BoundaryStateTracker.js +263 -0
  61. package/dist/esm/lib/recorder/CompositeRecorder.js +3 -1
  62. package/dist/esm/lib/recorder/InOutRecorder.js +5 -6
  63. package/dist/esm/lib/recorder/KeyedRecorder.js +2 -4
  64. package/dist/esm/lib/recorder/QualityRecorder.js +15 -14
  65. package/dist/esm/lib/recorder/SequenceRecorder.js +11 -12
  66. package/dist/esm/lib/recorder/TopologyRecorder.js +36 -40
  67. package/dist/esm/lib/recorder/index.js +2 -1
  68. package/dist/esm/lib/recorder/qualityTrace.js +4 -5
  69. package/dist/esm/lib/runner/ExecutionRuntime.js +20 -4
  70. package/dist/esm/lib/runner/FlowChartExecutor.js +99 -55
  71. package/dist/esm/lib/runner/RunContext.js +5 -3
  72. package/dist/esm/lib/runner/RunnableChart.js +7 -9
  73. package/dist/esm/lib/runner/getSubtreeSnapshot.js +4 -5
  74. package/dist/esm/lib/schema/errors.js +4 -3
  75. package/dist/esm/lib/schema/validate.js +4 -5
  76. package/dist/esm/lib/scope/ScopeFacade.js +52 -35
  77. package/dist/esm/lib/scope/providers/baseStateCompatible.js +9 -9
  78. package/dist/esm/lib/scope/providers/guards.js +2 -2
  79. package/dist/esm/lib/scope/recorders/DebugRecorder.js +9 -7
  80. package/dist/esm/lib/scope/recorders/MetricRecorder.js +10 -8
  81. package/dist/esm/lib/scope/state/zod/defineScopeFromZod.js +2 -3
  82. package/dist/esm/lib/scope/state/zod/resolver.js +2 -3
  83. package/dist/esm/lib/scope/state/zod/scopeFactory.js +16 -20
  84. package/dist/esm/lib/scope/state/zod/utils/validateHelper.js +57 -14
  85. package/dist/esm/trace.js +4 -1
  86. package/dist/lib/builder/FlowChartBuilder.js +171 -61
  87. package/dist/lib/contract/openapi.js +4 -5
  88. package/dist/lib/contract/schema.js +115 -4
  89. package/dist/lib/decide/decide.js +4 -5
  90. package/dist/lib/decide/evidence.js +3 -2
  91. package/dist/lib/detach/drivers/immediate.js +70 -0
  92. package/dist/lib/detach/drivers/microtaskBatch.js +117 -0
  93. package/dist/lib/detach/drivers/sendBeacon.js +82 -0
  94. package/dist/lib/detach/drivers/setImmediate.js +85 -0
  95. package/dist/lib/detach/drivers/setTimeout.js +73 -0
  96. package/dist/lib/detach/drivers/workerThread.js +121 -0
  97. package/dist/lib/detach/flush.js +95 -0
  98. package/dist/lib/detach/handle.js +140 -0
  99. package/dist/lib/detach/registry.js +106 -0
  100. package/dist/lib/detach/runChild.js +67 -0
  101. package/dist/lib/detach/spawn.js +92 -0
  102. package/dist/lib/detach/types.js +38 -0
  103. package/dist/lib/engine/errors/errorInfo.js +4 -4
  104. package/dist/lib/engine/graph/StageNode.js +2 -2
  105. package/dist/lib/engine/handlers/ChildrenExecutor.js +9 -8
  106. package/dist/lib/engine/handlers/ContinuationResolver.js +12 -9
  107. package/dist/lib/engine/handlers/DeciderHandler.js +7 -8
  108. package/dist/lib/engine/handlers/ExtractorRunner.js +14 -8
  109. package/dist/lib/engine/handlers/NodeResolver.js +5 -3
  110. package/dist/lib/engine/handlers/RuntimeStructureManager.js +11 -14
  111. package/dist/lib/engine/handlers/SelectorHandler.js +9 -9
  112. package/dist/lib/engine/handlers/StageRunner.js +9 -11
  113. package/dist/lib/engine/handlers/SubflowExecutor.js +13 -12
  114. package/dist/lib/engine/handlers/SubflowInputMapper.js +4 -4
  115. package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +85 -96
  116. package/dist/lib/engine/narrative/FlowRecorderDispatcher.js +18 -36
  117. package/dist/lib/engine/narrative/NarrativeFlowRecorder.js +6 -5
  118. package/dist/lib/engine/narrative/recorders/AdaptiveNarrativeFlowRecorder.js +7 -6
  119. package/dist/lib/engine/narrative/recorders/ManifestFlowRecorder.js +10 -10
  120. package/dist/lib/engine/narrative/recorders/MilestoneNarrativeFlowRecorder.js +5 -3
  121. package/dist/lib/engine/narrative/recorders/ProgressiveNarrativeFlowRecorder.js +4 -3
  122. package/dist/lib/engine/narrative/recorders/RLENarrativeFlowRecorder.js +4 -4
  123. package/dist/lib/engine/narrative/recorders/SeparateNarrativeFlowRecorder.js +5 -6
  124. package/dist/lib/engine/narrative/recorders/SilentNarrativeFlowRecorder.js +5 -6
  125. package/dist/lib/engine/narrative/recorders/WindowedNarrativeFlowRecorder.js +5 -3
  126. package/dist/lib/engine/traversal/FlowchartTraverser.js +97 -71
  127. package/dist/lib/memory/DiagnosticCollector.js +6 -8
  128. package/dist/lib/memory/EventLog.js +5 -3
  129. package/dist/lib/memory/SharedMemory.js +3 -2
  130. package/dist/lib/memory/StageContext.js +44 -14
  131. package/dist/lib/memory/TransactionBuffer.js +9 -8
  132. package/dist/lib/memory/backtrack.js +3 -4
  133. package/dist/lib/memory/commitLogUtils.js +2 -2
  134. package/dist/lib/memory/utils.js +2 -3
  135. package/dist/lib/pause/types.js +33 -14
  136. package/dist/lib/reactive/createTypedScope.js +10 -8
  137. package/dist/lib/reactive/types.js +3 -1
  138. package/dist/lib/recorder/BoundaryStateTracker.js +267 -0
  139. package/dist/lib/recorder/CompositeRecorder.js +3 -1
  140. package/dist/lib/recorder/InOutRecorder.js +5 -6
  141. package/dist/lib/recorder/KeyedRecorder.js +2 -4
  142. package/dist/lib/recorder/QualityRecorder.js +15 -14
  143. package/dist/lib/recorder/SequenceRecorder.js +11 -12
  144. package/dist/lib/recorder/TopologyRecorder.js +36 -40
  145. package/dist/lib/recorder/index.js +4 -2
  146. package/dist/lib/recorder/qualityTrace.js +4 -5
  147. package/dist/lib/runner/ExecutionRuntime.js +20 -4
  148. package/dist/lib/runner/FlowChartExecutor.js +99 -55
  149. package/dist/lib/runner/RunContext.js +5 -3
  150. package/dist/lib/runner/RunnableChart.js +7 -9
  151. package/dist/lib/runner/getSubtreeSnapshot.js +4 -5
  152. package/dist/lib/schema/errors.js +4 -3
  153. package/dist/lib/schema/validate.js +4 -5
  154. package/dist/lib/scope/ScopeFacade.js +52 -35
  155. package/dist/lib/scope/providers/baseStateCompatible.js +9 -9
  156. package/dist/lib/scope/providers/guards.js +2 -2
  157. package/dist/lib/scope/recorders/DebugRecorder.js +9 -7
  158. package/dist/lib/scope/recorders/MetricRecorder.js +10 -8
  159. package/dist/lib/scope/state/zod/defineScopeFromZod.js +2 -3
  160. package/dist/lib/scope/state/zod/resolver.js +2 -3
  161. package/dist/lib/scope/state/zod/scopeFactory.js +16 -20
  162. package/dist/lib/scope/state/zod/utils/validateHelper.js +57 -14
  163. package/dist/trace.js +6 -2
  164. package/dist/types/advanced.d.ts +1 -0
  165. package/dist/types/detach.d.ts +59 -0
  166. package/dist/types/lib/builder/FlowChartBuilder.d.ts +81 -0
  167. package/dist/types/lib/detach/drivers/immediate.d.ts +39 -0
  168. package/dist/types/lib/detach/drivers/microtaskBatch.d.ts +57 -0
  169. package/dist/types/lib/detach/drivers/sendBeacon.d.ts +38 -0
  170. package/dist/types/lib/detach/drivers/setImmediate.d.ts +32 -0
  171. package/dist/types/lib/detach/drivers/setTimeout.d.ts +34 -0
  172. package/dist/types/lib/detach/drivers/workerThread.d.ts +50 -0
  173. package/dist/types/lib/detach/flush.d.ts +62 -0
  174. package/dist/types/lib/detach/handle.d.ts +83 -0
  175. package/dist/types/lib/detach/registry.d.ts +82 -0
  176. package/dist/types/lib/detach/runChild.d.ts +41 -0
  177. package/dist/types/lib/detach/spawn.d.ts +64 -0
  178. package/dist/types/lib/detach/types.d.ts +200 -0
  179. package/dist/types/lib/engine/traversal/FlowchartTraverser.d.ts +0 -1
  180. package/dist/types/lib/engine/types.d.ts +0 -1
  181. package/dist/types/lib/reactive/types.d.ts +4 -0
  182. package/dist/types/lib/recorder/BoundaryStateTracker.d.ts +215 -0
  183. package/dist/types/lib/recorder/index.d.ts +1 -0
  184. package/dist/types/lib/runner/FlowChartExecutor.d.ts +28 -0
  185. package/dist/types/lib/scope/ScopeFacade.d.ts +4 -0
  186. package/dist/types/lib/scope/state/zod/utils/validateHelper.d.ts +13 -1
  187. package/dist/types/trace.d.ts +1 -0
  188. package/package.json +6 -1
@@ -19,6 +19,7 @@
19
19
  */
20
20
  Object.defineProperty(exports, "__esModule", { value: true });
21
21
  exports.FlowChartExecutor = void 0;
22
+ const spawn_js_1 = require("../detach/spawn.js");
22
23
  const CombinedNarrativeRecorder_js_1 = require("../engine/narrative/CombinedNarrativeRecorder.js");
23
24
  const ManifestFlowRecorder_js_1 = require("../engine/narrative/recorders/ManifestFlowRecorder.js");
24
25
  const runtimeStageId_js_1 = require("../engine/runtimeStageId.js");
@@ -33,6 +34,38 @@ const validateInput_js_1 = require("./validateInput.js");
33
34
  /** Default scope factory — creates a plain ScopeFacade for each stage. */
34
35
  const defaultScopeFactory = (ctx, stageName, readOnly, env) => new ScopeFacade_js_1.ScopeFacade(ctx, stageName, readOnly, env);
35
36
  class FlowChartExecutor {
37
+ traverser;
38
+ /** Shared execution counter — survives pause/resume. Reset on fresh run(). */
39
+ _executionCounter = { value: 0 };
40
+ narrativeEnabled = false;
41
+ narrativeOptions;
42
+ combinedRecorder;
43
+ flowRecorders = [];
44
+ scopeRecorders = [];
45
+ redactionPolicy;
46
+ sharedRedactedKeys = new Set();
47
+ sharedRedactedFieldsByKey = new Map();
48
+ lastCheckpoint;
49
+ /**
50
+ * `true` once `run()` (or a previous `resume()`) has executed on
51
+ * this instance. `resume()` branches on it:
52
+ *
53
+ * • true → reuse the constructor-time runtime (same-executor
54
+ * continuity: execution tree, recorders, narrative
55
+ * accumulate across pause/resume cycles)
56
+ * • false → seed a fresh runtime from `checkpoint.sharedState`
57
+ * (cross-executor / cross-process resume: new instance
58
+ * reconstructed from a serialized checkpoint)
59
+ *
60
+ * Without this flag, fresh executors silently discarded the
61
+ * checkpoint's sharedState and resume handlers couldn't read pre-pause
62
+ * scope. See `test/lib/pause/cross-executor-resume.test.ts`.
63
+ */
64
+ _hasRunBefore = false;
65
+ // SYNC REQUIRED: every optional field here must mirror FlowChartExecutorOptions
66
+ // AND be assigned in the constructor's options-resolution block (the `else if` branch).
67
+ // Adding a field to only one of the three places causes silent omission.
68
+ flowChartArgs;
36
69
  /**
37
70
  * Create a FlowChartExecutor.
38
71
  *
@@ -50,30 +83,6 @@ class FlowChartExecutor {
50
83
  * @param factoryOrOptions - A `ScopeFactory<TScope>` OR a `FlowChartExecutorOptions<TScope>` options object.
51
84
  */
52
85
  constructor(flowChart, factoryOrOptions) {
53
- var _a;
54
- /** Shared execution counter — survives pause/resume. Reset on fresh run(). */
55
- this._executionCounter = { value: 0 };
56
- this.narrativeEnabled = false;
57
- this.flowRecorders = [];
58
- this.scopeRecorders = [];
59
- this.sharedRedactedKeys = new Set();
60
- this.sharedRedactedFieldsByKey = new Map();
61
- /**
62
- * `true` once `run()` (or a previous `resume()`) has executed on
63
- * this instance. `resume()` branches on it:
64
- *
65
- * • true → reuse the constructor-time runtime (same-executor
66
- * continuity: execution tree, recorders, narrative
67
- * accumulate across pause/resume cycles)
68
- * • false → seed a fresh runtime from `checkpoint.sharedState`
69
- * (cross-executor / cross-process resume: new instance
70
- * reconstructed from a serialized checkpoint)
71
- *
72
- * Without this flag, fresh executors silently discarded the
73
- * checkpoint's sharedState and resume handlers couldn't read pre-pause
74
- * scope. See `test/lib/pause/cross-executor-resume.test.ts`.
75
- */
76
- this._hasRunBefore = false;
77
86
  // Detect options-object form vs factory form
78
87
  let scopeFactory;
79
88
  let defaultValuesForContext;
@@ -101,7 +110,7 @@ class FlowChartExecutor {
101
110
  }
102
111
  this.flowChartArgs = {
103
112
  flowChart,
104
- scopeFactory: (_a = scopeFactory !== null && scopeFactory !== void 0 ? scopeFactory : flowChart.scopeFactory) !== null && _a !== void 0 ? _a : defaultScopeFactory,
113
+ scopeFactory: scopeFactory ?? flowChart.scopeFactory ?? defaultScopeFactory,
105
114
  defaultValuesForContext,
106
115
  initialContext,
107
116
  readOnlyContext,
@@ -113,15 +122,14 @@ class FlowChartExecutor {
113
122
  this.traverser = this.createTraverser();
114
123
  }
115
124
  createTraverser(signal, readOnlyContextOverride, env, maxDepth, overrides) {
116
- var _a, _b, _c, _d;
117
125
  const args = this.flowChartArgs;
118
126
  const fc = args.flowChart;
119
- const narrativeFlag = this.narrativeEnabled || ((_a = fc.enableNarrative) !== null && _a !== void 0 ? _a : false);
127
+ const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);
120
128
  // ── Composed scope factory ─────────────────────────────────────────
121
129
  // Collect all scope modifiers (recorders, redaction) into a single list,
122
130
  // then create ONE factory that applies them in a loop. Replaces the
123
131
  // previous 4-deep closure nesting with a flat, debuggable composition.
124
- if (overrides === null || overrides === void 0 ? void 0 : overrides.preserveRecorders) {
132
+ if (overrides?.preserveRecorders) {
125
133
  // Resume mode: keep existing combinedRecorder so narrative accumulates
126
134
  }
127
135
  else if (narrativeFlag) {
@@ -184,10 +192,10 @@ class FlowChartExecutor {
184
192
  mod(scope);
185
193
  return scope;
186
194
  });
187
- const effectiveRoot = (_b = overrides === null || overrides === void 0 ? void 0 : overrides.root) !== null && _b !== void 0 ? _b : fc.root;
188
- const effectiveInitialContext = (_c = overrides === null || overrides === void 0 ? void 0 : overrides.initialContext) !== null && _c !== void 0 ? _c : args.initialContext;
195
+ const effectiveRoot = overrides?.root ?? fc.root;
196
+ const effectiveInitialContext = overrides?.initialContext ?? args.initialContext;
189
197
  let runtime;
190
- if (overrides === null || overrides === void 0 ? void 0 : overrides.existingRuntime) {
198
+ if (overrides?.existingRuntime) {
191
199
  // Resume mode: reuse existing runtime so execution tree continues from pause point.
192
200
  // Preserve the original root for getSnapshot() (full tree), then advance
193
201
  // rootStageContext to a continuation from the leaf (for traversal).
@@ -214,7 +222,7 @@ class FlowChartExecutor {
214
222
  stageMap: fc.stageMap,
215
223
  scopeFactory,
216
224
  executionRuntime: runtime,
217
- readOnlyContext: readOnlyContextOverride !== null && readOnlyContextOverride !== void 0 ? readOnlyContextOverride : args.readOnlyContext,
225
+ readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,
218
226
  throttlingErrorChecker: args.throttlingErrorChecker,
219
227
  streamHandlers: args.streamHandlers,
220
228
  extractor: fc.extractor,
@@ -223,13 +231,13 @@ class FlowChartExecutor {
223
231
  enrichSnapshots: args.enrichSnapshots,
224
232
  narrativeEnabled: narrativeFlag,
225
233
  buildTimeStructure: fc.buildTimeStructure,
226
- logger: (_d = fc.logger) !== null && _d !== void 0 ? _d : types_js_1.defaultLogger,
234
+ logger: fc.logger ?? types_js_1.defaultLogger,
227
235
  signal,
228
236
  executionEnv: env,
229
237
  flowRecorders: this.buildFlowRecordersList(),
230
238
  executionCounter: this._executionCounter,
231
- ...((overrides === null || overrides === void 0 ? void 0 : overrides.subflowsOverride) && { subflows: overrides.subflowsOverride }),
232
- ...((overrides === null || overrides === void 0 ? void 0 : overrides.subflowStatesForResume) && {
239
+ ...(overrides?.subflowsOverride && { subflows: overrides.subflowsOverride }),
240
+ ...(overrides?.subflowStatesForResume && {
233
241
  subflowStatesForResume: overrides.subflowStatesForResume,
234
242
  }),
235
243
  ...(maxDepth !== undefined && { maxDepth }),
@@ -252,7 +260,6 @@ class FlowChartExecutor {
252
260
  * most recent run. Never includes actual values.
253
261
  */
254
262
  getRedactionReport() {
255
- var _a, _b;
256
263
  const fieldRedactions = {};
257
264
  for (const [key, fields] of this.sharedRedactedFieldsByKey) {
258
265
  fieldRedactions[key] = [...fields];
@@ -260,7 +267,7 @@ class FlowChartExecutor {
260
267
  return {
261
268
  redactedKeys: [...this.sharedRedactedKeys],
262
269
  fieldRedactions,
263
- patterns: ((_b = (_a = this.redactionPolicy) === null || _a === void 0 ? void 0 : _a.patterns) !== null && _b !== void 0 ? _b : []).map((p) => p.source),
270
+ patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),
264
271
  };
265
272
  }
266
273
  // ─── Pause/Resume ───
@@ -311,7 +318,6 @@ class FlowChartExecutor {
311
318
  * ```
312
319
  */
313
320
  async resume(checkpoint, resumeInput, options) {
314
- var _a, _b, _c;
315
321
  this.lastCheckpoint = undefined;
316
322
  // ── Validate checkpoint structure (may come from untrusted external storage) ──
317
323
  if (!checkpoint ||
@@ -433,11 +439,11 @@ class FlowChartExecutor {
433
439
  }
434
440
  // Replace the leaf subflow's root with the resume chain so the
435
441
  // body runs from the pause point forward.
436
- subflowsOverride = { ...((_a = fc.subflows) !== null && _a !== void 0 ? _a : {}) };
442
+ subflowsOverride = { ...(fc.subflows ?? {}) };
437
443
  subflowsOverride[leafSubflowId] = { root: innerResumeChain };
438
444
  }
439
445
  const resumeInitialContext = checkpoint.sharedState;
440
- 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, {
446
+ this.traverser = this.createTraverser(options?.signal, undefined, options?.env, options?.maxDepth, {
441
447
  root: resumeRoot,
442
448
  initialContext: resumeInitialContext,
443
449
  preserveRecorders: true,
@@ -457,7 +463,7 @@ class FlowChartExecutor {
457
463
  if (this.combinedRecorder)
458
464
  this.combinedRecorder.onResume(flowResumeEvent);
459
465
  for (const r of this.flowRecorders)
460
- (_b = r.onResume) === null || _b === void 0 ? void 0 : _b.call(r, flowResumeEvent);
466
+ r.onResume?.(flowResumeEvent);
461
467
  const scopeResumeEvent = {
462
468
  stageName: pausedNode.name,
463
469
  stageId: pausedNode.id,
@@ -467,7 +473,7 @@ class FlowChartExecutor {
467
473
  timestamp: Date.now(),
468
474
  };
469
475
  for (const r of this.scopeRecorders)
470
- (_c = r.onResume) === null || _c === void 0 ? void 0 : _c.call(r, scopeResumeEvent);
476
+ r.onResume?.(scopeResumeEvent);
471
477
  try {
472
478
  return await this.traverser.execute();
473
479
  }
@@ -497,7 +503,6 @@ class FlowChartExecutor {
497
503
  * Handles subflow paths by drilling into registered subflows.
498
504
  */
499
505
  findNodeInGraph(stageId, subflowPath) {
500
- var _a;
501
506
  const fc = this.flowChartArgs.flowChart;
502
507
  if (subflowPath.length === 0) {
503
508
  // Top-level: DFS from root
@@ -506,7 +511,7 @@ class FlowChartExecutor {
506
511
  // Subflow: drill into the subflow chain, then search from the last subflow's root
507
512
  let subflowRoot;
508
513
  for (const sfId of subflowPath) {
509
- const subflow = (_a = fc.subflows) === null || _a === void 0 ? void 0 : _a[sfId];
514
+ const subflow = fc.subflows?.[sfId];
510
515
  if (!subflow)
511
516
  return undefined;
512
517
  subflowRoot = subflow.root;
@@ -596,6 +601,47 @@ class FlowChartExecutor {
596
601
  this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== recorder.id);
597
602
  this.scopeRecorders.push(recorder);
598
603
  }
604
+ // ─── Detach (T4) ─────────────────────────────────────────────────────────
605
+ //
606
+ // Bare-executor entry point for fire-and-forget child flowchart execution.
607
+ // Use from outside any chart (consumer code that wants to detach work
608
+ // without first running a parent chart). For detach FROM INSIDE a stage,
609
+ // use `scope.$detachAndJoinLater(...)` / `scope.$detachAndForget(...)` —
610
+ // those mint refIds from the calling stage's runtimeStageId for trace
611
+ // correlation; the bare-executor entries use a synthetic prefix
612
+ // (`__executor__`) instead.
613
+ /**
614
+ * Detach a child flowchart on the given driver and return a `DetachHandle`
615
+ * the caller can `wait()` on (Promise) or read `.status` from (sync).
616
+ *
617
+ * The driver is a REQUIRED first argument — there is no library-default,
618
+ * to keep the engine free of driver imports and to make the choice of
619
+ * scheduling algorithm explicit at the call site.
620
+ *
621
+ * @example
622
+ * ```typescript
623
+ * import { microtaskBatchDriver } from 'footprintjs/detach';
624
+ *
625
+ * const exec = new FlowChartExecutor(parentChart);
626
+ * const handle = exec.detachAndJoinLater(microtaskBatchDriver, telemetryChart, { event: 'x' });
627
+ * await handle.wait(); // optional
628
+ * ```
629
+ */
630
+ detachAndJoinLater(driver, child, input) {
631
+ return (0, spawn_js_1.detachAndJoinLater)(driver, child, input, '__executor__');
632
+ }
633
+ /**
634
+ * Detach a child flowchart on the given driver and DISCARD the handle.
635
+ * Use for telemetry exports / fire-and-forget side effects where the
636
+ * caller doesn't care about the result.
637
+ *
638
+ * Errors raised by the child still land on the (discarded) handle — they
639
+ * go silent unless surfaced through a recorder. For observable detach,
640
+ * prefer `detachAndJoinLater` and surface failures via `.wait().catch()`.
641
+ */
642
+ detachAndForget(driver, child, input) {
643
+ (0, spawn_js_1.detachAndForget)(driver, child, input, '__executor__');
644
+ }
599
645
  /** Detach all scope Recorders with the given ID. */
600
646
  detachRecorder(id) {
601
647
  this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);
@@ -773,17 +819,16 @@ class FlowChartExecutor {
773
819
  return recorders.length > 0 ? recorders : undefined;
774
820
  }
775
821
  async run(options) {
776
- var _a, _b;
777
- let signal = options === null || options === void 0 ? void 0 : options.signal;
822
+ let signal = options?.signal;
778
823
  let timeoutId;
779
824
  // Create an internal AbortController for timeoutMs
780
- if ((options === null || options === void 0 ? void 0 : options.timeoutMs) && !signal) {
825
+ if (options?.timeoutMs && !signal) {
781
826
  const controller = new AbortController();
782
827
  signal = controller.signal;
783
828
  timeoutId = setTimeout(() => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)), options.timeoutMs);
784
829
  }
785
830
  // Validate input against inputSchema if both are present
786
- let validatedInput = options === null || options === void 0 ? void 0 : options.input;
831
+ let validatedInput = options?.input;
787
832
  if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {
788
833
  validatedInput = (0, validateInput_js_1.validateInput)(this.flowChartArgs.flowChart.inputSchema, validatedInput);
789
834
  }
@@ -791,16 +836,16 @@ class FlowChartExecutor {
791
836
  // cross-run accumulation. The combinedRecorder is NOT cleared here — createTraverser() always
792
837
  // creates a fresh CombinedNarrativeRecorder instance on each run, so stale state is never an issue.
793
838
  for (const r of this.flowRecorders) {
794
- (_a = r.clear) === null || _a === void 0 ? void 0 : _a.call(r);
839
+ r.clear?.();
795
840
  }
796
841
  for (const r of this.scopeRecorders) {
797
- (_b = r.clear) === null || _b === void 0 ? void 0 : _b.call(r);
842
+ r.clear?.();
798
843
  }
799
844
  this.lastCheckpoint = undefined;
800
845
  this._executionCounter = { value: 0 }; // Reset counter on fresh run
801
846
  this._hasRunBefore = true; // mark so a later resume() takes the
802
847
  // same-executor branch (reuse runtime, accumulate execution tree).
803
- this.traverser = this.createTraverser(signal, validatedInput, options === null || options === void 0 ? void 0 : options.env, options === null || options === void 0 ? void 0 : options.maxDepth);
848
+ this.traverser = this.createTraverser(signal, validatedInput, options?.env, options?.maxDepth);
804
849
  try {
805
850
  return await this.traverser.execute();
806
851
  }
@@ -925,9 +970,8 @@ class FlowChartExecutor {
925
970
  * Returns empty array if no ManifestFlowRecorder is attached.
926
971
  */
927
972
  getSubflowManifest() {
928
- var _a;
929
973
  const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder_js_1.ManifestFlowRecorder);
930
- return (_a = recorder === null || recorder === void 0 ? void 0 : recorder.getManifest()) !== null && _a !== void 0 ? _a : [];
974
+ return recorder?.getManifest() ?? [];
931
975
  }
932
976
  /**
933
977
  * Returns the full spec for a dynamically-registered subflow.
@@ -935,8 +979,8 @@ class FlowChartExecutor {
935
979
  */
936
980
  getSubflowSpec(subflowId) {
937
981
  const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder_js_1.ManifestFlowRecorder);
938
- return recorder === null || recorder === void 0 ? void 0 : recorder.getSpec(subflowId);
982
+ return recorder?.getSpec(subflowId);
939
983
  }
940
984
  }
941
985
  exports.FlowChartExecutor = FlowChartExecutor;
942
- //# 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 { FlowChart } from '../builder/types.js';\nimport type { CombinedNarrativeRecorderOptions } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.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 { buildRuntimeStageId } from '../engine/runtimeStageId.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExecutorResult,\n  type ExtractorError,\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 { CombinedRecorder } from '../recorder/CombinedRecorder.js';\nimport { hasEmitRecorderMethods, hasFlowRecorderMethods, hasRecorderMethods } from '../recorder/CombinedRecorder.js';\nimport type { EmitRecorder } from '../recorder/EmitRecorder.js';\nimport { isDevMode } from '../scope/detectCircular.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  /** Shared execution counter — survives pause/resume. Reset on fresh run(). */\n  private _executionCounter = { value: 0 };\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   * `true` once `run()` (or a previous `resume()`) has executed on\n   * this instance. `resume()` branches on it:\n   *\n   *   • true  → reuse the constructor-time runtime (same-executor\n   *             continuity: execution tree, recorders, narrative\n   *             accumulate across pause/resume cycles)\n   *   • false → seed a fresh runtime from `checkpoint.sharedState`\n   *             (cross-executor / cross-process resume: new instance\n   *             reconstructed from a serialized checkpoint)\n   *\n   * Without this flag, fresh executors silently discarded the\n   * checkpoint's sharedState and resume handlers couldn't read pre-pause\n   * scope. See `test/lib/pause/cross-executor-resume.test.ts`.\n   */\n  private _hasRunBefore = false;\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      /** Per-subflow scope captures from a checkpoint — passed through\n       *  to HandlerDeps so SubflowExecutor can re-seed nested runtimes\n       *  on the resume path. Undefined on normal run() paths. */\n      subflowStatesForResume?: Record<string, Record<string, unknown>>;\n      /** Resume-only override of the subflows dict — substitutes the\n       *  leaf subflow's root with a resume chain so the subflow body\n       *  picks up at the pause point. Other entries pass through\n       *  unchanged. */\n      subflowsOverride?: Record<string, { root: StageNode<TOut, TScope> }>;\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    // When a redaction policy is configured, maintain a parallel redacted\n    // mirror of `globalStore` during traversal. Each commit applies the\n    // already-computed redacted patches — same ones fed to the event log —\n    // so `getSnapshot({ redact: true })` returns a scrubbed sharedState at\n    // zero post-pass cost. Skipped when no policy exists (zero allocation).\n    if (this.redactionPolicy) {\n      runtime.enableRedactedMirror();\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,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n      executionCounter: this._executionCounter,\n      ...(overrides?.subflowsOverride && { subflows: overrides.subflowsOverride }),\n      ...(overrides?.subflowStatesForResume && {\n        subflowStatesForResume: overrides.subflowStatesForResume,\n      }),\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.\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    // Determine continuation: for branch children (decider/selector),\n    // pausedNode.next is undefined. The checkpoint's\n    // continuationStageId (collected during traversal bubble-up)\n    // points to the invoker's next node.\n    //\n    // For pauses inside a subflow, the continuation lives INSIDE the\n    // leaf subflow (e.g., the loop target back to `messages`). Search\n    // the leaf subflow first; fall back to top-level for root-level\n    // pauses.\n    const sfStates = checkpoint.subflowStates;\n    const leafSubflowId =\n      checkpoint.subflowPath.length > 0 ? checkpoint.subflowPath[checkpoint.subflowPath.length - 1] : undefined;\n    let continuationNext = pausedNode.next;\n    if (!continuationNext && checkpoint.continuationStageId) {\n      // Search leaf subflow first (loop targets / branch joins live there),\n      // then fall back to top level.\n      continuationNext = leafSubflowId\n        ? this.findNodeInGraph(checkpoint.continuationStageId, checkpoint.subflowPath)\n        : undefined;\n      if (!continuationNext) {\n        continuationNext = this.findNodeInGraph(checkpoint.continuationStageId, []);\n      }\n    }\n\n    // The \"inner\" resume chain: resumeFn → continuation. This is what\n    // runs INSIDE the leaf subflow's body. For a root-level pause\n    // (subflowPath empty), this is also the top-level resume root.\n    const innerResumeChain: StageNode<TOut, TScope> = {\n      name: pausedNode.name,\n      id: pausedNode.id,\n      description: pausedNode.description,\n      fn: resumeStageFn,\n      next: continuationNext,\n    };\n\n    // Don't clear recorders — resume continues from previous state.\n    // Narrative, metrics, debug entries accumulate across pause/resume.\n    //\n    // Two-mode resume:\n    //   • Same-executor (run() previously called on THIS instance):\n    //     reuse the existing runtime so the execution tree continues\n    //     from the pause point and recorders/narrative accumulate.\n    //   • Cross-executor (fresh executor reconstructed from a stored\n    //     checkpoint): seed a NEW runtime from `checkpoint.sharedState`\n    //     so resume handlers can read pre-pause scope. The execution\n    //     tree starts at the resume node — we don't have the previous\n    //     traversal's tree on a fresh process anyway.\n    const sameExecutor = this._hasRunBefore;\n    const existingRuntime = sameExecutor\n      ? (this.traverser.getRuntime() as InstanceType<typeof ExecutionRuntime>)\n      : undefined;\n    this._hasRunBefore = true; // any path that resumes counts as a run\n\n    // Pick the resume root + initial context.\n    //\n    //   ROOT-LEVEL PAUSE (subflowPath empty):\n    //     resume root = innerResumeChain (run resumeFn at top level).\n    //     initialContext = checkpoint.sharedState.\n    //\n    //   SUBFLOW-NESTED PAUSE (subflowPath non-empty):\n    //     The pause was INSIDE a subflow's body. To run the subflow's\n    //     outputMapper and the parent's continuation, we have to enter\n    //     through the OUTER MOUNT (the parent's node that mounts the\n    //     leaf subflow). We swap the leaf subflow's root with\n    //     innerResumeChain so SubflowExecutor:\n    //       1. enters the subflow boundary,\n    //       2. seeds the nested runtime from subflowStates[leaf]\n    //          (skipping the inputMapper — see SubflowExecutor.ts),\n    //       3. runs the resumeFn → continuation chain,\n    //       4. runs the outputMapper at exit,\n    //       5. parent traversal continues normally.\n    //\n    //     Cross-executor: initialContext = checkpoint.sharedState (the\n    //       parent's view at pause time — outputMapper writes back into it).\n    //     Same-executor: existingRuntime is reused; initialContext is moot\n    //       for the subflow frame (already in the runtime stack), but we\n    //       still pass sharedState for consistency.\n    const fc = this.flowChartArgs.flowChart;\n    let resumeRoot: StageNode<TOut, TScope> = innerResumeChain;\n    let subflowsOverride: Record<string, { root: StageNode<TOut, TScope> }> | undefined;\n    if (leafSubflowId !== undefined) {\n      // Find the OUTER mount node for the FIRST entry on the path.\n      // For single-level pauses, this is the only mount we need to\n      // enter through. For nested mounts the pattern would extend, but\n      // single-level covers all current use cases (Sequence(Agent),\n      // Conditional(Agent), Parallel branches with paused agents).\n      const outerSubflowId = checkpoint.subflowPath[0];\n      const outerMount = this.findMountInGraph(fc.root, outerSubflowId);\n      if (outerMount) {\n        resumeRoot = outerMount;\n      }\n      // Replace the leaf subflow's root with the resume chain so the\n      // body runs from the pause point forward.\n      subflowsOverride = { ...(fc.subflows ?? {}) };\n      subflowsOverride[leafSubflowId] = { root: innerResumeChain };\n    }\n    const resumeInitialContext = checkpoint.sharedState;\n\n    this.traverser = this.createTraverser(options?.signal, undefined, options?.env, options?.maxDepth, {\n      root: resumeRoot,\n      initialContext: resumeInitialContext,\n      preserveRecorders: true,\n      ...(existingRuntime ? { existingRuntime } : {}),\n      // Hand the per-subflow scope captures down to SubflowExecutor.\n      // Always present on a checkpoint — empty `{}` for root pauses.\n      subflowStatesForResume: sfStates,\n      ...(subflowsOverride && { subflowsOverride }),\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      runtimeStageId: buildRuntimeStageId(pausedNode.id, this._executionCounter.value),\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          subflowStates: error.subflowStates,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          ...(error.invokerStageId && { invokerStageId: error.invokerStageId }),\n          ...(error.continuationStageId && { continuationStageId: error.continuationStageId }),\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  /**\n   * Find the mount node (the node that mounts a subflow boundary)\n   * for a given subflowId, by DFS from `start`. Used by `resume()` to\n   * locate the OUTER node we have to enter through so the subflow's\n   * outputMapper and parent continuation execute.\n   *\n   * Cycle-safe via visited set. Returns the first match (DFS order).\n   */\n  private findMountInGraph(\n    start: StageNode<TOut, TScope>,\n    subflowId: string,\n    visited = new Set<string>(),\n  ): StageNode<TOut, TScope> | undefined {\n    if (start.isLoopRef) return undefined;\n    if (visited.has(start.id)) return undefined;\n    visited.add(start.id);\n    if (start.subflowId === subflowId) return start;\n    if (start.children) {\n      for (const child of start.children) {\n        const found = this.findMountInGraph(child, subflowId, visited);\n        if (found) return found;\n      }\n    }\n    if (start.next) return this.findMountInGraph(start.next, subflowId, visited);\n    return undefined;\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  // ─── Combined Recorder Management ───\n\n  /**\n   * Attach a recorder that may observe multiple event streams (scope\n   * data-flow, control-flow, or both). Detects at runtime which streams the\n   * recorder has methods for and routes it to the correct internal channels.\n   *\n   * Preferred over calling `attachRecorder` and `attachFlowRecorder`\n   * separately, because forgetting one of the two is a silent foot-gun —\n   * half your events never fire and there is no runtime warning. With\n   * `attachCombinedRecorder` the library guarantees the recorder's declared\n   * methods all fire, and adds no overhead versus two explicit calls.\n   *\n   * ## Idempotency\n   *\n   * Idempotent by `id` across ALL channels — re-attaching with the same `id`\n   * replaces the previous instance everywhere it was registered. Mixing\n   * `attachCombinedRecorder(x)` with a prior `attachRecorder(y)` or\n   * `attachFlowRecorder(y)` that share `x.id === y.id` is also safe: the\n   * combined attach replaces the single-channel registration on whichever\n   * channel(s) `x` has methods for. No duplicate firings occur.\n   *\n   * ## Narrative activation\n   *\n   * If the recorder has any control-flow methods, `enableNarrative()` is\n   * called as a side effect (the narrative subsystem is required to emit\n   * control-flow events). Data-flow-only recorders do NOT activate the\n   * narrative.\n   *\n   * ## Detection rule\n   *\n   * Only **own** event methods count (see `hasRecorderMethods`). Methods\n   * inherited via the prototype chain are ignored — this protects against\n   * accidental `Object.prototype` pollution attaching handlers you never\n   * declared. A recorder that provides only `clear`/`toSnapshot` is a\n   * no-op and emits a dev-mode warning to surface the likely mistake.\n   *\n   * @example\n   * ```typescript\n   * const audit: CombinedRecorder = {\n   *   id: 'audit',\n   *   onWrite: (e) => log('scope write', e.key),\n   *   onDecision: (e) => log('routed to', e.chosen),\n   * };\n   * executor.attachCombinedRecorder(audit);\n   * ```\n   */\n  attachCombinedRecorder(recorder: CombinedRecorder): void {\n    const hasData = hasRecorderMethods(recorder);\n    const hasFlow = hasFlowRecorderMethods(recorder);\n    const hasEmit = hasEmitRecorderMethods(recorder);\n\n    // Emit recorders live on the SAME channel as data-flow recorders\n    // (ScopeFacade iterates `_recorders` for onEmit dispatch). So\n    // attachEmitRecorder internally calls attachRecorder — but we want to\n    // avoid double-attach when the recorder implements BOTH onEmit AND\n    // other Recorder methods. Short-circuit: if hasData OR hasEmit, the\n    // recorder lands on the scope-recorder list exactly once.\n    if (hasData || hasEmit) this.attachRecorder(recorder as Recorder);\n    if (hasFlow) this.attachFlowRecorder(recorder as FlowRecorder);\n\n    if (!hasData && !hasFlow && !hasEmit && isDevMode()) {\n      // Dev-mode only: silent skips are invisible and produce hard-to-debug\n      // \"why didn't my recorder fire\" reports. Per library convention, gated\n      // on the central isDevMode() flag (not process.env) so consumers can\n      // control dev tooling centrally via enableDevMode()/disableDevMode().\n      // eslint-disable-next-line no-console\n      console.warn(\n        `[footprintjs] attachCombinedRecorder: recorder '${recorder.id}' has ` +\n          'no observer event methods — nothing to attach. Did you forget to ' +\n          'add an on* handler (onWrite, onDecision, onSubflowEntry, ...)? ' +\n          'Note: only OWN properties count; methods on the prototype chain ' +\n          'are ignored on purpose.',\n      );\n    }\n  }\n\n  /**\n   * Detach a combined recorder from all channels it was attached to.\n   * Safe to call if the recorder was only on one channel or never attached.\n   */\n  detachCombinedRecorder(id: string): void {\n    this.detachRecorder(id);\n    this.detachFlowRecorder(id);\n  }\n\n  // ─── Emit Recorder Management (Phase 3) ───\n\n  /**\n   * Attach an `EmitRecorder` — an observer for consumer-emitted structured\n   * events fired via `scope.$emit(name, payload)`.\n   *\n   * Internally, emit recorders share the scope-recorder channel because\n   * emit events fire from inside `ScopeFacade` during stage execution,\n   * same timing as `onRead`/`onWrite`. This method is a convenience that\n   * delegates to `attachRecorder` — consumers can also use\n   * `attachRecorder` directly for a recorder that implements BOTH\n   * `onWrite` and `onEmit`. Either approach places the recorder on the\n   * same underlying list, so `onEmit` fires exactly once per event.\n   *\n   * **Idempotent by `id`:** replaces existing recorder with same `id`.\n   *\n   * @example\n   * ```typescript\n   * executor.attachEmitRecorder({\n   *   id: 'token-meter',\n   *   onEmit: (e) => {\n   *     if (e.name === 'agentfootprint.llm.tokens') trackTokens(e.payload);\n   *   },\n   * });\n   * ```\n   */\n  attachEmitRecorder(recorder: EmitRecorder): void {\n    this.attachRecorder(recorder as Recorder);\n  }\n\n  /** Detach an `EmitRecorder` by id. Safe to call if never attached. */\n  detachEmitRecorder(id: string): void {\n    this.detachRecorder(id);\n  }\n\n  /**\n   * Returns a defensive copy of attached recorders filtered to those that\n   * implement `onEmit`. Useful for inspection during testing.\n   */\n  getEmitRecorders(): EmitRecorder[] {\n    return this.scopeRecorders.filter(\n      (r): r is EmitRecorder => typeof (r as { onEmit?: unknown }).onEmit === 'function',\n    );\n  }\n\n  /**\n   * Returns structured narrative entries — the single public narrative API.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and\n   * depth. Consumers render however they want; call `.map(e => e.text)`\n   * if a flat `string[]` is needed locally.\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,\n   * includes the CombinedNarrativeRecorder (which builds merged flow+data\n   * entries inline). 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    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\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._executionCounter = { value: 0 }; // Reset counter on fresh run\n    this._hasRunBefore = true; // mark so a later resume() takes the\n    // same-executor branch (reuse runtime, accumulate execution tree).\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        // Subflow scope capture survives ONLY on the signal — the\n        // nested runtimes are GC'd as the stack unwinds. Promote to\n        // the checkpoint here so cross-executor resume can restore\n        // pre-pause subflow scope (e.g. an Agent's `scope.history`).\n        // Empty `{}` for root-level pauses (no subflows entered).\n        this.lastCheckpoint = {\n          sharedState: snapshot.sharedState,\n          executionTree: snapshot.executionTree,\n          pausedStageId: error.stageId,\n          subflowPath: error.subflowPath,\n          pauseData: error.pauseData,\n          subflowStates: error.subflowStates,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          // Invoker context — collected during traversal bubble-up (not tree-walked)\n          ...(error.invokerStageId && { invokerStageId: error.invokerStageId }),\n          ...(error.continuationStageId && { continuationStageId: error.continuationStageId }),\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  /**\n   * Returns the runtime snapshot.\n   *\n   * @param options.redact  When `true`, `sharedState` comes from the parallel\n   *   redacted mirror (if maintained — see `setRedactionPolicy`). This is\n   *   the safe view for exporting traces externally (paste into a viewer,\n   *   share with support). When no redaction policy is configured the\n   *   redacted mirror is not maintained, so this flag is a no-op —\n   *   `sharedState` is the raw working memory either way. Default `false`.\n   *\n   *   The commit log is already redacted at write-time regardless of this\n   *   flag, and the execution tree carries only structural metadata.\n   */\n  getSnapshot(options?: { redact?: boolean }): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot(options) 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 snap = r.toSnapshot();\n        recorderSnapshots.push({\n          id: r.id,\n          name: snap.name,\n          description: snap.description,\n          preferredOperation: snap.preferredOperation,\n          data: snap.data,\n        });\n      }\n    }\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const snap = r.toSnapshot();\n        recorderSnapshots.push({\n          id: r.id,\n          name: snap.name,\n          description: snap.description,\n          preferredOperation: snap.preferredOperation,\n          data: snap.data,\n        });\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"]}
986
+ //# 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 { FlowChart } from '../builder/types.js';\nimport { detachAndForget as _detachAndForget, detachAndJoinLater as _detachAndJoinLater } from '../detach/spawn.js';\nimport type { CombinedNarrativeRecorderOptions } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.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 { buildRuntimeStageId } from '../engine/runtimeStageId.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExecutorResult,\n  type ExtractorError,\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 { CombinedRecorder } from '../recorder/CombinedRecorder.js';\nimport { hasEmitRecorderMethods, hasFlowRecorderMethods, hasRecorderMethods } from '../recorder/CombinedRecorder.js';\nimport type { EmitRecorder } from '../recorder/EmitRecorder.js';\nimport { isDevMode } from '../scope/detectCircular.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  /** Shared execution counter — survives pause/resume. Reset on fresh run(). */\n  private _executionCounter = { value: 0 };\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   * `true` once `run()` (or a previous `resume()`) has executed on\n   * this instance. `resume()` branches on it:\n   *\n   *   • true  → reuse the constructor-time runtime (same-executor\n   *             continuity: execution tree, recorders, narrative\n   *             accumulate across pause/resume cycles)\n   *   • false → seed a fresh runtime from `checkpoint.sharedState`\n   *             (cross-executor / cross-process resume: new instance\n   *             reconstructed from a serialized checkpoint)\n   *\n   * Without this flag, fresh executors silently discarded the\n   * checkpoint's sharedState and resume handlers couldn't read pre-pause\n   * scope. See `test/lib/pause/cross-executor-resume.test.ts`.\n   */\n  private _hasRunBefore = false;\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      /** Per-subflow scope captures from a checkpoint — passed through\n       *  to HandlerDeps so SubflowExecutor can re-seed nested runtimes\n       *  on the resume path. Undefined on normal run() paths. */\n      subflowStatesForResume?: Record<string, Record<string, unknown>>;\n      /** Resume-only override of the subflows dict — substitutes the\n       *  leaf subflow's root with a resume chain so the subflow body\n       *  picks up at the pause point. Other entries pass through\n       *  unchanged. */\n      subflowsOverride?: Record<string, { root: StageNode<TOut, TScope> }>;\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    // When a redaction policy is configured, maintain a parallel redacted\n    // mirror of `globalStore` during traversal. Each commit applies the\n    // already-computed redacted patches — same ones fed to the event log —\n    // so `getSnapshot({ redact: true })` returns a scrubbed sharedState at\n    // zero post-pass cost. Skipped when no policy exists (zero allocation).\n    if (this.redactionPolicy) {\n      runtime.enableRedactedMirror();\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,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n      executionCounter: this._executionCounter,\n      ...(overrides?.subflowsOverride && { subflows: overrides.subflowsOverride }),\n      ...(overrides?.subflowStatesForResume && {\n        subflowStatesForResume: overrides.subflowStatesForResume,\n      }),\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.\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    // Determine continuation: for branch children (decider/selector),\n    // pausedNode.next is undefined. The checkpoint's\n    // continuationStageId (collected during traversal bubble-up)\n    // points to the invoker's next node.\n    //\n    // For pauses inside a subflow, the continuation lives INSIDE the\n    // leaf subflow (e.g., the loop target back to `messages`). Search\n    // the leaf subflow first; fall back to top-level for root-level\n    // pauses.\n    const sfStates = checkpoint.subflowStates;\n    const leafSubflowId =\n      checkpoint.subflowPath.length > 0 ? checkpoint.subflowPath[checkpoint.subflowPath.length - 1] : undefined;\n    let continuationNext = pausedNode.next;\n    if (!continuationNext && checkpoint.continuationStageId) {\n      // Search leaf subflow first (loop targets / branch joins live there),\n      // then fall back to top level.\n      continuationNext = leafSubflowId\n        ? this.findNodeInGraph(checkpoint.continuationStageId, checkpoint.subflowPath)\n        : undefined;\n      if (!continuationNext) {\n        continuationNext = this.findNodeInGraph(checkpoint.continuationStageId, []);\n      }\n    }\n\n    // The \"inner\" resume chain: resumeFn → continuation. This is what\n    // runs INSIDE the leaf subflow's body. For a root-level pause\n    // (subflowPath empty), this is also the top-level resume root.\n    const innerResumeChain: StageNode<TOut, TScope> = {\n      name: pausedNode.name,\n      id: pausedNode.id,\n      description: pausedNode.description,\n      fn: resumeStageFn,\n      next: continuationNext,\n    };\n\n    // Don't clear recorders — resume continues from previous state.\n    // Narrative, metrics, debug entries accumulate across pause/resume.\n    //\n    // Two-mode resume:\n    //   • Same-executor (run() previously called on THIS instance):\n    //     reuse the existing runtime so the execution tree continues\n    //     from the pause point and recorders/narrative accumulate.\n    //   • Cross-executor (fresh executor reconstructed from a stored\n    //     checkpoint): seed a NEW runtime from `checkpoint.sharedState`\n    //     so resume handlers can read pre-pause scope. The execution\n    //     tree starts at the resume node — we don't have the previous\n    //     traversal's tree on a fresh process anyway.\n    const sameExecutor = this._hasRunBefore;\n    const existingRuntime = sameExecutor\n      ? (this.traverser.getRuntime() as InstanceType<typeof ExecutionRuntime>)\n      : undefined;\n    this._hasRunBefore = true; // any path that resumes counts as a run\n\n    // Pick the resume root + initial context.\n    //\n    //   ROOT-LEVEL PAUSE (subflowPath empty):\n    //     resume root = innerResumeChain (run resumeFn at top level).\n    //     initialContext = checkpoint.sharedState.\n    //\n    //   SUBFLOW-NESTED PAUSE (subflowPath non-empty):\n    //     The pause was INSIDE a subflow's body. To run the subflow's\n    //     outputMapper and the parent's continuation, we have to enter\n    //     through the OUTER MOUNT (the parent's node that mounts the\n    //     leaf subflow). We swap the leaf subflow's root with\n    //     innerResumeChain so SubflowExecutor:\n    //       1. enters the subflow boundary,\n    //       2. seeds the nested runtime from subflowStates[leaf]\n    //          (skipping the inputMapper — see SubflowExecutor.ts),\n    //       3. runs the resumeFn → continuation chain,\n    //       4. runs the outputMapper at exit,\n    //       5. parent traversal continues normally.\n    //\n    //     Cross-executor: initialContext = checkpoint.sharedState (the\n    //       parent's view at pause time — outputMapper writes back into it).\n    //     Same-executor: existingRuntime is reused; initialContext is moot\n    //       for the subflow frame (already in the runtime stack), but we\n    //       still pass sharedState for consistency.\n    const fc = this.flowChartArgs.flowChart;\n    let resumeRoot: StageNode<TOut, TScope> = innerResumeChain;\n    let subflowsOverride: Record<string, { root: StageNode<TOut, TScope> }> | undefined;\n    if (leafSubflowId !== undefined) {\n      // Find the OUTER mount node for the FIRST entry on the path.\n      // For single-level pauses, this is the only mount we need to\n      // enter through. For nested mounts the pattern would extend, but\n      // single-level covers all current use cases (Sequence(Agent),\n      // Conditional(Agent), Parallel branches with paused agents).\n      const outerSubflowId = checkpoint.subflowPath[0];\n      const outerMount = this.findMountInGraph(fc.root, outerSubflowId);\n      if (outerMount) {\n        resumeRoot = outerMount;\n      }\n      // Replace the leaf subflow's root with the resume chain so the\n      // body runs from the pause point forward.\n      subflowsOverride = { ...(fc.subflows ?? {}) };\n      subflowsOverride[leafSubflowId] = { root: innerResumeChain };\n    }\n    const resumeInitialContext = checkpoint.sharedState;\n\n    this.traverser = this.createTraverser(options?.signal, undefined, options?.env, options?.maxDepth, {\n      root: resumeRoot,\n      initialContext: resumeInitialContext,\n      preserveRecorders: true,\n      ...(existingRuntime ? { existingRuntime } : {}),\n      // Hand the per-subflow scope captures down to SubflowExecutor.\n      // Always present on a checkpoint — empty `{}` for root pauses.\n      subflowStatesForResume: sfStates,\n      ...(subflowsOverride && { subflowsOverride }),\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      runtimeStageId: buildRuntimeStageId(pausedNode.id, this._executionCounter.value),\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          subflowStates: error.subflowStates,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          ...(error.invokerStageId && { invokerStageId: error.invokerStageId }),\n          ...(error.continuationStageId && { continuationStageId: error.continuationStageId }),\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  /**\n   * Find the mount node (the node that mounts a subflow boundary)\n   * for a given subflowId, by DFS from `start`. Used by `resume()` to\n   * locate the OUTER node we have to enter through so the subflow's\n   * outputMapper and parent continuation execute.\n   *\n   * Cycle-safe via visited set. Returns the first match (DFS order).\n   */\n  private findMountInGraph(\n    start: StageNode<TOut, TScope>,\n    subflowId: string,\n    visited = new Set<string>(),\n  ): StageNode<TOut, TScope> | undefined {\n    if (start.isLoopRef) return undefined;\n    if (visited.has(start.id)) return undefined;\n    visited.add(start.id);\n    if (start.subflowId === subflowId) return start;\n    if (start.children) {\n      for (const child of start.children) {\n        const found = this.findMountInGraph(child, subflowId, visited);\n        if (found) return found;\n      }\n    }\n    if (start.next) return this.findMountInGraph(start.next, subflowId, visited);\n    return undefined;\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 (T4) ─────────────────────────────────────────────────────────\n  //\n  // Bare-executor entry point for fire-and-forget child flowchart execution.\n  // Use from outside any chart (consumer code that wants to detach work\n  // without first running a parent chart). For detach FROM INSIDE a stage,\n  // use `scope.$detachAndJoinLater(...)` / `scope.$detachAndForget(...)` —\n  // those mint refIds from the calling stage's runtimeStageId for trace\n  // correlation; the bare-executor entries use a synthetic prefix\n  // (`__executor__`) instead.\n\n  /**\n   * Detach a child flowchart on the given driver and return a `DetachHandle`\n   * the caller can `wait()` on (Promise) or read `.status` from (sync).\n   *\n   * The driver is a REQUIRED first argument — there is no library-default,\n   * to keep the engine free of driver imports and to make the choice of\n   * scheduling algorithm explicit at the call site.\n   *\n   * @example\n   * ```typescript\n   * import { microtaskBatchDriver } from 'footprintjs/detach';\n   *\n   * const exec = new FlowChartExecutor(parentChart);\n   * const handle = exec.detachAndJoinLater(microtaskBatchDriver, telemetryChart, { event: 'x' });\n   * await handle.wait(); // optional\n   * ```\n   */\n  detachAndJoinLater(\n    driver: import('../detach/types.js').DetachDriver,\n    child: import('../builder/types.js').FlowChart,\n    input?: unknown,\n  ): import('../detach/types.js').DetachHandle {\n    return _detachAndJoinLater(driver, child, input, '__executor__');\n  }\n\n  /**\n   * Detach a child flowchart on the given driver and DISCARD the handle.\n   * Use for telemetry exports / fire-and-forget side effects where the\n   * caller doesn't care about the result.\n   *\n   * Errors raised by the child still land on the (discarded) handle — they\n   * go silent unless surfaced through a recorder. For observable detach,\n   * prefer `detachAndJoinLater` and surface failures via `.wait().catch()`.\n   */\n  detachAndForget(\n    driver: import('../detach/types.js').DetachDriver,\n    child: import('../builder/types.js').FlowChart,\n    input?: unknown,\n  ): void {\n    _detachAndForget(driver, child, input, '__executor__');\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  // ─── Combined Recorder Management ───\n\n  /**\n   * Attach a recorder that may observe multiple event streams (scope\n   * data-flow, control-flow, or both). Detects at runtime which streams the\n   * recorder has methods for and routes it to the correct internal channels.\n   *\n   * Preferred over calling `attachRecorder` and `attachFlowRecorder`\n   * separately, because forgetting one of the two is a silent foot-gun —\n   * half your events never fire and there is no runtime warning. With\n   * `attachCombinedRecorder` the library guarantees the recorder's declared\n   * methods all fire, and adds no overhead versus two explicit calls.\n   *\n   * ## Idempotency\n   *\n   * Idempotent by `id` across ALL channels — re-attaching with the same `id`\n   * replaces the previous instance everywhere it was registered. Mixing\n   * `attachCombinedRecorder(x)` with a prior `attachRecorder(y)` or\n   * `attachFlowRecorder(y)` that share `x.id === y.id` is also safe: the\n   * combined attach replaces the single-channel registration on whichever\n   * channel(s) `x` has methods for. No duplicate firings occur.\n   *\n   * ## Narrative activation\n   *\n   * If the recorder has any control-flow methods, `enableNarrative()` is\n   * called as a side effect (the narrative subsystem is required to emit\n   * control-flow events). Data-flow-only recorders do NOT activate the\n   * narrative.\n   *\n   * ## Detection rule\n   *\n   * Only **own** event methods count (see `hasRecorderMethods`). Methods\n   * inherited via the prototype chain are ignored — this protects against\n   * accidental `Object.prototype` pollution attaching handlers you never\n   * declared. A recorder that provides only `clear`/`toSnapshot` is a\n   * no-op and emits a dev-mode warning to surface the likely mistake.\n   *\n   * @example\n   * ```typescript\n   * const audit: CombinedRecorder = {\n   *   id: 'audit',\n   *   onWrite: (e) => log('scope write', e.key),\n   *   onDecision: (e) => log('routed to', e.chosen),\n   * };\n   * executor.attachCombinedRecorder(audit);\n   * ```\n   */\n  attachCombinedRecorder(recorder: CombinedRecorder): void {\n    const hasData = hasRecorderMethods(recorder);\n    const hasFlow = hasFlowRecorderMethods(recorder);\n    const hasEmit = hasEmitRecorderMethods(recorder);\n\n    // Emit recorders live on the SAME channel as data-flow recorders\n    // (ScopeFacade iterates `_recorders` for onEmit dispatch). So\n    // attachEmitRecorder internally calls attachRecorder — but we want to\n    // avoid double-attach when the recorder implements BOTH onEmit AND\n    // other Recorder methods. Short-circuit: if hasData OR hasEmit, the\n    // recorder lands on the scope-recorder list exactly once.\n    if (hasData || hasEmit) this.attachRecorder(recorder as Recorder);\n    if (hasFlow) this.attachFlowRecorder(recorder as FlowRecorder);\n\n    if (!hasData && !hasFlow && !hasEmit && isDevMode()) {\n      // Dev-mode only: silent skips are invisible and produce hard-to-debug\n      // \"why didn't my recorder fire\" reports. Per library convention, gated\n      // on the central isDevMode() flag (not process.env) so consumers can\n      // control dev tooling centrally via enableDevMode()/disableDevMode().\n      // eslint-disable-next-line no-console\n      console.warn(\n        `[footprintjs] attachCombinedRecorder: recorder '${recorder.id}' has ` +\n          'no observer event methods — nothing to attach. Did you forget to ' +\n          'add an on* handler (onWrite, onDecision, onSubflowEntry, ...)? ' +\n          'Note: only OWN properties count; methods on the prototype chain ' +\n          'are ignored on purpose.',\n      );\n    }\n  }\n\n  /**\n   * Detach a combined recorder from all channels it was attached to.\n   * Safe to call if the recorder was only on one channel or never attached.\n   */\n  detachCombinedRecorder(id: string): void {\n    this.detachRecorder(id);\n    this.detachFlowRecorder(id);\n  }\n\n  // ─── Emit Recorder Management (Phase 3) ───\n\n  /**\n   * Attach an `EmitRecorder` — an observer for consumer-emitted structured\n   * events fired via `scope.$emit(name, payload)`.\n   *\n   * Internally, emit recorders share the scope-recorder channel because\n   * emit events fire from inside `ScopeFacade` during stage execution,\n   * same timing as `onRead`/`onWrite`. This method is a convenience that\n   * delegates to `attachRecorder` — consumers can also use\n   * `attachRecorder` directly for a recorder that implements BOTH\n   * `onWrite` and `onEmit`. Either approach places the recorder on the\n   * same underlying list, so `onEmit` fires exactly once per event.\n   *\n   * **Idempotent by `id`:** replaces existing recorder with same `id`.\n   *\n   * @example\n   * ```typescript\n   * executor.attachEmitRecorder({\n   *   id: 'token-meter',\n   *   onEmit: (e) => {\n   *     if (e.name === 'agentfootprint.llm.tokens') trackTokens(e.payload);\n   *   },\n   * });\n   * ```\n   */\n  attachEmitRecorder(recorder: EmitRecorder): void {\n    this.attachRecorder(recorder as Recorder);\n  }\n\n  /** Detach an `EmitRecorder` by id. Safe to call if never attached. */\n  detachEmitRecorder(id: string): void {\n    this.detachRecorder(id);\n  }\n\n  /**\n   * Returns a defensive copy of attached recorders filtered to those that\n   * implement `onEmit`. Useful for inspection during testing.\n   */\n  getEmitRecorders(): EmitRecorder[] {\n    return this.scopeRecorders.filter(\n      (r): r is EmitRecorder => typeof (r as { onEmit?: unknown }).onEmit === 'function',\n    );\n  }\n\n  /**\n   * Returns structured narrative entries — the single public narrative API.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and\n   * depth. Consumers render however they want; call `.map(e => e.text)`\n   * if a flat `string[]` is needed locally.\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,\n   * includes the CombinedNarrativeRecorder (which builds merged flow+data\n   * entries inline). 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    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\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._executionCounter = { value: 0 }; // Reset counter on fresh run\n    this._hasRunBefore = true; // mark so a later resume() takes the\n    // same-executor branch (reuse runtime, accumulate execution tree).\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        // Subflow scope capture survives ONLY on the signal — the\n        // nested runtimes are GC'd as the stack unwinds. Promote to\n        // the checkpoint here so cross-executor resume can restore\n        // pre-pause subflow scope (e.g. an Agent's `scope.history`).\n        // Empty `{}` for root-level pauses (no subflows entered).\n        this.lastCheckpoint = {\n          sharedState: snapshot.sharedState,\n          executionTree: snapshot.executionTree,\n          pausedStageId: error.stageId,\n          subflowPath: error.subflowPath,\n          pauseData: error.pauseData,\n          subflowStates: error.subflowStates,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          // Invoker context — collected during traversal bubble-up (not tree-walked)\n          ...(error.invokerStageId && { invokerStageId: error.invokerStageId }),\n          ...(error.continuationStageId && { continuationStageId: error.continuationStageId }),\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  /**\n   * Returns the runtime snapshot.\n   *\n   * @param options.redact  When `true`, `sharedState` comes from the parallel\n   *   redacted mirror (if maintained — see `setRedactionPolicy`). This is\n   *   the safe view for exporting traces externally (paste into a viewer,\n   *   share with support). When no redaction policy is configured the\n   *   redacted mirror is not maintained, so this flag is a no-op —\n   *   `sharedState` is the raw working memory either way. Default `false`.\n   *\n   *   The commit log is already redacted at write-time regardless of this\n   *   flag, and the execution tree carries only structural metadata.\n   */\n  getSnapshot(options?: { redact?: boolean }): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot(options) 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 snap = r.toSnapshot();\n        recorderSnapshots.push({\n          id: r.id,\n          name: snap.name,\n          description: snap.description,\n          preferredOperation: snap.preferredOperation,\n          data: snap.data,\n        });\n      }\n    }\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const snap = r.toSnapshot();\n        recorderSnapshots.push({\n          id: r.id,\n          name: snap.name,\n          description: snap.description,\n          preferredOperation: snap.preferredOperation,\n          data: snap.data,\n        });\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"]}