footprintjs 3.0.21 → 4.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (106) hide show
  1. package/CLAUDE.md +2 -0
  2. package/dist/advanced.js +20 -23
  3. package/dist/esm/advanced.js +2 -3
  4. package/dist/esm/lib/builder/FlowChartBuilder.js +41 -16
  5. package/dist/esm/lib/builder/typedFlowChart.js +10 -20
  6. package/dist/esm/lib/builder/types.js +1 -1
  7. package/dist/esm/lib/contract/defineContract.js +33 -7
  8. package/dist/esm/lib/contract/openapi.js +12 -33
  9. package/dist/esm/lib/contract/types.js +1 -1
  10. package/dist/esm/lib/decide/decide.js +33 -3
  11. package/dist/esm/lib/decide/types.js +1 -1
  12. package/dist/esm/lib/engine/handlers/NodeResolver.js +22 -7
  13. package/dist/esm/lib/engine/handlers/RuntimeStructureManager.js +35 -10
  14. package/dist/esm/lib/engine/handlers/SelectorHandler.js +2 -2
  15. package/dist/esm/lib/engine/handlers/StageRunner.js +2 -2
  16. package/dist/esm/lib/engine/handlers/SubflowInputMapper.js +2 -2
  17. package/dist/esm/lib/engine/index.js +1 -2
  18. package/dist/esm/lib/engine/narrative/CombinedNarrativeBuilder.js +1 -1
  19. package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +5 -3
  20. package/dist/esm/lib/engine/narrative/FlowRecorderDispatcher.js +41 -32
  21. package/dist/esm/lib/engine/narrative/NarrativeFlowRecorder.js +7 -12
  22. package/dist/esm/lib/engine/narrative/index.js +1 -2
  23. package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +361 -270
  24. package/dist/esm/lib/engine/types.js +1 -1
  25. package/dist/esm/lib/memory/StageContext.js +1 -10
  26. package/dist/esm/lib/memory/pathOps.js +31 -3
  27. package/dist/esm/lib/memory/utils.js +30 -10
  28. package/dist/esm/lib/runner/FlowChartExecutor.js +59 -12
  29. package/dist/esm/lib/runner/RunnableChart.js +25 -1
  30. package/dist/esm/lib/runner/getSubtreeSnapshot.js +17 -27
  31. package/dist/esm/lib/runner/index.js +1 -1
  32. package/dist/esm/lib/scope/ScopeFacade.js +36 -7
  33. package/dist/esm/lib/scope/index.js +1 -1
  34. package/dist/esm/lib/scope/providers/baseStateCompatible.js +2 -2
  35. package/dist/esm/lib/scope/providers/types.js +1 -1
  36. package/dist/esm/lib/scope/recorders/index.js +1 -1
  37. package/dist/esm/lib/scope/types.js +1 -1
  38. package/dist/lib/builder/FlowChartBuilder.js +41 -16
  39. package/dist/lib/builder/typedFlowChart.js +11 -22
  40. package/dist/lib/builder/types.js +1 -1
  41. package/dist/lib/contract/defineContract.js +33 -7
  42. package/dist/lib/contract/openapi.js +12 -33
  43. package/dist/lib/contract/types.js +1 -1
  44. package/dist/lib/decide/decide.js +33 -3
  45. package/dist/lib/decide/types.js +1 -1
  46. package/dist/lib/engine/handlers/NodeResolver.js +22 -7
  47. package/dist/lib/engine/handlers/RuntimeStructureManager.js +35 -10
  48. package/dist/lib/engine/handlers/SelectorHandler.js +2 -2
  49. package/dist/lib/engine/handlers/StageRunner.js +2 -2
  50. package/dist/lib/engine/handlers/SubflowInputMapper.js +2 -2
  51. package/dist/lib/engine/index.js +2 -4
  52. package/dist/lib/engine/narrative/CombinedNarrativeBuilder.js +1 -1
  53. package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +5 -3
  54. package/dist/lib/engine/narrative/FlowRecorderDispatcher.js +41 -32
  55. package/dist/lib/engine/narrative/NarrativeFlowRecorder.js +7 -12
  56. package/dist/lib/engine/narrative/index.js +2 -4
  57. package/dist/lib/engine/traversal/FlowchartTraverser.js +361 -270
  58. package/dist/lib/engine/types.js +1 -1
  59. package/dist/lib/memory/StageContext.js +1 -10
  60. package/dist/lib/memory/pathOps.js +31 -3
  61. package/dist/lib/memory/utils.js +30 -10
  62. package/dist/lib/runner/FlowChartExecutor.js +59 -12
  63. package/dist/lib/runner/RunnableChart.js +25 -1
  64. package/dist/lib/runner/getSubtreeSnapshot.js +17 -27
  65. package/dist/lib/runner/index.js +1 -1
  66. package/dist/lib/scope/ScopeFacade.js +36 -7
  67. package/dist/lib/scope/index.js +1 -1
  68. package/dist/lib/scope/providers/baseStateCompatible.js +2 -2
  69. package/dist/lib/scope/providers/types.js +1 -1
  70. package/dist/lib/scope/recorders/index.js +1 -1
  71. package/dist/lib/scope/types.js +1 -1
  72. package/dist/types/advanced.d.ts +2 -3
  73. package/dist/types/lib/builder/FlowChartBuilder.d.ts +2 -1
  74. package/dist/types/lib/builder/typedFlowChart.d.ts +9 -18
  75. package/dist/types/lib/builder/types.d.ts +12 -3
  76. package/dist/types/lib/contract/openapi.d.ts +9 -3
  77. package/dist/types/lib/contract/types.d.ts +13 -2
  78. package/dist/types/lib/decide/decide.d.ts +10 -0
  79. package/dist/types/lib/decide/types.d.ts +20 -0
  80. package/dist/types/lib/engine/handlers/NodeResolver.d.ts +10 -3
  81. package/dist/types/lib/engine/handlers/RuntimeStructureManager.d.ts +2 -1
  82. package/dist/types/lib/engine/index.d.ts +0 -1
  83. package/dist/types/lib/engine/narrative/CombinedNarrativeBuilder.d.ts +1 -1
  84. package/dist/types/lib/engine/narrative/FlowRecorderDispatcher.d.ts +3 -2
  85. package/dist/types/lib/engine/narrative/NarrativeFlowRecorder.d.ts +0 -1
  86. package/dist/types/lib/engine/narrative/index.d.ts +0 -1
  87. package/dist/types/lib/engine/traversal/FlowchartTraverser.d.ts +53 -0
  88. package/dist/types/lib/engine/types.d.ts +10 -3
  89. package/dist/types/lib/memory/StageContext.d.ts +0 -3
  90. package/dist/types/lib/memory/pathOps.d.ts +26 -2
  91. package/dist/types/lib/memory/utils.d.ts +16 -3
  92. package/dist/types/lib/runner/FlowChartExecutor.d.ts +78 -7
  93. package/dist/types/lib/runner/RunnableChart.d.ts +2 -0
  94. package/dist/types/lib/runner/index.d.ts +1 -0
  95. package/dist/types/lib/scope/ScopeFacade.d.ts +18 -1
  96. package/dist/types/lib/scope/index.d.ts +0 -1
  97. package/dist/types/lib/scope/providers/types.d.ts +1 -1
  98. package/dist/types/lib/scope/recorders/index.d.ts +0 -1
  99. package/dist/types/lib/scope/types.d.ts +7 -1
  100. package/package.json +1 -1
  101. package/dist/esm/lib/engine/narrative/ControlFlowNarrativeGenerator.js +0 -110
  102. package/dist/esm/lib/scope/recorders/NarrativeRecorder.js +0 -149
  103. package/dist/lib/engine/narrative/ControlFlowNarrativeGenerator.js +0 -114
  104. package/dist/lib/scope/recorders/NarrativeRecorder.js +0 -153
  105. package/dist/types/lib/engine/narrative/ControlFlowNarrativeGenerator.d.ts +0 -31
  106. package/dist/types/lib/scope/recorders/NarrativeRecorder.d.ts +0 -50
@@ -36,11 +36,32 @@ export class FlowchartTraverser {
36
36
  var _a, _b;
37
37
  // Execution state
38
38
  this.subflowResults = new Map();
39
+ /**
40
+ * Per-traverser set of lazy subflow IDs that have been resolved by THIS run.
41
+ * Used instead of writing `node.subflowResolver = undefined` back to the shared
42
+ * StageNode graph — avoids a race where a concurrent traverser clears the shared
43
+ * resolver before another traverser has finished using it.
44
+ */
45
+ this.resolvedLazySubflows = new Set();
46
+ /**
47
+ * Recursion depth counter for executeNode.
48
+ * Each recursive executeNode call increments this; decrements on exit (try/finally).
49
+ * Prevents call-stack overflow on infinite loops or excessively deep stage chains.
50
+ */
51
+ this._executeDepth = 0;
52
+ const maxDepth = (_a = opts.maxDepth) !== null && _a !== void 0 ? _a : FlowchartTraverser.MAX_EXECUTE_DEPTH;
53
+ if (maxDepth < 1)
54
+ throw new Error('FlowchartTraverser: maxDepth must be >= 1');
55
+ this._maxDepth = maxDepth;
39
56
  this.root = opts.root;
40
- this.stageMap = opts.stageMap;
57
+ // Shallow-copy stageMap and subflows so that lazy-resolution mutations
58
+ // (prefixed entries added during execution) stay scoped to THIS traverser
59
+ // and do not escape to the shared FlowChart object. Without the copy,
60
+ // concurrent FlowChartExecutor runs sharing the same FlowChart would race
61
+ // on these two mutable dictionaries.
62
+ this.stageMap = new Map(opts.stageMap);
41
63
  this.executionRuntime = opts.executionRuntime;
42
- // Always initialize as object so lazy resolution can mutate in place
43
- this.subflows = (_a = opts.subflows) !== null && _a !== void 0 ? _a : {};
64
+ this.subflows = opts.subflows ? { ...opts.subflows } : {};
44
65
  this.logger = opts.logger;
45
66
  this.signal = opts.signal;
46
67
  // Structure manager (deep-clones build-time structure)
@@ -49,8 +70,7 @@ export class FlowchartTraverser {
49
70
  // Extractor runner
50
71
  this.extractorRunner = new ExtractorRunner(opts.extractor, (_b = opts.enrichSnapshots) !== null && _b !== void 0 ? _b : false, this.executionRuntime, this.logger);
51
72
  // Narrative generator
52
- // When narrative is enabled with FlowRecorders, use FlowRecorderDispatcher.
53
- // When narrative is enabled without FlowRecorders, use legacy ControlFlowNarrativeGenerator.
73
+ // When narrative is enabled, use FlowRecorderDispatcher (with default NarrativeFlowRecorder or custom recorders).
54
74
  // When disabled, use NullControlFlowNarrativeGenerator (zero-cost).
55
75
  if (opts.narrativeEnabled) {
56
76
  const dispatcher = new FlowRecorderDispatcher();
@@ -71,8 +91,10 @@ export class FlowchartTraverser {
71
91
  }
72
92
  // Build shared deps bag
73
93
  const deps = this.createDeps(opts);
94
+ // Build O(1) node ID map from the root graph (avoids repeated DFS on every loopTo())
95
+ const nodeIdMap = this.buildNodeIdMap(opts.root);
74
96
  // Initialize handler modules
75
- this.nodeResolver = new NodeResolver(deps);
97
+ this.nodeResolver = new NodeResolver(deps, nodeIdMap);
76
98
  this.childrenExecutor = new ChildrenExecutor(deps, this.executeNode.bind(this));
77
99
  this.stageRunner = new StageRunner(deps);
78
100
  this.continuationResolver = new ContinuationResolver(deps, this.nodeResolver, (nodeId, count) => this.structureManager.updateIterationCount(nodeId, count));
@@ -86,7 +108,7 @@ export class FlowchartTraverser {
86
108
  stageMap: this.stageMap,
87
109
  root: this.root,
88
110
  executionRuntime: this.executionRuntime,
89
- ScopeFactory: opts.scopeFactory,
111
+ scopeFactory: opts.scopeFactory,
90
112
  subflows: this.subflows,
91
113
  throttlingErrorChecker: opts.throttlingErrorChecker,
92
114
  streamHandlers: opts.streamHandlers,
@@ -138,9 +160,37 @@ export class FlowchartTraverser {
138
160
  return this.flowRecorderDispatcher;
139
161
  }
140
162
  // ─────────────────────── Core Traversal ───────────────────────
163
+ /**
164
+ * Build an O(1) ID→node map from the root graph.
165
+ * Used by NodeResolver to avoid repeated DFS on every loopTo() call.
166
+ * Depth-guarded at MAX_EXECUTE_DEPTH to prevent infinite recursion on cyclic graphs.
167
+ */
168
+ buildNodeIdMap(root) {
169
+ const map = new Map();
170
+ const visit = (node, depth) => {
171
+ if (depth > FlowchartTraverser.MAX_EXECUTE_DEPTH)
172
+ return;
173
+ if (map.has(node.id))
174
+ return; // already visited (avoids infinite loops on cyclic refs)
175
+ map.set(node.id, node);
176
+ if (node.children) {
177
+ for (const child of node.children)
178
+ visit(child, depth + 1);
179
+ }
180
+ if (node.next)
181
+ visit(node.next, depth + 1);
182
+ };
183
+ visit(root, 0);
184
+ return map;
185
+ }
141
186
  getStageFn(node) {
142
187
  if (typeof node.fn === 'function')
143
188
  return node.fn;
189
+ // Primary: look up by id (stable identifier, keyed by FlowChartBuilder)
190
+ const byId = this.stageMap.get(node.id);
191
+ if (byId !== undefined)
192
+ return byId;
193
+ // Fallback: look up by name (supports hand-crafted stageMaps in tests and advanced use)
144
194
  return this.stageMap.get(node.name);
145
195
  }
146
196
  async executeStage(node, stageFunc, context, breakFn) {
@@ -152,294 +202,319 @@ export class FlowchartTraverser {
152
202
  */
153
203
  async executeNode(node, context, breakFlag, branchPath) {
154
204
  var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r, _s;
155
- // Attach builder metadata to context for snapshot enrichment
156
- if (node.description)
157
- context.description = node.description;
158
- if (node.isSubflowRoot && node.subflowId)
159
- context.subflowId = node.subflowId;
160
- // Build traversal context for recorder events created once per stage, shared by all events
161
- const traversalContext = {
162
- stageId: (_a = node.id) !== null && _a !== void 0 ? _a : context.stageId,
163
- stageName: node.name,
164
- parentStageId: (_b = context.parent) === null || _b === void 0 ? void 0 : _b.stageId,
165
- subflowId: context.subflowId,
166
- subflowPath: branchPath || undefined,
167
- depth: this.computeContextDepth(context),
168
- };
169
- // ─── Phase 0a: LAZY RESOLVE — deferred subflow resolution ───
170
- if (node.isSubflowRoot && node.subflowResolver && !this.subflows[node.subflowId]) {
171
- const resolved = node.subflowResolver();
172
- const prefixedRoot = this.prefixNodeTree(resolved.root, node.subflowId);
173
- // Register the resolved subflow (same path as eager registration)
174
- this.subflows[node.subflowId] = { root: prefixedRoot };
175
- // Merge stageMap entries
176
- for (const [key, fn] of resolved.stageMap) {
177
- const prefixedKey = `${node.subflowId}/${key}`;
178
- if (!this.stageMap.has(prefixedKey)) {
179
- this.stageMap.set(prefixedKey, fn);
180
- }
205
+ // ─── Recursion depth guard ───
206
+ // Each `await executeNode(...)` keeps the calling frame on the stack.
207
+ // Without a cap, an infinite loop or an excessively deep stage chain will
208
+ // eventually overflow the V8 call stack (~10 000 frames) with a cryptic
209
+ // "Maximum call stack size exceeded" error. We fail early with a clear
210
+ // message so users can diagnose the cause (infinite loop, missing break, etc.).
211
+ // The increment is inside `try` so `finally` always decrements — no fragile
212
+ // gap between check and try entry.
213
+ try {
214
+ if (++this._executeDepth > this._maxDepth) {
215
+ throw new Error(`FlowchartTraverser: maximum traversal depth exceeded (${this._maxDepth}). ` +
216
+ 'Check for infinite loops or missing break conditions in your flowchart. ' +
217
+ `Last stage: '${node.name}'. ` +
218
+ 'For loopTo() pipelines, consider adding a break condition or using RunOptions.maxDepth to raise the limit.');
181
219
  }
182
- // Merge nested subflows
183
- if (resolved.subflows) {
184
- for (const [key, def] of Object.entries(resolved.subflows)) {
220
+ // Attach builder metadata to context for snapshot enrichment
221
+ if (node.description)
222
+ context.description = node.description;
223
+ if (node.isSubflowRoot && node.subflowId)
224
+ context.subflowId = node.subflowId;
225
+ // Build traversal context for recorder events — created once per stage, shared by all events
226
+ const traversalContext = {
227
+ stageId: (_a = node.id) !== null && _a !== void 0 ? _a : context.stageId,
228
+ stageName: node.name,
229
+ parentStageId: (_b = context.parent) === null || _b === void 0 ? void 0 : _b.stageId,
230
+ subflowId: context.subflowId,
231
+ subflowPath: branchPath || undefined,
232
+ depth: this.computeContextDepth(context),
233
+ };
234
+ // ─── Phase 0a: LAZY RESOLVE — deferred subflow resolution ───
235
+ // Guard uses the per-traverser resolvedLazySubflows set (not the shared node) so
236
+ // concurrent traversers do not race on node.subflowResolver or clear it for each other.
237
+ if (node.isSubflowRoot && node.subflowResolver && !this.resolvedLazySubflows.has(node.subflowId)) {
238
+ const resolved = node.subflowResolver();
239
+ const prefixedRoot = this.prefixNodeTree(resolved.root, node.subflowId);
240
+ // Register the resolved subflow (same path as eager registration)
241
+ this.subflows[node.subflowId] = { root: prefixedRoot };
242
+ // Merge stageMap entries
243
+ for (const [key, fn] of resolved.stageMap) {
185
244
  const prefixedKey = `${node.subflowId}/${key}`;
186
- if (!this.subflows[prefixedKey]) {
187
- this.subflows[prefixedKey] = def;
245
+ if (!this.stageMap.has(prefixedKey)) {
246
+ this.stageMap.set(prefixedKey, fn);
188
247
  }
189
248
  }
249
+ // Merge nested subflows
250
+ if (resolved.subflows) {
251
+ for (const [key, def] of Object.entries(resolved.subflows)) {
252
+ const prefixedKey = `${node.subflowId}/${key}`;
253
+ if (!this.subflows[prefixedKey]) {
254
+ this.subflows[prefixedKey] = def;
255
+ }
256
+ }
257
+ }
258
+ // Update runtime structure with the now-resolved spec
259
+ this.structureManager.updateDynamicSubflow(node.id, node.subflowId, node.subflowName, resolved.buildTimeStructure);
260
+ // Mark as resolved for THIS traverser — per-traverser set prevents re-entry
261
+ // without mutating the shared StageNode graph (which would race concurrent traversers).
262
+ this.resolvedLazySubflows.add(node.subflowId);
190
263
  }
191
- // Update runtime structure with the now-resolved spec
192
- this.structureManager.updateDynamicSubflow(node.id, node.subflowId, node.subflowName, resolved.buildTimeStructure);
193
- // Clear resolver — resolved once
194
- node.subflowResolver = undefined;
195
- }
196
- // ─── Phase 0: CLASSIFY — subflow detection ───
197
- if (node.isSubflowRoot && node.subflowId) {
198
- const resolvedNode = this.nodeResolver.resolveSubflowReference(node);
199
- const previousSubflowId = this.extractorRunner.currentSubflowId;
200
- this.extractorRunner.currentSubflowId = node.subflowId;
201
- let subflowOutput;
202
- try {
203
- subflowOutput = await this.subflowExecutor.executeSubflow(resolvedNode, context, breakFlag, branchPath, this.subflowResults, traversalContext);
204
- }
205
- finally {
206
- this.extractorRunner.currentSubflowId = previousSubflowId;
207
- }
208
- const isReferenceBasedSubflow = resolvedNode !== node;
209
- const hasChildren = Boolean(node.children && node.children.length > 0);
210
- const shouldExecuteContinuation = isReferenceBasedSubflow || hasChildren;
211
- if (node.next && shouldExecuteContinuation) {
212
- const nextCtx = context.createNext(branchPath, node.next.name, node.next.id);
213
- return await this.executeNode(node.next, nextCtx, breakFlag, branchPath);
214
- }
215
- return subflowOutput;
216
- }
217
- const stageFunc = this.getStageFn(node);
218
- const hasStageFunction = Boolean(stageFunc);
219
- const isScopeBasedDecider = Boolean(node.deciderFn);
220
- const isScopeBasedSelector = Boolean(node.selectorFn);
221
- const isDeciderNode = isScopeBasedDecider;
222
- const hasChildren = Boolean((_c = node.children) === null || _c === void 0 ? void 0 : _c.length);
223
- const hasNext = Boolean(node.next);
224
- const originalNext = node.next;
225
- // ─── Phase 1: VALIDATE — node invariants ───
226
- if (!hasStageFunction && !isDeciderNode && !isScopeBasedSelector && !hasChildren) {
227
- const errorMessage = `Node '${node.name}' must define: embedded fn OR a stageMap entry OR have children/decider`;
228
- this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });
229
- throw new Error(errorMessage);
230
- }
231
- if (isDeciderNode && !hasChildren) {
232
- const errorMessage = 'Decider node needs to have children to execute';
233
- this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });
234
- throw new Error(errorMessage);
235
- }
236
- if (isScopeBasedSelector && !hasChildren) {
237
- const errorMessage = 'Selector node needs to have children to execute';
238
- this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });
239
- throw new Error(errorMessage);
240
- }
241
- // Role markers for debug panels
242
- if (!hasStageFunction) {
243
- if (isDeciderNode)
244
- context.setAsDecider();
245
- else if (hasChildren)
246
- context.setAsFork();
247
- }
248
- const breakFn = () => (breakFlag.shouldBreak = true);
249
- // ─── Phase 2a: SELECTOR — scope-based multi-choice ───
250
- if (isScopeBasedSelector) {
251
- const previousForkId = this.extractorRunner.currentForkId;
252
- this.extractorRunner.currentForkId = node.id;
253
- try {
254
- const selectorResult = await this.selectorHandler.handleScopeBased(node, stageFunc, context, breakFlag, branchPath, this.executeStage.bind(this), this.executeNode.bind(this), this.extractorRunner.callExtractor.bind(this.extractorRunner), this.extractorRunner.getStagePath.bind(this.extractorRunner), traversalContext);
255
- if (hasNext) {
264
+ // ─── Phase 0: CLASSIFY subflow detection ───
265
+ if (node.isSubflowRoot && node.subflowId) {
266
+ const resolvedNode = this.nodeResolver.resolveSubflowReference(node);
267
+ const previousSubflowId = this.extractorRunner.currentSubflowId;
268
+ this.extractorRunner.currentSubflowId = node.subflowId;
269
+ let subflowOutput;
270
+ try {
271
+ subflowOutput = await this.subflowExecutor.executeSubflow(resolvedNode, context, breakFlag, branchPath, this.subflowResults, traversalContext);
272
+ }
273
+ finally {
274
+ this.extractorRunner.currentSubflowId = previousSubflowId;
275
+ }
276
+ const isReferenceBasedSubflow = resolvedNode !== node;
277
+ const hasChildren = Boolean(node.children && node.children.length > 0);
278
+ const shouldExecuteContinuation = isReferenceBasedSubflow || hasChildren;
279
+ if (node.next && shouldExecuteContinuation) {
256
280
  const nextCtx = context.createNext(branchPath, node.next.name, node.next.id);
257
281
  return await this.executeNode(node.next, nextCtx, breakFlag, branchPath);
258
282
  }
259
- return selectorResult;
283
+ return subflowOutput;
260
284
  }
261
- finally {
262
- this.extractorRunner.currentForkId = previousForkId;
285
+ const stageFunc = this.getStageFn(node);
286
+ const hasStageFunction = Boolean(stageFunc);
287
+ const isScopeBasedDecider = Boolean(node.deciderFn);
288
+ const isScopeBasedSelector = Boolean(node.selectorFn);
289
+ const isDeciderNode = isScopeBasedDecider;
290
+ const hasChildren = Boolean((_c = node.children) === null || _c === void 0 ? void 0 : _c.length);
291
+ const hasNext = Boolean(node.next);
292
+ const originalNext = node.next;
293
+ // ─── Phase 1: VALIDATE — node invariants ───
294
+ if (!hasStageFunction && !isDeciderNode && !isScopeBasedSelector && !hasChildren) {
295
+ const errorMessage = `Node '${node.name}' must define: embedded fn OR a stageMap entry OR have children/decider`;
296
+ this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });
297
+ throw new Error(errorMessage);
263
298
  }
264
- }
265
- // ─── Phase 2b: DECIDER scope-based single-choice conditional branch ───
266
- if (isDeciderNode) {
267
- const deciderResult = await this.deciderHandler.handleScopeBased(node, stageFunc, context, breakFlag, branchPath, this.executeStage.bind(this), this.executeNode.bind(this), this.extractorRunner.callExtractor.bind(this.extractorRunner), this.extractorRunner.getStagePath.bind(this.extractorRunner), traversalContext);
268
- // After branch execution, follow decider's own next (e.g., loopTo target)
269
- if (hasNext && !breakFlag.shouldBreak) {
270
- const nextNode = originalNext;
271
- const isLoopRef = !this.getStageFn(nextNode) &&
272
- !((_d = nextNode.children) === null || _d === void 0 ? void 0 : _d.length) &&
273
- !nextNode.deciderFn &&
274
- !nextNode.selectorFn &&
275
- !nextNode.isSubflowRoot;
276
- if (isLoopRef) {
277
- return this.continuationResolver.resolve(nextNode, node, context, breakFlag, branchPath, this.executeNode.bind(this));
278
- }
279
- this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);
280
- const nextCtx = context.createNext(branchPath, nextNode.name, nextNode.id);
281
- return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);
299
+ if (isDeciderNode && !hasChildren) {
300
+ const errorMessage = 'Decider node needs to have children to execute';
301
+ this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });
302
+ throw new Error(errorMessage);
282
303
  }
283
- return deciderResult;
284
- }
285
- // ─── Abort check cooperative cancellation ───
286
- if ((_e = this.signal) === null || _e === void 0 ? void 0 : _e.aborted) {
287
- const reason = this.signal.reason instanceof Error ? this.signal.reason : new Error((_f = this.signal.reason) !== null && _f !== void 0 ? _f : 'Aborted');
288
- throw reason;
289
- }
290
- // ─── Phase 3: EXECUTE — run stage function ───
291
- let stageOutput;
292
- let dynamicNext;
293
- if (stageFunc) {
294
- try {
295
- stageOutput = await this.executeStage(node, stageFunc, context, breakFn);
296
- }
297
- catch (error) {
298
- context.commit();
299
- this.extractorRunner.callExtractor(node, context, this.extractorRunner.getStagePath(node, branchPath, context.stageName), undefined, { type: 'stageExecutionError', message: error.toString() });
300
- this.narrativeGenerator.onError(node.name, error.toString(), error, traversalContext);
301
- this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });
302
- context.addError('stageExecutionError', error.toString());
303
- throw error;
304
+ if (isScopeBasedSelector && !hasChildren) {
305
+ const errorMessage = 'Selector node needs to have children to execute';
306
+ this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });
307
+ throw new Error(errorMessage);
304
308
  }
305
- context.commit();
306
- this.extractorRunner.callExtractor(node, context, this.extractorRunner.getStagePath(node, branchPath, context.stageName), stageOutput);
307
- this.narrativeGenerator.onStageExecuted(node.name, node.description, traversalContext);
308
- if (breakFlag.shouldBreak) {
309
- this.narrativeGenerator.onBreak(node.name, traversalContext);
310
- this.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);
311
- return stageOutput;
309
+ // Role markers for debug panels
310
+ if (!hasStageFunction) {
311
+ if (isDeciderNode)
312
+ context.setAsDecider();
313
+ else if (hasChildren)
314
+ context.setAsFork();
312
315
  }
313
- // ─── Phase 4: DYNAMIC StageNode return detection ───
314
- if (stageOutput && typeof stageOutput === 'object' && isStageNodeReturn(stageOutput)) {
315
- const dynamicNode = stageOutput;
316
- context.addLog('isDynamic', true);
317
- context.addLog('dynamicPattern', 'StageNodeReturn');
318
- // Dynamic subflow auto-registration
319
- if (dynamicNode.isSubflowRoot && dynamicNode.subflowDef && dynamicNode.subflowId) {
320
- context.addLog('dynamicPattern', 'dynamicSubflow');
321
- context.addLog('dynamicSubflowId', dynamicNode.subflowId);
322
- // Structural-only subflow: has buildTimeStructure but no executable root.
323
- // Used for pre-executed subflows (e.g., inner flows that already ran).
324
- // Annotates the node for visualization without re-executing.
325
- if (!dynamicNode.subflowDef.root) {
326
- context.addLog('dynamicPattern', 'structuralSubflow');
327
- node.isSubflowRoot = true;
328
- node.subflowId = dynamicNode.subflowId;
329
- node.subflowName = dynamicNode.subflowName;
330
- node.description = (_g = dynamicNode.description) !== null && _g !== void 0 ? _g : node.description;
331
- this.structureManager.updateDynamicSubflow(node.id, dynamicNode.subflowId, dynamicNode.subflowName, dynamicNode.subflowDef.buildTimeStructure);
332
- // Fall through to Phase 5 (continuation) — no subflow execution needed
333
- }
334
- else {
335
- // Full dynamic subflow: register + execute
336
- this.autoRegisterSubflowDef(dynamicNode.subflowId, dynamicNode.subflowDef, node.id);
337
- node.isSubflowRoot = true;
338
- node.subflowId = dynamicNode.subflowId;
339
- node.subflowName = dynamicNode.subflowName;
340
- node.subflowMountOptions = dynamicNode.subflowMountOptions;
341
- this.structureManager.updateDynamicSubflow(node.id, dynamicNode.subflowId, dynamicNode.subflowName, (_h = dynamicNode.subflowDef) === null || _h === void 0 ? void 0 : _h.buildTimeStructure);
342
- return await this.executeNode(node, context, breakFlag, branchPath);
316
+ const breakFn = () => (breakFlag.shouldBreak = true);
317
+ // ─── Phase 2a: SELECTOR scope-based multi-choice ───
318
+ if (isScopeBasedSelector) {
319
+ const previousForkId = this.extractorRunner.currentForkId;
320
+ this.extractorRunner.currentForkId = node.id;
321
+ try {
322
+ const selectorResult = await this.selectorHandler.handleScopeBased(node, stageFunc, context, breakFlag, branchPath, this.executeStage.bind(this), this.executeNode.bind(this), this.extractorRunner.callExtractor.bind(this.extractorRunner), this.extractorRunner.getStagePath.bind(this.extractorRunner), traversalContext);
323
+ if (hasNext) {
324
+ const nextCtx = context.createNext(branchPath, node.next.name, node.next.id);
325
+ return await this.executeNode(node.next, nextCtx, breakFlag, branchPath);
343
326
  }
327
+ return selectorResult;
344
328
  }
345
- // Check children for subflowDef
346
- if (dynamicNode.children) {
347
- for (const child of dynamicNode.children) {
348
- if (child.isSubflowRoot && child.subflowDef && child.subflowId) {
349
- this.autoRegisterSubflowDef(child.subflowId, child.subflowDef, child.id);
350
- this.structureManager.updateDynamicSubflow(child.id, child.subflowId, child.subflowName, (_j = child.subflowDef) === null || _j === void 0 ? void 0 : _j.buildTimeStructure);
351
- }
352
- }
329
+ finally {
330
+ this.extractorRunner.currentForkId = previousForkId;
353
331
  }
354
- // Dynamic children (fork pattern)
355
- if (dynamicNode.children && dynamicNode.children.length > 0) {
356
- node.children = dynamicNode.children;
357
- context.addLog('dynamicChildCount', dynamicNode.children.length);
358
- context.addLog('dynamicChildIds', dynamicNode.children.map((c) => c.id));
359
- this.structureManager.updateDynamicChildren(node.id, dynamicNode.children, Boolean(dynamicNode.nextNodeSelector), Boolean(dynamicNode.deciderFn));
360
- if (typeof dynamicNode.nextNodeSelector === 'function') {
361
- node.nextNodeSelector = dynamicNode.nextNodeSelector;
362
- context.addLog('hasSelector', true);
332
+ }
333
+ // ─── Phase 2b: DECIDER scope-based single-choice conditional branch ───
334
+ if (isDeciderNode) {
335
+ const deciderResult = await this.deciderHandler.handleScopeBased(node, stageFunc, context, breakFlag, branchPath, this.executeStage.bind(this), this.executeNode.bind(this), this.extractorRunner.callExtractor.bind(this.extractorRunner), this.extractorRunner.getStagePath.bind(this.extractorRunner), traversalContext);
336
+ // After branch execution, follow decider's own next (e.g., loopTo target)
337
+ if (hasNext && !breakFlag.shouldBreak) {
338
+ const nextNode = originalNext;
339
+ // Use the isLoopRef flag set by loopTo() — do not rely on stageMap absence,
340
+ // since id-keyed stageMaps would otherwise cause loop targets to be executed directly.
341
+ const isLoopRef = nextNode.isLoopRef === true ||
342
+ (!this.getStageFn(nextNode) &&
343
+ !((_d = nextNode.children) === null || _d === void 0 ? void 0 : _d.length) &&
344
+ !nextNode.deciderFn &&
345
+ !nextNode.selectorFn &&
346
+ !nextNode.isSubflowRoot);
347
+ if (isLoopRef) {
348
+ return this.continuationResolver.resolve(nextNode, node, context, breakFlag, branchPath, this.executeNode.bind(this));
363
349
  }
350
+ this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);
351
+ const nextCtx = context.createNext(branchPath, nextNode.name, nextNode.id);
352
+ return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);
364
353
  }
365
- // Dynamic next (linear continuation)
366
- if (dynamicNode.next) {
367
- dynamicNext = dynamicNode.next;
368
- this.structureManager.updateDynamicNext(node.id, dynamicNode.next);
369
- node.next = dynamicNode.next;
370
- context.addLog('hasDynamicNext', true);
371
- }
372
- stageOutput = undefined;
354
+ return deciderResult;
373
355
  }
374
- // Restore original next to avoid stale reference on loop revisit
375
- if (dynamicNext) {
376
- node.next = originalNext;
356
+ // ─── Abort check cooperative cancellation ───
357
+ if ((_e = this.signal) === null || _e === void 0 ? void 0 : _e.aborted) {
358
+ const reason = this.signal.reason instanceof Error ? this.signal.reason : new Error((_f = this.signal.reason) !== null && _f !== void 0 ? _f : 'Aborted');
359
+ throw reason;
377
360
  }
378
- }
379
- // ─── Phase 5: CHILDREN — fork dispatch ───
380
- const hasChildrenAfterStage = Boolean((_k = node.children) === null || _k === void 0 ? void 0 : _k.length);
381
- if (hasChildrenAfterStage) {
382
- context.addLog('totalChildren', (_l = node.children) === null || _l === void 0 ? void 0 : _l.length);
383
- context.addLog('orderOfExecution', 'ChildrenAfterStage');
384
- let nodeChildrenResults;
385
- if (node.nextNodeSelector) {
386
- const previousForkId = this.extractorRunner.currentForkId;
387
- this.extractorRunner.currentForkId = node.id;
361
+ // ─── Phase 3: EXECUTE — run stage function ───
362
+ let stageOutput;
363
+ let dynamicNext;
364
+ if (stageFunc) {
388
365
  try {
389
- nodeChildrenResults = await this.childrenExecutor.executeSelectedChildren(node.nextNodeSelector, node.children, stageOutput, context, branchPath, traversalContext);
366
+ stageOutput = await this.executeStage(node, stageFunc, context, breakFn);
390
367
  }
391
- finally {
392
- this.extractorRunner.currentForkId = previousForkId;
368
+ catch (error) {
369
+ context.commit();
370
+ this.extractorRunner.callExtractor(node, context, this.extractorRunner.getStagePath(node, branchPath, context.stageName), undefined, { type: 'stageExecutionError', message: error.toString() });
371
+ this.narrativeGenerator.onError(node.name, error.toString(), error, traversalContext);
372
+ this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });
373
+ context.addError('stageExecutionError', error.toString());
374
+ throw error;
375
+ }
376
+ context.commit();
377
+ this.extractorRunner.callExtractor(node, context, this.extractorRunner.getStagePath(node, branchPath, context.stageName), stageOutput);
378
+ this.narrativeGenerator.onStageExecuted(node.name, node.description, traversalContext);
379
+ if (breakFlag.shouldBreak) {
380
+ this.narrativeGenerator.onBreak(node.name, traversalContext);
381
+ this.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);
382
+ return stageOutput;
383
+ }
384
+ // ─── Phase 4: DYNAMIC — StageNode return detection ───
385
+ if (stageOutput && typeof stageOutput === 'object' && isStageNodeReturn(stageOutput)) {
386
+ const dynamicNode = stageOutput;
387
+ context.addLog('isDynamic', true);
388
+ context.addLog('dynamicPattern', 'StageNodeReturn');
389
+ // Dynamic subflow auto-registration
390
+ if (dynamicNode.isSubflowRoot && dynamicNode.subflowDef && dynamicNode.subflowId) {
391
+ context.addLog('dynamicPattern', 'dynamicSubflow');
392
+ context.addLog('dynamicSubflowId', dynamicNode.subflowId);
393
+ // Structural-only subflow: has buildTimeStructure but no executable root.
394
+ // Used for pre-executed subflows (e.g., inner flows that already ran).
395
+ // Annotates the node for visualization without re-executing.
396
+ if (!dynamicNode.subflowDef.root) {
397
+ context.addLog('dynamicPattern', 'structuralSubflow');
398
+ node.isSubflowRoot = true;
399
+ node.subflowId = dynamicNode.subflowId;
400
+ node.subflowName = dynamicNode.subflowName;
401
+ node.description = (_g = dynamicNode.description) !== null && _g !== void 0 ? _g : node.description;
402
+ this.structureManager.updateDynamicSubflow(node.id, dynamicNode.subflowId, dynamicNode.subflowName, dynamicNode.subflowDef.buildTimeStructure);
403
+ // Fall through to Phase 5 (continuation) — no subflow execution needed
404
+ }
405
+ else {
406
+ // Full dynamic subflow: register + execute
407
+ this.autoRegisterSubflowDef(dynamicNode.subflowId, dynamicNode.subflowDef, node.id);
408
+ node.isSubflowRoot = true;
409
+ node.subflowId = dynamicNode.subflowId;
410
+ node.subflowName = dynamicNode.subflowName;
411
+ node.subflowMountOptions = dynamicNode.subflowMountOptions;
412
+ this.structureManager.updateDynamicSubflow(node.id, dynamicNode.subflowId, dynamicNode.subflowName, (_h = dynamicNode.subflowDef) === null || _h === void 0 ? void 0 : _h.buildTimeStructure);
413
+ return await this.executeNode(node, context, breakFlag, branchPath);
414
+ }
415
+ }
416
+ // Check children for subflowDef
417
+ if (dynamicNode.children) {
418
+ for (const child of dynamicNode.children) {
419
+ if (child.isSubflowRoot && child.subflowDef && child.subflowId) {
420
+ this.autoRegisterSubflowDef(child.subflowId, child.subflowDef, child.id);
421
+ this.structureManager.updateDynamicSubflow(child.id, child.subflowId, child.subflowName, (_j = child.subflowDef) === null || _j === void 0 ? void 0 : _j.buildTimeStructure);
422
+ }
423
+ }
424
+ }
425
+ // Dynamic children (fork pattern)
426
+ if (dynamicNode.children && dynamicNode.children.length > 0) {
427
+ node.children = dynamicNode.children;
428
+ context.addLog('dynamicChildCount', dynamicNode.children.length);
429
+ context.addLog('dynamicChildIds', dynamicNode.children.map((c) => c.id));
430
+ this.structureManager.updateDynamicChildren(node.id, dynamicNode.children, Boolean(dynamicNode.nextNodeSelector), Boolean(dynamicNode.deciderFn));
431
+ if (typeof dynamicNode.nextNodeSelector === 'function') {
432
+ node.nextNodeSelector = dynamicNode.nextNodeSelector;
433
+ context.addLog('hasSelector', true);
434
+ }
435
+ }
436
+ // Dynamic next (linear continuation)
437
+ if (dynamicNode.next) {
438
+ dynamicNext = dynamicNode.next;
439
+ this.structureManager.updateDynamicNext(node.id, dynamicNode.next);
440
+ node.next = dynamicNode.next;
441
+ context.addLog('hasDynamicNext', true);
442
+ }
443
+ stageOutput = undefined;
444
+ }
445
+ // Restore original next to avoid stale reference on loop revisit
446
+ if (dynamicNext) {
447
+ node.next = originalNext;
393
448
  }
394
449
  }
395
- else {
396
- const childCount = (_o = (_m = node.children) === null || _m === void 0 ? void 0 : _m.length) !== null && _o !== void 0 ? _o : 0;
397
- const childNames = (_p = node.children) === null || _p === void 0 ? void 0 : _p.map((c) => c.name).join(', ');
398
- context.addFlowDebugMessage('children', `Executing all ${childCount} children in parallel: ${childNames}`, {
399
- count: childCount,
400
- targetStage: (_q = node.children) === null || _q === void 0 ? void 0 : _q.map((c) => c.name),
401
- });
402
- const previousForkId = this.extractorRunner.currentForkId;
403
- this.extractorRunner.currentForkId = node.id;
404
- try {
405
- nodeChildrenResults = await this.childrenExecutor.executeNodeChildren(node, context, undefined, branchPath, traversalContext);
450
+ // ─── Phase 5: CHILDREN — fork dispatch ───
451
+ const hasChildrenAfterStage = Boolean((_k = node.children) === null || _k === void 0 ? void 0 : _k.length);
452
+ if (hasChildrenAfterStage) {
453
+ context.addLog('totalChildren', (_l = node.children) === null || _l === void 0 ? void 0 : _l.length);
454
+ context.addLog('orderOfExecution', 'ChildrenAfterStage');
455
+ let nodeChildrenResults;
456
+ if (node.nextNodeSelector) {
457
+ const previousForkId = this.extractorRunner.currentForkId;
458
+ this.extractorRunner.currentForkId = node.id;
459
+ try {
460
+ nodeChildrenResults = await this.childrenExecutor.executeSelectedChildren(node.nextNodeSelector, node.children, stageOutput, context, branchPath, traversalContext);
461
+ }
462
+ finally {
463
+ this.extractorRunner.currentForkId = previousForkId;
464
+ }
406
465
  }
407
- finally {
408
- this.extractorRunner.currentForkId = previousForkId;
466
+ else {
467
+ const childCount = (_o = (_m = node.children) === null || _m === void 0 ? void 0 : _m.length) !== null && _o !== void 0 ? _o : 0;
468
+ const childNames = (_p = node.children) === null || _p === void 0 ? void 0 : _p.map((c) => c.name).join(', ');
469
+ context.addFlowDebugMessage('children', `Executing all ${childCount} children in parallel: ${childNames}`, {
470
+ count: childCount,
471
+ targetStage: (_q = node.children) === null || _q === void 0 ? void 0 : _q.map((c) => c.name),
472
+ });
473
+ const previousForkId = this.extractorRunner.currentForkId;
474
+ this.extractorRunner.currentForkId = node.id;
475
+ try {
476
+ nodeChildrenResults = await this.childrenExecutor.executeNodeChildren(node, context, undefined, branchPath, traversalContext);
477
+ }
478
+ finally {
479
+ this.extractorRunner.currentForkId = previousForkId;
480
+ }
481
+ }
482
+ // Fork-only: return bundle
483
+ if (!hasNext && !dynamicNext) {
484
+ return nodeChildrenResults;
485
+ }
486
+ // Capture dynamic children as synthetic subflow result for UI
487
+ const isDynamic = (_s = (_r = context.debug) === null || _r === void 0 ? void 0 : _r.logContext) === null || _s === void 0 ? void 0 : _s.isDynamic;
488
+ if (isDynamic && node.children && node.children.length > 0) {
489
+ this.captureDynamicChildrenResult(node, context);
409
490
  }
410
491
  }
411
- // Fork-only: return bundle
412
- if (!hasNext && !dynamicNext) {
413
- return nodeChildrenResults;
492
+ // ─── Phase 6: CONTINUE — dynamic next / linear next ───
493
+ if (dynamicNext) {
494
+ return this.continuationResolver.resolve(dynamicNext, node, context, breakFlag, branchPath, this.executeNode.bind(this));
414
495
  }
415
- // Capture dynamic children as synthetic subflow result for UI
416
- const isDynamic = (_s = (_r = context.debug) === null || _r === void 0 ? void 0 : _r.logContext) === null || _s === void 0 ? void 0 : _s.isDynamic;
417
- if (isDynamic && node.children && node.children.length > 0) {
418
- this.captureDynamicChildrenResult(node, context);
496
+ if (hasNext) {
497
+ const nextNode = originalNext;
498
+ // Detect loop reference nodes created by loopTo() — marked with isLoopRef flag.
499
+ // Route through ContinuationResolver for proper ID resolution, iteration
500
+ // tracking, and narrative generation.
501
+ const isLoopReference = nextNode.isLoopRef;
502
+ if (isLoopReference) {
503
+ return this.continuationResolver.resolve(nextNode, node, context, breakFlag, branchPath, this.executeNode.bind(this), traversalContext);
504
+ }
505
+ this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);
506
+ context.addFlowDebugMessage('next', `Moving to ${nextNode.name} stage`, {
507
+ targetStage: nextNode.name,
508
+ });
509
+ const nextCtx = context.createNext(branchPath, nextNode.name, nextNode.id);
510
+ return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);
419
511
  }
512
+ // ─── Phase 7: LEAF — no continuation ───
513
+ return stageOutput;
420
514
  }
421
- // ─── Phase 6: CONTINUE — dynamic next / linear next ───
422
- if (dynamicNext) {
423
- return this.continuationResolver.resolve(dynamicNext, node, context, breakFlag, branchPath, this.executeNode.bind(this));
515
+ finally {
516
+ this._executeDepth--;
424
517
  }
425
- if (hasNext) {
426
- const nextNode = originalNext;
427
- // Detect loop reference nodes created by loopTo() — marked with isLoopRef flag.
428
- // Route through ContinuationResolver for proper ID resolution, iteration
429
- // tracking, and narrative generation.
430
- const isLoopReference = nextNode.isLoopRef;
431
- if (isLoopReference) {
432
- return this.continuationResolver.resolve(nextNode, node, context, breakFlag, branchPath, this.executeNode.bind(this), traversalContext);
433
- }
434
- this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);
435
- context.addFlowDebugMessage('next', `Moving to ${nextNode.name} stage`, {
436
- targetStage: nextNode.name,
437
- });
438
- const nextCtx = context.createNext(branchPath, nextNode.name, nextNode.id);
439
- return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);
440
- }
441
- // ─── Phase 7: LEAF — no continuation ───
442
- return stageOutput;
443
518
  }
444
519
  // ─────────────────────── Private Helpers ───────────────────────
445
520
  captureDynamicChildrenResult(node, context) {
@@ -504,11 +579,8 @@ export class FlowchartTraverser {
504
579
  }
505
580
  autoRegisterSubflowDef(subflowId, subflowDef, mountNodeId) {
506
581
  var _a, _b, _c, _d, _e;
507
- let subflowsDict = this.subflows;
508
- if (!subflowsDict) {
509
- subflowsDict = {};
510
- this.subflows = subflowsDict;
511
- }
582
+ // this.subflows is always initialized in the constructor; the null guard below is unreachable.
583
+ const subflowsDict = this.subflows;
512
584
  // First-write-wins
513
585
  const isNewRegistration = !subflowsDict[subflowId];
514
586
  if (isNewRegistration && subflowDef.root) {
@@ -543,4 +615,23 @@ export class FlowchartTraverser {
543
615
  }
544
616
  }
545
617
  }
546
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowchartTraverser.js","sourceRoot":"","sources":["../../../../../src/lib/engine/traversal/FlowchartTraverser.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowchartTraverser — Pre-order DFS traversal of StageNode graph.\n *\n * Unified traversal algorithm for all node shapes:\n *   const pre = await prep();\n *   const [x, y] = await Promise.all([fx(pre), fy(pre)]);\n *   return await next(x, y);\n *\n * For each node, executeNode follows 7 phases:\n *   0. CLASSIFY  — subflow detection, early delegation\n *   1. VALIDATE  — node invariants, role markers\n *   2. EXECUTE   — run stage fn, commit, break check\n *   3. DYNAMIC   — StageNode return detection, subflow auto-registration, structure updates\n *   4. CHILDREN  — fork/selector/decider dispatch\n *   5. CONTINUE  — dynamic next / linear next resolution\n *   6. LEAF      — no continuation, return output\n *\n * Break semantics: If a stage calls breakFn(), commit and STOP.\n * Patch model: Stage writes into local patch; commitPatch() after return or throw.\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { ScopeProtectionMode } from '../../scope/protection/types.js';\nimport { isStageNodeReturn } from '../graph/StageNode.js';\nimport { ChildrenExecutor } from '../handlers/ChildrenExecutor.js';\nimport { ContinuationResolver } from '../handlers/ContinuationResolver.js';\nimport { DeciderHandler } from '../handlers/DeciderHandler.js';\nimport { ExtractorRunner } from '../handlers/ExtractorRunner.js';\nimport { NodeResolver } from '../handlers/NodeResolver.js';\nimport { RuntimeStructureManager } from '../handlers/RuntimeStructureManager.js';\nimport { SelectorHandler } from '../handlers/SelectorHandler.js';\nimport { StageRunner } from '../handlers/StageRunner.js';\nimport { SubflowExecutor } from '../handlers/SubflowExecutor.js';\nimport { FlowRecorderDispatcher } from '../narrative/FlowRecorderDispatcher.js';\nimport { NarrativeFlowRecorder } from '../narrative/NarrativeFlowRecorder.js';\nimport { NullControlFlowNarrativeGenerator } from '../narrative/NullControlFlowNarrativeGenerator.js';\nimport type { FlowRecorder, IControlFlowNarrative, TraversalContext } from '../narrative/types.js';\nimport type {\n  ExtractorError,\n  HandlerDeps,\n  IExecutionRuntime,\n  ILogger,\n  NodeResultType,\n  ScopeFactory,\n  SerializedPipelineStructure,\n  StageFunction,\n  StageNode,\n  StreamHandlers,\n  SubflowResult,\n  TraversalExtractor,\n  TraversalResult,\n} from '../types.js';\n\nexport interface TraverserOptions<TOut = any, TScope = any> {\n  root: StageNode<TOut, TScope>;\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  scopeFactory: ScopeFactory<TScope>;\n  executionRuntime: IExecutionRuntime;\n  readOnlyContext?: unknown;\n  /** Execution environment — propagates to subflows automatically. */\n  executionEnv?: import('../../engine/types').ExecutionEnv;\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  streamHandlers?: StreamHandlers;\n  extractor?: TraversalExtractor;\n  scopeProtectionMode?: ScopeProtectionMode;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  enrichSnapshots?: boolean;\n  narrativeEnabled?: boolean;\n  buildTimeStructure?: SerializedPipelineStructure;\n  logger: ILogger;\n  signal?: AbortSignal;\n  /** Pre-configured FlowRecorders to attach when narrative is enabled. */\n  flowRecorders?: FlowRecorder[];\n}\n\nexport class FlowchartTraverser<TOut = any, TScope = any> {\n  private readonly root: StageNode<TOut, TScope>;\n  private stageMap: Map<string, StageFunction<TOut, TScope>>;\n  private readonly executionRuntime: IExecutionRuntime;\n  private subflows: Record<string, { root: StageNode<TOut, TScope> }>;\n  private readonly logger: ILogger;\n  private readonly signal?: AbortSignal;\n\n  // Handler modules\n  private readonly nodeResolver: NodeResolver<TOut, TScope>;\n  private readonly childrenExecutor: ChildrenExecutor<TOut, TScope>;\n  private readonly subflowExecutor: SubflowExecutor<TOut, TScope>;\n  private readonly stageRunner: StageRunner<TOut, TScope>;\n  private readonly continuationResolver: ContinuationResolver<TOut, TScope>;\n  private readonly deciderHandler: DeciderHandler<TOut, TScope>;\n  private readonly selectorHandler: SelectorHandler<TOut, TScope>;\n  private readonly structureManager: RuntimeStructureManager;\n  private readonly extractorRunner: ExtractorRunner<TOut, TScope>;\n  private readonly narrativeGenerator: IControlFlowNarrative;\n  private readonly flowRecorderDispatcher: FlowRecorderDispatcher | undefined;\n\n  // Execution state\n  private subflowResults: Map<string, SubflowResult> = new Map();\n\n  constructor(opts: TraverserOptions<TOut, TScope>) {\n    this.root = opts.root;\n    this.stageMap = opts.stageMap;\n    this.executionRuntime = opts.executionRuntime;\n    // Always initialize as object so lazy resolution can mutate in place\n    this.subflows = opts.subflows ?? {};\n    this.logger = opts.logger;\n    this.signal = opts.signal;\n\n    // Structure manager (deep-clones build-time structure)\n    this.structureManager = new RuntimeStructureManager();\n    this.structureManager.init(opts.buildTimeStructure);\n\n    // Extractor runner\n    this.extractorRunner = new ExtractorRunner(\n      opts.extractor,\n      opts.enrichSnapshots ?? false,\n      this.executionRuntime,\n      this.logger,\n    );\n\n    // Narrative generator\n    // When narrative is enabled with FlowRecorders, use FlowRecorderDispatcher.\n    // When narrative is enabled without FlowRecorders, use legacy ControlFlowNarrativeGenerator.\n    // When disabled, use NullControlFlowNarrativeGenerator (zero-cost).\n    if (opts.narrativeEnabled) {\n      const dispatcher = new FlowRecorderDispatcher();\n      this.flowRecorderDispatcher = dispatcher;\n\n      // If custom FlowRecorders are provided, use them; otherwise attach default NarrativeFlowRecorder\n      if (opts.flowRecorders && opts.flowRecorders.length > 0) {\n        for (const recorder of opts.flowRecorders) {\n          dispatcher.attach(recorder);\n        }\n      } else {\n        dispatcher.attach(new NarrativeFlowRecorder());\n      }\n\n      this.narrativeGenerator = dispatcher;\n    } else {\n      this.narrativeGenerator = new NullControlFlowNarrativeGenerator();\n    }\n\n    // Build shared deps bag\n    const deps = this.createDeps(opts);\n\n    // Initialize handler modules\n    this.nodeResolver = new NodeResolver(deps);\n    this.childrenExecutor = new ChildrenExecutor(deps, this.executeNode.bind(this));\n    this.stageRunner = new StageRunner(deps);\n    this.continuationResolver = new ContinuationResolver(deps, this.nodeResolver, (nodeId, count) =>\n      this.structureManager.updateIterationCount(nodeId, count),\n    );\n    this.deciderHandler = new DeciderHandler(deps);\n    this.selectorHandler = new SelectorHandler(deps, this.childrenExecutor);\n    this.subflowExecutor = new SubflowExecutor(\n      deps,\n      this.nodeResolver,\n      this.executeStage.bind(this),\n      this.extractorRunner.callExtractor.bind(this.extractorRunner),\n      this.getStageFn.bind(this),\n    );\n  }\n\n  private createDeps(opts: TraverserOptions<TOut, TScope>): HandlerDeps<TOut, TScope> {\n    return {\n      stageMap: this.stageMap,\n      root: this.root,\n      executionRuntime: this.executionRuntime,\n      ScopeFactory: opts.scopeFactory,\n      subflows: this.subflows,\n      throttlingErrorChecker: opts.throttlingErrorChecker,\n      streamHandlers: opts.streamHandlers,\n      scopeProtectionMode: opts.scopeProtectionMode ?? 'error',\n      readOnlyContext: opts.readOnlyContext,\n      executionEnv: opts.executionEnv,\n      narrativeGenerator: this.narrativeGenerator,\n      logger: this.logger,\n      signal: opts.signal,\n    };\n  }\n\n  // ─────────────────────── Public API ───────────────────────\n\n  async execute(): Promise<TraversalResult> {\n    const context = this.executionRuntime.rootStageContext;\n    return await this.executeNode(this.root, context, { shouldBreak: false }, '');\n  }\n\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.structureManager.getStructure();\n  }\n\n  getSnapshot() {\n    return this.executionRuntime.getSnapshot();\n  }\n\n  getRuntime() {\n    return this.executionRuntime;\n  }\n\n  setRootObject(path: string[], key: string, value: unknown) {\n    this.executionRuntime.setRootObject(path, key, value);\n  }\n\n  getBranchIds() {\n    return this.executionRuntime.getPipelines();\n  }\n\n  getRuntimeRoot(): StageNode {\n    return this.root;\n  }\n\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.subflowResults;\n  }\n\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.extractorRunner.getExtractedResults() as Map<string, TResult>;\n  }\n\n  getExtractorErrors(): ExtractorError[] {\n    return this.extractorRunner.getExtractorErrors();\n  }\n\n  getNarrative(): string[] {\n    return this.narrativeGenerator.getSentences();\n  }\n\n  /** Returns the FlowRecorderDispatcher, or undefined if narrative is disabled. */\n  getFlowRecorderDispatcher(): FlowRecorderDispatcher | undefined {\n    return this.flowRecorderDispatcher;\n  }\n\n  // ─────────────────────── Core Traversal ───────────────────────\n\n  private getStageFn(node: StageNode<TOut, TScope>): StageFunction<TOut, TScope> | undefined {\n    if (typeof node.fn === 'function') return node.fn as StageFunction<TOut, TScope>;\n    return this.stageMap.get(node.name);\n  }\n\n  private async executeStage(\n    node: StageNode<TOut, TScope>,\n    stageFunc: StageFunction<TOut, TScope>,\n    context: StageContext,\n    breakFn: () => void,\n  ) {\n    return this.stageRunner.run(node, stageFunc, context, breakFn);\n  }\n\n  /**\n   * Pre-order DFS traversal — the core algorithm.\n   * Each call processes one node through all 7 phases.\n   */\n  private async executeNode(\n    node: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath?: string,\n  ): Promise<any> {\n    // Attach builder metadata to context for snapshot enrichment\n    if (node.description) context.description = node.description;\n    if (node.isSubflowRoot && node.subflowId) context.subflowId = node.subflowId;\n\n    // Build traversal context for recorder events — created once per stage, shared by all events\n    const traversalContext: TraversalContext = {\n      stageId: node.id ?? context.stageId,\n      stageName: node.name,\n      parentStageId: context.parent?.stageId,\n      subflowId: context.subflowId,\n      subflowPath: branchPath || undefined,\n      depth: this.computeContextDepth(context),\n    };\n\n    // ─── Phase 0a: LAZY RESOLVE — deferred subflow resolution ───\n    if (node.isSubflowRoot && node.subflowResolver && !this.subflows[node.subflowId!]) {\n      const resolved = node.subflowResolver();\n      const prefixedRoot = this.prefixNodeTree(resolved.root as StageNode<TOut, TScope>, node.subflowId!);\n\n      // Register the resolved subflow (same path as eager registration)\n      this.subflows[node.subflowId!] = { root: prefixedRoot };\n\n      // Merge stageMap entries\n      for (const [key, fn] of resolved.stageMap) {\n        const prefixedKey = `${node.subflowId}/${key}`;\n        if (!this.stageMap.has(prefixedKey)) {\n          this.stageMap.set(prefixedKey, fn as StageFunction<TOut, TScope>);\n        }\n      }\n\n      // Merge nested subflows\n      if (resolved.subflows) {\n        for (const [key, def] of Object.entries(resolved.subflows)) {\n          const prefixedKey = `${node.subflowId}/${key}`;\n          if (!this.subflows[prefixedKey]) {\n            this.subflows[prefixedKey] = def as { root: StageNode<TOut, TScope> };\n          }\n        }\n      }\n\n      // Update runtime structure with the now-resolved spec\n      this.structureManager.updateDynamicSubflow(\n        node.id,\n        node.subflowId!,\n        node.subflowName,\n        resolved.buildTimeStructure,\n      );\n\n      // Clear resolver — resolved once\n      node.subflowResolver = undefined;\n    }\n\n    // ─── Phase 0: CLASSIFY — subflow detection ───\n    if (node.isSubflowRoot && node.subflowId) {\n      const resolvedNode = this.nodeResolver.resolveSubflowReference(node);\n      const previousSubflowId = this.extractorRunner.currentSubflowId;\n      this.extractorRunner.currentSubflowId = node.subflowId;\n\n      let subflowOutput: any;\n      try {\n        subflowOutput = await this.subflowExecutor.executeSubflow(\n          resolvedNode,\n          context,\n          breakFlag,\n          branchPath,\n          this.subflowResults,\n          traversalContext,\n        );\n      } finally {\n        this.extractorRunner.currentSubflowId = previousSubflowId;\n      }\n\n      const isReferenceBasedSubflow = resolvedNode !== node;\n      const hasChildren = Boolean(node.children && node.children.length > 0);\n      const shouldExecuteContinuation = isReferenceBasedSubflow || hasChildren;\n\n      if (node.next && shouldExecuteContinuation) {\n        const nextCtx = context.createNext(branchPath as string, node.next.name, node.next.id);\n        return await this.executeNode(node.next, nextCtx, breakFlag, branchPath);\n      }\n\n      return subflowOutput;\n    }\n\n    const stageFunc = this.getStageFn(node);\n    const hasStageFunction = Boolean(stageFunc);\n    const isScopeBasedDecider = Boolean(node.deciderFn);\n    const isScopeBasedSelector = Boolean(node.selectorFn);\n    const isDeciderNode = isScopeBasedDecider;\n    const hasChildren = Boolean(node.children?.length);\n    const hasNext = Boolean(node.next);\n    const originalNext = node.next;\n\n    // ─── Phase 1: VALIDATE — node invariants ───\n    if (!hasStageFunction && !isDeciderNode && !isScopeBasedSelector && !hasChildren) {\n      const errorMessage = `Node '${node.name}' must define: embedded fn OR a stageMap entry OR have children/decider`;\n      this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n    if (isDeciderNode && !hasChildren) {\n      const errorMessage = 'Decider node needs to have children to execute';\n      this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n    if (isScopeBasedSelector && !hasChildren) {\n      const errorMessage = 'Selector node needs to have children to execute';\n      this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n\n    // Role markers for debug panels\n    if (!hasStageFunction) {\n      if (isDeciderNode) context.setAsDecider();\n      else if (hasChildren) context.setAsFork();\n    }\n\n    const breakFn = () => (breakFlag.shouldBreak = true);\n\n    // ─── Phase 2a: SELECTOR — scope-based multi-choice ───\n    if (isScopeBasedSelector) {\n      const previousForkId = this.extractorRunner.currentForkId;\n      this.extractorRunner.currentForkId = node.id;\n\n      try {\n        const selectorResult = await this.selectorHandler.handleScopeBased(\n          node,\n          stageFunc!,\n          context,\n          breakFlag,\n          branchPath,\n          this.executeStage.bind(this),\n          this.executeNode.bind(this),\n          this.extractorRunner.callExtractor.bind(this.extractorRunner),\n          this.extractorRunner.getStagePath.bind(this.extractorRunner),\n          traversalContext,\n        );\n\n        if (hasNext) {\n          const nextCtx = context.createNext(branchPath as string, node.next!.name, node.next!.id);\n          return await this.executeNode(node.next!, nextCtx, breakFlag, branchPath);\n        }\n        return selectorResult;\n      } finally {\n        this.extractorRunner.currentForkId = previousForkId;\n      }\n    }\n\n    // ─── Phase 2b: DECIDER — scope-based single-choice conditional branch ───\n    if (isDeciderNode) {\n      const deciderResult = await this.deciderHandler.handleScopeBased(\n        node,\n        stageFunc!,\n        context,\n        breakFlag,\n        branchPath,\n        this.executeStage.bind(this),\n        this.executeNode.bind(this),\n        this.extractorRunner.callExtractor.bind(this.extractorRunner),\n        this.extractorRunner.getStagePath.bind(this.extractorRunner),\n        traversalContext,\n      );\n\n      // After branch execution, follow decider's own next (e.g., loopTo target)\n      if (hasNext && !breakFlag.shouldBreak) {\n        const nextNode = originalNext!;\n        const isLoopRef =\n          !this.getStageFn(nextNode) &&\n          !nextNode.children?.length &&\n          !nextNode.deciderFn &&\n          !nextNode.selectorFn &&\n          !nextNode.isSubflowRoot;\n\n        if (isLoopRef) {\n          return this.continuationResolver.resolve(\n            nextNode,\n            node,\n            context,\n            breakFlag,\n            branchPath,\n            this.executeNode.bind(this),\n          );\n        }\n\n        this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);\n        const nextCtx = context.createNext(branchPath as string, nextNode.name, nextNode.id);\n        return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);\n      }\n\n      return deciderResult;\n    }\n\n    // ─── Abort check — cooperative cancellation ───\n    if (this.signal?.aborted) {\n      const reason =\n        this.signal.reason instanceof Error ? this.signal.reason : new Error(this.signal.reason ?? 'Aborted');\n      throw reason;\n    }\n\n    // ─── Phase 3: EXECUTE — run stage function ───\n    let stageOutput: TOut | undefined;\n    let dynamicNext: StageNode<TOut, TScope> | undefined;\n\n    if (stageFunc) {\n      try {\n        stageOutput = await this.executeStage(node, stageFunc, context, breakFn);\n      } catch (error: any) {\n        context.commit();\n        this.extractorRunner.callExtractor(\n          node,\n          context,\n          this.extractorRunner.getStagePath(node, branchPath, context.stageName),\n          undefined,\n          { type: 'stageExecutionError', message: error.toString() },\n        );\n        this.narrativeGenerator.onError(node.name, error.toString(), error, traversalContext);\n        this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });\n        context.addError('stageExecutionError', error.toString());\n        throw error;\n      }\n      context.commit();\n      this.extractorRunner.callExtractor(\n        node,\n        context,\n        this.extractorRunner.getStagePath(node, branchPath, context.stageName),\n        stageOutput,\n      );\n      this.narrativeGenerator.onStageExecuted(node.name, node.description, traversalContext);\n\n      if (breakFlag.shouldBreak) {\n        this.narrativeGenerator.onBreak(node.name, traversalContext);\n        this.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);\n        return stageOutput;\n      }\n\n      // ─── Phase 4: DYNAMIC — StageNode return detection ───\n      if (stageOutput && typeof stageOutput === 'object' && isStageNodeReturn(stageOutput)) {\n        const dynamicNode = stageOutput as StageNode<TOut, TScope>;\n        context.addLog('isDynamic', true);\n        context.addLog('dynamicPattern', 'StageNodeReturn');\n\n        // Dynamic subflow auto-registration\n        if (dynamicNode.isSubflowRoot && dynamicNode.subflowDef && dynamicNode.subflowId) {\n          context.addLog('dynamicPattern', 'dynamicSubflow');\n          context.addLog('dynamicSubflowId', dynamicNode.subflowId);\n\n          // Structural-only subflow: has buildTimeStructure but no executable root.\n          // Used for pre-executed subflows (e.g., inner flows that already ran).\n          // Annotates the node for visualization without re-executing.\n          if (!dynamicNode.subflowDef.root) {\n            context.addLog('dynamicPattern', 'structuralSubflow');\n            node.isSubflowRoot = true;\n            node.subflowId = dynamicNode.subflowId;\n            node.subflowName = dynamicNode.subflowName;\n            node.description = dynamicNode.description ?? node.description;\n\n            this.structureManager.updateDynamicSubflow(\n              node.id,\n              dynamicNode.subflowId!,\n              dynamicNode.subflowName,\n              dynamicNode.subflowDef.buildTimeStructure,\n            );\n\n            // Fall through to Phase 5 (continuation) — no subflow execution needed\n          } else {\n            // Full dynamic subflow: register + execute\n            this.autoRegisterSubflowDef(dynamicNode.subflowId, dynamicNode.subflowDef, node.id);\n\n            node.isSubflowRoot = true;\n            node.subflowId = dynamicNode.subflowId;\n            node.subflowName = dynamicNode.subflowName;\n            node.subflowMountOptions = dynamicNode.subflowMountOptions;\n\n            this.structureManager.updateDynamicSubflow(\n              node.id,\n              dynamicNode.subflowId!,\n              dynamicNode.subflowName,\n              dynamicNode.subflowDef?.buildTimeStructure,\n            );\n\n            return await this.executeNode(node, context, breakFlag, branchPath);\n          }\n        }\n\n        // Check children for subflowDef\n        if (dynamicNode.children) {\n          for (const child of dynamicNode.children) {\n            if (child.isSubflowRoot && child.subflowDef && child.subflowId) {\n              this.autoRegisterSubflowDef(child.subflowId, child.subflowDef, child.id);\n              this.structureManager.updateDynamicSubflow(\n                child.id,\n                child.subflowId!,\n                child.subflowName,\n                child.subflowDef?.buildTimeStructure,\n              );\n            }\n          }\n        }\n\n        // Dynamic children (fork pattern)\n        if (dynamicNode.children && dynamicNode.children.length > 0) {\n          node.children = dynamicNode.children;\n          context.addLog('dynamicChildCount', dynamicNode.children.length);\n          context.addLog(\n            'dynamicChildIds',\n            dynamicNode.children.map((c) => c.id),\n          );\n\n          this.structureManager.updateDynamicChildren(\n            node.id,\n            dynamicNode.children,\n            Boolean(dynamicNode.nextNodeSelector),\n            Boolean(dynamicNode.deciderFn),\n          );\n\n          if (typeof dynamicNode.nextNodeSelector === 'function') {\n            node.nextNodeSelector = dynamicNode.nextNodeSelector;\n            context.addLog('hasSelector', true);\n          }\n        }\n\n        // Dynamic next (linear continuation)\n        if (dynamicNode.next) {\n          dynamicNext = dynamicNode.next;\n          this.structureManager.updateDynamicNext(node.id, dynamicNode.next);\n          node.next = dynamicNode.next;\n          context.addLog('hasDynamicNext', true);\n        }\n\n        stageOutput = undefined;\n      }\n\n      // Restore original next to avoid stale reference on loop revisit\n      if (dynamicNext) {\n        node.next = originalNext;\n      }\n    }\n\n    // ─── Phase 5: CHILDREN — fork dispatch ───\n    const hasChildrenAfterStage = Boolean(node.children?.length);\n\n    if (hasChildrenAfterStage) {\n      context.addLog('totalChildren', node.children?.length);\n      context.addLog('orderOfExecution', 'ChildrenAfterStage');\n\n      let nodeChildrenResults: Record<string, NodeResultType>;\n\n      if (node.nextNodeSelector) {\n        const previousForkId = this.extractorRunner.currentForkId;\n        this.extractorRunner.currentForkId = node.id;\n        try {\n          nodeChildrenResults = await this.childrenExecutor.executeSelectedChildren(\n            node.nextNodeSelector,\n            node.children!,\n            stageOutput,\n            context,\n            branchPath as string,\n            traversalContext,\n          );\n        } finally {\n          this.extractorRunner.currentForkId = previousForkId;\n        }\n      } else {\n        const childCount = node.children?.length ?? 0;\n        const childNames = node.children?.map((c) => c.name).join(', ');\n        context.addFlowDebugMessage('children', `Executing all ${childCount} children in parallel: ${childNames}`, {\n          count: childCount,\n          targetStage: node.children?.map((c) => c.name),\n        });\n\n        const previousForkId = this.extractorRunner.currentForkId;\n        this.extractorRunner.currentForkId = node.id;\n        try {\n          nodeChildrenResults = await this.childrenExecutor.executeNodeChildren(\n            node,\n            context,\n            undefined,\n            branchPath,\n            traversalContext,\n          );\n        } finally {\n          this.extractorRunner.currentForkId = previousForkId;\n        }\n      }\n\n      // Fork-only: return bundle\n      if (!hasNext && !dynamicNext) {\n        return nodeChildrenResults!;\n      }\n\n      // Capture dynamic children as synthetic subflow result for UI\n      const isDynamic = context.debug?.logContext?.isDynamic;\n      if (isDynamic && node.children && node.children.length > 0) {\n        this.captureDynamicChildrenResult(node, context);\n      }\n    }\n\n    // ─── Phase 6: CONTINUE — dynamic next / linear next ───\n    if (dynamicNext) {\n      return this.continuationResolver.resolve(\n        dynamicNext,\n        node,\n        context,\n        breakFlag,\n        branchPath,\n        this.executeNode.bind(this),\n      );\n    }\n\n    if (hasNext) {\n      const nextNode = originalNext!;\n\n      // Detect loop reference nodes created by loopTo() — marked with isLoopRef flag.\n      // Route through ContinuationResolver for proper ID resolution, iteration\n      // tracking, and narrative generation.\n      const isLoopReference = nextNode.isLoopRef;\n\n      if (isLoopReference) {\n        return this.continuationResolver.resolve(\n          nextNode,\n          node,\n          context,\n          breakFlag,\n          branchPath,\n          this.executeNode.bind(this),\n          traversalContext,\n        );\n      }\n\n      this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);\n      context.addFlowDebugMessage('next', `Moving to ${nextNode.name} stage`, {\n        targetStage: nextNode.name,\n      });\n      const nextCtx = context.createNext(branchPath as string, nextNode.name, nextNode.id);\n      return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);\n    }\n\n    // ─── Phase 7: LEAF — no continuation ───\n    return stageOutput;\n  }\n\n  // ─────────────────────── Private Helpers ───────────────────────\n\n  private captureDynamicChildrenResult(node: StageNode<TOut, TScope>, context: StageContext): void {\n    const parentStageId = context.getStageId();\n\n    const childStructure: any = {\n      id: `${node.id}-children`,\n      name: 'Dynamic Children',\n      type: 'fork',\n      children: node.children!.map((c) => ({\n        id: c.id,\n        name: c.name,\n        type: 'stage',\n      })),\n    };\n\n    const childStages: Record<string, unknown> = {};\n    if (context.children) {\n      for (const childCtx of context.children) {\n        const snapshot = childCtx.getSnapshot();\n        childStages[snapshot.name || snapshot.id] = {\n          name: snapshot.name,\n          output: snapshot.logs,\n          errors: snapshot.errors,\n          metrics: snapshot.metrics,\n          status: snapshot.errors && Object.keys(snapshot.errors).length > 0 ? 'error' : 'success',\n        };\n      }\n    }\n\n    this.subflowResults.set(node.id, {\n      subflowId: node.id,\n      subflowName: node.name,\n      treeContext: {\n        globalContext: {},\n        stageContexts: childStages as unknown as Record<string, unknown>,\n        history: [],\n      },\n      parentStageId,\n      pipelineStructure: childStructure,\n    });\n  }\n\n  private computeContextDepth(context: StageContext): number {\n    let depth = 0;\n    let current = context.parent;\n    while (current) {\n      depth++;\n      current = current.parent;\n    }\n    return depth;\n  }\n\n  private prefixNodeTree(node: StageNode<TOut, TScope>, prefix: string): StageNode<TOut, TScope> {\n    if (!node) return node;\n    const clone: StageNode<TOut, TScope> = { ...node };\n    clone.name = `${prefix}/${node.name}`;\n    if (clone.subflowId) clone.subflowId = `${prefix}/${clone.subflowId}`;\n    if (clone.next) clone.next = this.prefixNodeTree(clone.next, prefix);\n    if (clone.children) {\n      clone.children = clone.children.map((c) => this.prefixNodeTree(c, prefix));\n    }\n    return clone;\n  }\n\n  private autoRegisterSubflowDef(\n    subflowId: string,\n    subflowDef: NonNullable<StageNode['subflowDef']>,\n    mountNodeId?: string,\n  ): void {\n    let subflowsDict = this.subflows;\n    if (!subflowsDict) {\n      subflowsDict = {};\n      this.subflows = subflowsDict;\n    }\n\n    // First-write-wins\n    const isNewRegistration = !subflowsDict[subflowId];\n    if (isNewRegistration && subflowDef.root) {\n      subflowsDict[subflowId] = {\n        root: subflowDef.root as StageNode<TOut, TScope>,\n        ...(subflowDef.buildTimeStructure ? { buildTimeStructure: subflowDef.buildTimeStructure } : {}),\n      } as any;\n    }\n\n    // Merge stageMap entries (parent entries preserved)\n    if (subflowDef.stageMap) {\n      for (const [key, fn] of Array.from(subflowDef.stageMap.entries())) {\n        if (!this.stageMap.has(key)) {\n          this.stageMap.set(key, fn as StageFunction<TOut, TScope>);\n        }\n      }\n    }\n\n    // Merge nested subflows\n    if (subflowDef.subflows) {\n      for (const [key, def] of Object.entries(subflowDef.subflows)) {\n        if (!subflowsDict[key]) {\n          subflowsDict[key] = def as { root: StageNode<TOut, TScope> };\n        }\n      }\n    }\n\n    if (mountNodeId) {\n      this.structureManager.updateDynamicSubflow(\n        mountNodeId,\n        subflowId,\n        subflowDef.root?.subflowName || subflowDef.root?.name,\n        subflowDef.buildTimeStructure,\n      );\n    }\n\n    // Notify FlowRecorders only on first registration (matches first-write-wins)\n    if (isNewRegistration) {\n      const subflowName = subflowDef.root?.subflowName || subflowDef.root?.name || subflowId;\n      this.narrativeGenerator.onSubflowRegistered(\n        subflowId,\n        subflowName,\n        subflowDef.root?.description,\n        subflowDef.buildTimeStructure,\n      );\n    }\n  }\n}\n"]}
618
+ /**
619
+ * Default maximum recursive executeNode depth before an error is thrown.
620
+ * 500 comfortably covers any realistic pipeline depth (including deeply nested
621
+ * subflows) while preventing call-stack overflow (~10 000 frames in V8).
622
+ *
623
+ * **Note on counting:** the counter increments once per `executeNode` call, not once per
624
+ * logical user stage. Subflow root entry and subflow continuation after return each cost
625
+ * one tick. For pipelines with many nested subflows, budget roughly 2 × (avg stages per
626
+ * subflow) of headroom when computing a custom `maxDepth` via `RunOptions.maxDepth`.
627
+ *
628
+ * **Note on loops:** for `loopTo()` pipelines, this depth guard and `ContinuationResolver`'s
629
+ * iteration limit are independent — the lower one fires first. The default depth guard (500)
630
+ * fires before the default iteration limit (1000) for loop-heavy pipelines.
631
+ *
632
+ * @remarks Not safe for concurrent `.execute()` calls on the same instance — concurrent
633
+ * executions race on `_executeDepth`. Use a separate `FlowchartTraverser` per concurrent
634
+ * execution. `FlowChartExecutor.run()` always creates a fresh traverser per call.
635
+ */
636
+ FlowchartTraverser.MAX_EXECUTE_DEPTH = 500;
637
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowchartTraverser.js","sourceRoot":"","sources":["../../../../../src/lib/engine/traversal/FlowchartTraverser.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowchartTraverser — Pre-order DFS traversal of StageNode graph.\n *\n * Unified traversal algorithm for all node shapes:\n *   const pre = await prep();\n *   const [x, y] = await Promise.all([fx(pre), fy(pre)]);\n *   return await next(x, y);\n *\n * For each node, executeNode follows 7 phases:\n *   0. CLASSIFY  — subflow detection, early delegation\n *   1. VALIDATE  — node invariants, role markers\n *   2. EXECUTE   — run stage fn, commit, break check\n *   3. DYNAMIC   — StageNode return detection, subflow auto-registration, structure updates\n *   4. CHILDREN  — fork/selector/decider dispatch\n *   5. CONTINUE  — dynamic next / linear next resolution\n *   6. LEAF      — no continuation, return output\n *\n * Break semantics: If a stage calls breakFn(), commit and STOP.\n * Patch model: Stage writes into local patch; commitPatch() after return or throw.\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { ScopeProtectionMode } from '../../scope/protection/types.js';\nimport { isStageNodeReturn } from '../graph/StageNode.js';\nimport { ChildrenExecutor } from '../handlers/ChildrenExecutor.js';\nimport { ContinuationResolver } from '../handlers/ContinuationResolver.js';\nimport { DeciderHandler } from '../handlers/DeciderHandler.js';\nimport { ExtractorRunner } from '../handlers/ExtractorRunner.js';\nimport { NodeResolver } from '../handlers/NodeResolver.js';\nimport { RuntimeStructureManager } from '../handlers/RuntimeStructureManager.js';\nimport { SelectorHandler } from '../handlers/SelectorHandler.js';\nimport { StageRunner } from '../handlers/StageRunner.js';\nimport { SubflowExecutor } from '../handlers/SubflowExecutor.js';\nimport { FlowRecorderDispatcher } from '../narrative/FlowRecorderDispatcher.js';\nimport { NarrativeFlowRecorder } from '../narrative/NarrativeFlowRecorder.js';\nimport { NullControlFlowNarrativeGenerator } from '../narrative/NullControlFlowNarrativeGenerator.js';\nimport type { FlowRecorder, IControlFlowNarrative, TraversalContext } from '../narrative/types.js';\nimport type {\n  ExtractorError,\n  HandlerDeps,\n  IExecutionRuntime,\n  ILogger,\n  NodeResultType,\n  ScopeFactory,\n  SerializedPipelineStructure,\n  StageFunction,\n  StageNode,\n  StreamHandlers,\n  SubflowResult,\n  TraversalExtractor,\n  TraversalResult,\n} from '../types.js';\n\nexport interface TraverserOptions<TOut = any, TScope = any> {\n  root: StageNode<TOut, TScope>;\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  scopeFactory: ScopeFactory<TScope>;\n  executionRuntime: IExecutionRuntime;\n  readOnlyContext?: unknown;\n  /** Execution environment — propagates to subflows automatically. */\n  executionEnv?: import('../../engine/types').ExecutionEnv;\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  streamHandlers?: StreamHandlers;\n  extractor?: TraversalExtractor;\n  scopeProtectionMode?: ScopeProtectionMode;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  enrichSnapshots?: boolean;\n  narrativeEnabled?: boolean;\n  buildTimeStructure?: SerializedPipelineStructure;\n  logger: ILogger;\n  signal?: AbortSignal;\n  /** Pre-configured FlowRecorders to attach when narrative is enabled. */\n  flowRecorders?: FlowRecorder[];\n  /**\n   * Maximum recursive executeNode depth. Defaults to FlowchartTraverser.MAX_EXECUTE_DEPTH (500).\n   * Override in tests or unusually deep pipelines.\n   */\n  maxDepth?: number;\n}\n\nexport class FlowchartTraverser<TOut = any, TScope = any> {\n  private readonly root: StageNode<TOut, TScope>;\n  private stageMap: Map<string, StageFunction<TOut, TScope>>;\n  private readonly executionRuntime: IExecutionRuntime;\n  private subflows: Record<string, { root: StageNode<TOut, TScope> }>;\n  private readonly logger: ILogger;\n  private readonly signal?: AbortSignal;\n\n  // Handler modules\n  private readonly nodeResolver: NodeResolver<TOut, TScope>;\n  private readonly childrenExecutor: ChildrenExecutor<TOut, TScope>;\n  private readonly subflowExecutor: SubflowExecutor<TOut, TScope>;\n  private readonly stageRunner: StageRunner<TOut, TScope>;\n  private readonly continuationResolver: ContinuationResolver<TOut, TScope>;\n  private readonly deciderHandler: DeciderHandler<TOut, TScope>;\n  private readonly selectorHandler: SelectorHandler<TOut, TScope>;\n  private readonly structureManager: RuntimeStructureManager;\n  private readonly extractorRunner: ExtractorRunner<TOut, TScope>;\n  private readonly narrativeGenerator: IControlFlowNarrative;\n  private readonly flowRecorderDispatcher: FlowRecorderDispatcher | undefined;\n\n  // Execution state\n  private subflowResults: Map<string, SubflowResult> = new Map();\n\n  /**\n   * Per-traverser set of lazy subflow IDs that have been resolved by THIS run.\n   * Used instead of writing `node.subflowResolver = undefined` back to the shared\n   * StageNode graph — avoids a race where a concurrent traverser clears the shared\n   * resolver before another traverser has finished using it.\n   */\n  private readonly resolvedLazySubflows = new Set<string>();\n\n  /**\n   * Recursion depth counter for executeNode.\n   * Each recursive executeNode call increments this; decrements on exit (try/finally).\n   * Prevents call-stack overflow on infinite loops or excessively deep stage chains.\n   */\n  private _executeDepth = 0;\n\n  /**\n   * Per-instance maximum depth (set from TraverserOptions.maxDepth or the class default).\n   */\n  private readonly _maxDepth: number;\n\n  /**\n   * Default maximum recursive executeNode depth before an error is thrown.\n   * 500 comfortably covers any realistic pipeline depth (including deeply nested\n   * subflows) while preventing call-stack overflow (~10 000 frames in V8).\n   *\n   * **Note on counting:** the counter increments once per `executeNode` call, not once per\n   * logical user stage. Subflow root entry and subflow continuation after return each cost\n   * one tick. For pipelines with many nested subflows, budget roughly 2 × (avg stages per\n   * subflow) of headroom when computing a custom `maxDepth` via `RunOptions.maxDepth`.\n   *\n   * **Note on loops:** for `loopTo()` pipelines, this depth guard and `ContinuationResolver`'s\n   * iteration limit are independent — the lower one fires first. The default depth guard (500)\n   * fires before the default iteration limit (1000) for loop-heavy pipelines.\n   *\n   * @remarks Not safe for concurrent `.execute()` calls on the same instance — concurrent\n   * executions race on `_executeDepth`. Use a separate `FlowchartTraverser` per concurrent\n   * execution. `FlowChartExecutor.run()` always creates a fresh traverser per call.\n   */\n  static readonly MAX_EXECUTE_DEPTH = 500;\n\n  constructor(opts: TraverserOptions<TOut, TScope>) {\n    const maxDepth = opts.maxDepth ?? FlowchartTraverser.MAX_EXECUTE_DEPTH;\n    if (maxDepth < 1) throw new Error('FlowchartTraverser: maxDepth must be >= 1');\n    this._maxDepth = maxDepth;\n    this.root = opts.root;\n    // Shallow-copy stageMap and subflows so that lazy-resolution mutations\n    // (prefixed entries added during execution) stay scoped to THIS traverser\n    // and do not escape to the shared FlowChart object. Without the copy,\n    // concurrent FlowChartExecutor runs sharing the same FlowChart would race\n    // on these two mutable dictionaries.\n    this.stageMap = new Map(opts.stageMap);\n    this.executionRuntime = opts.executionRuntime;\n    this.subflows = opts.subflows ? { ...opts.subflows } : {};\n    this.logger = opts.logger;\n    this.signal = opts.signal;\n\n    // Structure manager (deep-clones build-time structure)\n    this.structureManager = new RuntimeStructureManager();\n    this.structureManager.init(opts.buildTimeStructure);\n\n    // Extractor runner\n    this.extractorRunner = new ExtractorRunner(\n      opts.extractor,\n      opts.enrichSnapshots ?? false,\n      this.executionRuntime,\n      this.logger,\n    );\n\n    // Narrative generator\n    // When narrative is enabled, use FlowRecorderDispatcher (with default NarrativeFlowRecorder or custom recorders).\n    // When disabled, use NullControlFlowNarrativeGenerator (zero-cost).\n    if (opts.narrativeEnabled) {\n      const dispatcher = new FlowRecorderDispatcher();\n      this.flowRecorderDispatcher = dispatcher;\n\n      // If custom FlowRecorders are provided, use them; otherwise attach default NarrativeFlowRecorder\n      if (opts.flowRecorders && opts.flowRecorders.length > 0) {\n        for (const recorder of opts.flowRecorders) {\n          dispatcher.attach(recorder);\n        }\n      } else {\n        dispatcher.attach(new NarrativeFlowRecorder());\n      }\n\n      this.narrativeGenerator = dispatcher;\n    } else {\n      this.narrativeGenerator = new NullControlFlowNarrativeGenerator();\n    }\n\n    // Build shared deps bag\n    const deps = this.createDeps(opts);\n\n    // Build O(1) node ID map from the root graph (avoids repeated DFS on every loopTo())\n    const nodeIdMap = this.buildNodeIdMap(opts.root);\n\n    // Initialize handler modules\n    this.nodeResolver = new NodeResolver(deps, nodeIdMap);\n    this.childrenExecutor = new ChildrenExecutor(deps, this.executeNode.bind(this));\n    this.stageRunner = new StageRunner(deps);\n    this.continuationResolver = new ContinuationResolver(deps, this.nodeResolver, (nodeId, count) =>\n      this.structureManager.updateIterationCount(nodeId, count),\n    );\n    this.deciderHandler = new DeciderHandler(deps);\n    this.selectorHandler = new SelectorHandler(deps, this.childrenExecutor);\n    this.subflowExecutor = new SubflowExecutor(\n      deps,\n      this.nodeResolver,\n      this.executeStage.bind(this),\n      this.extractorRunner.callExtractor.bind(this.extractorRunner),\n      this.getStageFn.bind(this),\n    );\n  }\n\n  private createDeps(opts: TraverserOptions<TOut, TScope>): HandlerDeps<TOut, TScope> {\n    return {\n      stageMap: this.stageMap,\n      root: this.root,\n      executionRuntime: this.executionRuntime,\n      scopeFactory: opts.scopeFactory,\n      subflows: this.subflows,\n      throttlingErrorChecker: opts.throttlingErrorChecker,\n      streamHandlers: opts.streamHandlers,\n      scopeProtectionMode: opts.scopeProtectionMode ?? 'error',\n      readOnlyContext: opts.readOnlyContext,\n      executionEnv: opts.executionEnv,\n      narrativeGenerator: this.narrativeGenerator,\n      logger: this.logger,\n      signal: opts.signal,\n    };\n  }\n\n  // ─────────────────────── Public API ───────────────────────\n\n  async execute(): Promise<TraversalResult> {\n    const context = this.executionRuntime.rootStageContext;\n    return await this.executeNode(this.root, context, { shouldBreak: false }, '');\n  }\n\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.structureManager.getStructure();\n  }\n\n  getSnapshot() {\n    return this.executionRuntime.getSnapshot();\n  }\n\n  getRuntime() {\n    return this.executionRuntime;\n  }\n\n  setRootObject(path: string[], key: string, value: unknown) {\n    this.executionRuntime.setRootObject(path, key, value);\n  }\n\n  getBranchIds() {\n    return this.executionRuntime.getPipelines();\n  }\n\n  getRuntimeRoot(): StageNode {\n    return this.root;\n  }\n\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.subflowResults;\n  }\n\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.extractorRunner.getExtractedResults() as Map<string, TResult>;\n  }\n\n  getExtractorErrors(): ExtractorError[] {\n    return this.extractorRunner.getExtractorErrors();\n  }\n\n  getNarrative(): string[] {\n    return this.narrativeGenerator.getSentences();\n  }\n\n  /** Returns the FlowRecorderDispatcher, or undefined if narrative is disabled. */\n  getFlowRecorderDispatcher(): FlowRecorderDispatcher | undefined {\n    return this.flowRecorderDispatcher;\n  }\n\n  // ─────────────────────── Core Traversal ───────────────────────\n\n  /**\n   * Build an O(1) ID→node map from the root graph.\n   * Used by NodeResolver to avoid repeated DFS on every loopTo() call.\n   * Depth-guarded at MAX_EXECUTE_DEPTH to prevent infinite recursion on cyclic graphs.\n   */\n  private buildNodeIdMap(root: StageNode<TOut, TScope>): Map<string, StageNode<TOut, TScope>> {\n    const map = new Map<string, StageNode<TOut, TScope>>();\n    const visit = (node: StageNode<TOut, TScope>, depth: number): void => {\n      if (depth > FlowchartTraverser.MAX_EXECUTE_DEPTH) return;\n      if (map.has(node.id)) return; // already visited (avoids infinite loops on cyclic refs)\n      map.set(node.id, node);\n      if (node.children) {\n        for (const child of node.children) visit(child, depth + 1);\n      }\n      if (node.next) visit(node.next, depth + 1);\n    };\n    visit(root, 0);\n    return map;\n  }\n\n  private getStageFn(node: StageNode<TOut, TScope>): StageFunction<TOut, TScope> | undefined {\n    if (typeof node.fn === 'function') return node.fn as StageFunction<TOut, TScope>;\n    // Primary: look up by id (stable identifier, keyed by FlowChartBuilder)\n    const byId = this.stageMap.get(node.id);\n    if (byId !== undefined) return byId;\n    // Fallback: look up by name (supports hand-crafted stageMaps in tests and advanced use)\n    return this.stageMap.get(node.name);\n  }\n\n  private async executeStage(\n    node: StageNode<TOut, TScope>,\n    stageFunc: StageFunction<TOut, TScope>,\n    context: StageContext,\n    breakFn: () => void,\n  ) {\n    return this.stageRunner.run(node, stageFunc, context, breakFn);\n  }\n\n  /**\n   * Pre-order DFS traversal — the core algorithm.\n   * Each call processes one node through all 7 phases.\n   */\n  private async executeNode(\n    node: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath?: string,\n  ): Promise<any> {\n    // ─── Recursion depth guard ───\n    // Each `await executeNode(...)` keeps the calling frame on the stack.\n    // Without a cap, an infinite loop or an excessively deep stage chain will\n    // eventually overflow the V8 call stack (~10 000 frames) with a cryptic\n    // \"Maximum call stack size exceeded\" error.  We fail early with a clear\n    // message so users can diagnose the cause (infinite loop, missing break, etc.).\n    // The increment is inside `try` so `finally` always decrements — no fragile\n    // gap between check and try entry.\n    try {\n      if (++this._executeDepth > this._maxDepth) {\n        throw new Error(\n          `FlowchartTraverser: maximum traversal depth exceeded (${this._maxDepth}). ` +\n            'Check for infinite loops or missing break conditions in your flowchart. ' +\n            `Last stage: '${node.name}'. ` +\n            'For loopTo() pipelines, consider adding a break condition or using RunOptions.maxDepth to raise the limit.',\n        );\n      }\n\n      // Attach builder metadata to context for snapshot enrichment\n      if (node.description) context.description = node.description;\n      if (node.isSubflowRoot && node.subflowId) context.subflowId = node.subflowId;\n\n      // Build traversal context for recorder events — created once per stage, shared by all events\n      const traversalContext: TraversalContext = {\n        stageId: node.id ?? context.stageId,\n        stageName: node.name,\n        parentStageId: context.parent?.stageId,\n        subflowId: context.subflowId,\n        subflowPath: branchPath || undefined,\n        depth: this.computeContextDepth(context),\n      };\n\n      // ─── Phase 0a: LAZY RESOLVE — deferred subflow resolution ───\n      // Guard uses the per-traverser resolvedLazySubflows set (not the shared node) so\n      // concurrent traversers do not race on node.subflowResolver or clear it for each other.\n      if (node.isSubflowRoot && node.subflowResolver && !this.resolvedLazySubflows.has(node.subflowId!)) {\n        const resolved = node.subflowResolver();\n        const prefixedRoot = this.prefixNodeTree(resolved.root as StageNode<TOut, TScope>, node.subflowId!);\n\n        // Register the resolved subflow (same path as eager registration)\n        this.subflows[node.subflowId!] = { root: prefixedRoot };\n\n        // Merge stageMap entries\n        for (const [key, fn] of resolved.stageMap) {\n          const prefixedKey = `${node.subflowId}/${key}`;\n          if (!this.stageMap.has(prefixedKey)) {\n            this.stageMap.set(prefixedKey, fn as StageFunction<TOut, TScope>);\n          }\n        }\n\n        // Merge nested subflows\n        if (resolved.subflows) {\n          for (const [key, def] of Object.entries(resolved.subflows)) {\n            const prefixedKey = `${node.subflowId}/${key}`;\n            if (!this.subflows[prefixedKey]) {\n              this.subflows[prefixedKey] = def as { root: StageNode<TOut, TScope> };\n            }\n          }\n        }\n\n        // Update runtime structure with the now-resolved spec\n        this.structureManager.updateDynamicSubflow(\n          node.id,\n          node.subflowId!,\n          node.subflowName,\n          resolved.buildTimeStructure,\n        );\n\n        // Mark as resolved for THIS traverser — per-traverser set prevents re-entry\n        // without mutating the shared StageNode graph (which would race concurrent traversers).\n        this.resolvedLazySubflows.add(node.subflowId!);\n      }\n\n      // ─── Phase 0: CLASSIFY — subflow detection ───\n      if (node.isSubflowRoot && node.subflowId) {\n        const resolvedNode = this.nodeResolver.resolveSubflowReference(node);\n        const previousSubflowId = this.extractorRunner.currentSubflowId;\n        this.extractorRunner.currentSubflowId = node.subflowId;\n\n        let subflowOutput: any;\n        try {\n          subflowOutput = await this.subflowExecutor.executeSubflow(\n            resolvedNode,\n            context,\n            breakFlag,\n            branchPath,\n            this.subflowResults,\n            traversalContext,\n          );\n        } finally {\n          this.extractorRunner.currentSubflowId = previousSubflowId;\n        }\n\n        const isReferenceBasedSubflow = resolvedNode !== node;\n        const hasChildren = Boolean(node.children && node.children.length > 0);\n        const shouldExecuteContinuation = isReferenceBasedSubflow || hasChildren;\n\n        if (node.next && shouldExecuteContinuation) {\n          const nextCtx = context.createNext(branchPath as string, node.next.name, node.next.id);\n          return await this.executeNode(node.next, nextCtx, breakFlag, branchPath);\n        }\n\n        return subflowOutput;\n      }\n\n      const stageFunc = this.getStageFn(node);\n      const hasStageFunction = Boolean(stageFunc);\n      const isScopeBasedDecider = Boolean(node.deciderFn);\n      const isScopeBasedSelector = Boolean(node.selectorFn);\n      const isDeciderNode = isScopeBasedDecider;\n      const hasChildren = Boolean(node.children?.length);\n      const hasNext = Boolean(node.next);\n      const originalNext = node.next;\n\n      // ─── Phase 1: VALIDATE — node invariants ───\n      if (!hasStageFunction && !isDeciderNode && !isScopeBasedSelector && !hasChildren) {\n        const errorMessage = `Node '${node.name}' must define: embedded fn OR a stageMap entry OR have children/decider`;\n        this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });\n        throw new Error(errorMessage);\n      }\n      if (isDeciderNode && !hasChildren) {\n        const errorMessage = 'Decider node needs to have children to execute';\n        this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });\n        throw new Error(errorMessage);\n      }\n      if (isScopeBasedSelector && !hasChildren) {\n        const errorMessage = 'Selector node needs to have children to execute';\n        this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error: errorMessage });\n        throw new Error(errorMessage);\n      }\n\n      // Role markers for debug panels\n      if (!hasStageFunction) {\n        if (isDeciderNode) context.setAsDecider();\n        else if (hasChildren) context.setAsFork();\n      }\n\n      const breakFn = () => (breakFlag.shouldBreak = true);\n\n      // ─── Phase 2a: SELECTOR — scope-based multi-choice ───\n      if (isScopeBasedSelector) {\n        const previousForkId = this.extractorRunner.currentForkId;\n        this.extractorRunner.currentForkId = node.id;\n\n        try {\n          const selectorResult = await this.selectorHandler.handleScopeBased(\n            node,\n            stageFunc!,\n            context,\n            breakFlag,\n            branchPath,\n            this.executeStage.bind(this),\n            this.executeNode.bind(this),\n            this.extractorRunner.callExtractor.bind(this.extractorRunner),\n            this.extractorRunner.getStagePath.bind(this.extractorRunner),\n            traversalContext,\n          );\n\n          if (hasNext) {\n            const nextCtx = context.createNext(branchPath as string, node.next!.name, node.next!.id);\n            return await this.executeNode(node.next!, nextCtx, breakFlag, branchPath);\n          }\n          return selectorResult;\n        } finally {\n          this.extractorRunner.currentForkId = previousForkId;\n        }\n      }\n\n      // ─── Phase 2b: DECIDER — scope-based single-choice conditional branch ───\n      if (isDeciderNode) {\n        const deciderResult = await this.deciderHandler.handleScopeBased(\n          node,\n          stageFunc!,\n          context,\n          breakFlag,\n          branchPath,\n          this.executeStage.bind(this),\n          this.executeNode.bind(this),\n          this.extractorRunner.callExtractor.bind(this.extractorRunner),\n          this.extractorRunner.getStagePath.bind(this.extractorRunner),\n          traversalContext,\n        );\n\n        // After branch execution, follow decider's own next (e.g., loopTo target)\n        if (hasNext && !breakFlag.shouldBreak) {\n          const nextNode = originalNext!;\n          // Use the isLoopRef flag set by loopTo() — do not rely on stageMap absence,\n          // since id-keyed stageMaps would otherwise cause loop targets to be executed directly.\n          const isLoopRef =\n            nextNode.isLoopRef === true ||\n            (!this.getStageFn(nextNode) &&\n              !nextNode.children?.length &&\n              !nextNode.deciderFn &&\n              !nextNode.selectorFn &&\n              !nextNode.isSubflowRoot);\n\n          if (isLoopRef) {\n            return this.continuationResolver.resolve(\n              nextNode,\n              node,\n              context,\n              breakFlag,\n              branchPath,\n              this.executeNode.bind(this),\n            );\n          }\n\n          this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);\n          const nextCtx = context.createNext(branchPath as string, nextNode.name, nextNode.id);\n          return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);\n        }\n\n        return deciderResult;\n      }\n\n      // ─── Abort check — cooperative cancellation ───\n      if (this.signal?.aborted) {\n        const reason =\n          this.signal.reason instanceof Error ? this.signal.reason : new Error(this.signal.reason ?? 'Aborted');\n        throw reason;\n      }\n\n      // ─── Phase 3: EXECUTE — run stage function ───\n      let stageOutput: TOut | undefined;\n      let dynamicNext: StageNode<TOut, TScope> | undefined;\n\n      if (stageFunc) {\n        try {\n          stageOutput = await this.executeStage(node, stageFunc, context, breakFn);\n        } catch (error: any) {\n          context.commit();\n          this.extractorRunner.callExtractor(\n            node,\n            context,\n            this.extractorRunner.getStagePath(node, branchPath, context.stageName),\n            undefined,\n            { type: 'stageExecutionError', message: error.toString() },\n          );\n          this.narrativeGenerator.onError(node.name, error.toString(), error, traversalContext);\n          this.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });\n          context.addError('stageExecutionError', error.toString());\n          throw error;\n        }\n        context.commit();\n        this.extractorRunner.callExtractor(\n          node,\n          context,\n          this.extractorRunner.getStagePath(node, branchPath, context.stageName),\n          stageOutput,\n        );\n        this.narrativeGenerator.onStageExecuted(node.name, node.description, traversalContext);\n\n        if (breakFlag.shouldBreak) {\n          this.narrativeGenerator.onBreak(node.name, traversalContext);\n          this.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);\n          return stageOutput;\n        }\n\n        // ─── Phase 4: DYNAMIC — StageNode return detection ───\n        if (stageOutput && typeof stageOutput === 'object' && isStageNodeReturn(stageOutput)) {\n          const dynamicNode = stageOutput as StageNode<TOut, TScope>;\n          context.addLog('isDynamic', true);\n          context.addLog('dynamicPattern', 'StageNodeReturn');\n\n          // Dynamic subflow auto-registration\n          if (dynamicNode.isSubflowRoot && dynamicNode.subflowDef && dynamicNode.subflowId) {\n            context.addLog('dynamicPattern', 'dynamicSubflow');\n            context.addLog('dynamicSubflowId', dynamicNode.subflowId);\n\n            // Structural-only subflow: has buildTimeStructure but no executable root.\n            // Used for pre-executed subflows (e.g., inner flows that already ran).\n            // Annotates the node for visualization without re-executing.\n            if (!dynamicNode.subflowDef.root) {\n              context.addLog('dynamicPattern', 'structuralSubflow');\n              node.isSubflowRoot = true;\n              node.subflowId = dynamicNode.subflowId;\n              node.subflowName = dynamicNode.subflowName;\n              node.description = dynamicNode.description ?? node.description;\n\n              this.structureManager.updateDynamicSubflow(\n                node.id,\n                dynamicNode.subflowId!,\n                dynamicNode.subflowName,\n                dynamicNode.subflowDef.buildTimeStructure,\n              );\n\n              // Fall through to Phase 5 (continuation) — no subflow execution needed\n            } else {\n              // Full dynamic subflow: register + execute\n              this.autoRegisterSubflowDef(dynamicNode.subflowId, dynamicNode.subflowDef, node.id);\n\n              node.isSubflowRoot = true;\n              node.subflowId = dynamicNode.subflowId;\n              node.subflowName = dynamicNode.subflowName;\n              node.subflowMountOptions = dynamicNode.subflowMountOptions;\n\n              this.structureManager.updateDynamicSubflow(\n                node.id,\n                dynamicNode.subflowId!,\n                dynamicNode.subflowName,\n                dynamicNode.subflowDef?.buildTimeStructure,\n              );\n\n              return await this.executeNode(node, context, breakFlag, branchPath);\n            }\n          }\n\n          // Check children for subflowDef\n          if (dynamicNode.children) {\n            for (const child of dynamicNode.children) {\n              if (child.isSubflowRoot && child.subflowDef && child.subflowId) {\n                this.autoRegisterSubflowDef(child.subflowId, child.subflowDef, child.id);\n                this.structureManager.updateDynamicSubflow(\n                  child.id,\n                  child.subflowId!,\n                  child.subflowName,\n                  child.subflowDef?.buildTimeStructure,\n                );\n              }\n            }\n          }\n\n          // Dynamic children (fork pattern)\n          if (dynamicNode.children && dynamicNode.children.length > 0) {\n            node.children = dynamicNode.children;\n            context.addLog('dynamicChildCount', dynamicNode.children.length);\n            context.addLog(\n              'dynamicChildIds',\n              dynamicNode.children.map((c) => c.id),\n            );\n\n            this.structureManager.updateDynamicChildren(\n              node.id,\n              dynamicNode.children,\n              Boolean(dynamicNode.nextNodeSelector),\n              Boolean(dynamicNode.deciderFn),\n            );\n\n            if (typeof dynamicNode.nextNodeSelector === 'function') {\n              node.nextNodeSelector = dynamicNode.nextNodeSelector;\n              context.addLog('hasSelector', true);\n            }\n          }\n\n          // Dynamic next (linear continuation)\n          if (dynamicNode.next) {\n            dynamicNext = dynamicNode.next;\n            this.structureManager.updateDynamicNext(node.id, dynamicNode.next);\n            node.next = dynamicNode.next;\n            context.addLog('hasDynamicNext', true);\n          }\n\n          stageOutput = undefined;\n        }\n\n        // Restore original next to avoid stale reference on loop revisit\n        if (dynamicNext) {\n          node.next = originalNext;\n        }\n      }\n\n      // ─── Phase 5: CHILDREN — fork dispatch ───\n      const hasChildrenAfterStage = Boolean(node.children?.length);\n\n      if (hasChildrenAfterStage) {\n        context.addLog('totalChildren', node.children?.length);\n        context.addLog('orderOfExecution', 'ChildrenAfterStage');\n\n        let nodeChildrenResults: Record<string, NodeResultType>;\n\n        if (node.nextNodeSelector) {\n          const previousForkId = this.extractorRunner.currentForkId;\n          this.extractorRunner.currentForkId = node.id;\n          try {\n            nodeChildrenResults = await this.childrenExecutor.executeSelectedChildren(\n              node.nextNodeSelector,\n              node.children!,\n              stageOutput,\n              context,\n              branchPath as string,\n              traversalContext,\n            );\n          } finally {\n            this.extractorRunner.currentForkId = previousForkId;\n          }\n        } else {\n          const childCount = node.children?.length ?? 0;\n          const childNames = node.children?.map((c) => c.name).join(', ');\n          context.addFlowDebugMessage('children', `Executing all ${childCount} children in parallel: ${childNames}`, {\n            count: childCount,\n            targetStage: node.children?.map((c) => c.name),\n          });\n\n          const previousForkId = this.extractorRunner.currentForkId;\n          this.extractorRunner.currentForkId = node.id;\n          try {\n            nodeChildrenResults = await this.childrenExecutor.executeNodeChildren(\n              node,\n              context,\n              undefined,\n              branchPath,\n              traversalContext,\n            );\n          } finally {\n            this.extractorRunner.currentForkId = previousForkId;\n          }\n        }\n\n        // Fork-only: return bundle\n        if (!hasNext && !dynamicNext) {\n          return nodeChildrenResults!;\n        }\n\n        // Capture dynamic children as synthetic subflow result for UI\n        const isDynamic = context.debug?.logContext?.isDynamic;\n        if (isDynamic && node.children && node.children.length > 0) {\n          this.captureDynamicChildrenResult(node, context);\n        }\n      }\n\n      // ─── Phase 6: CONTINUE — dynamic next / linear next ───\n      if (dynamicNext) {\n        return this.continuationResolver.resolve(\n          dynamicNext,\n          node,\n          context,\n          breakFlag,\n          branchPath,\n          this.executeNode.bind(this),\n        );\n      }\n\n      if (hasNext) {\n        const nextNode = originalNext!;\n\n        // Detect loop reference nodes created by loopTo() — marked with isLoopRef flag.\n        // Route through ContinuationResolver for proper ID resolution, iteration\n        // tracking, and narrative generation.\n        const isLoopReference = nextNode.isLoopRef;\n\n        if (isLoopReference) {\n          return this.continuationResolver.resolve(\n            nextNode,\n            node,\n            context,\n            breakFlag,\n            branchPath,\n            this.executeNode.bind(this),\n            traversalContext,\n          );\n        }\n\n        this.narrativeGenerator.onNext(node.name, nextNode.name, nextNode.description, traversalContext);\n        context.addFlowDebugMessage('next', `Moving to ${nextNode.name} stage`, {\n          targetStage: nextNode.name,\n        });\n        const nextCtx = context.createNext(branchPath as string, nextNode.name, nextNode.id);\n        return await this.executeNode(nextNode, nextCtx, breakFlag, branchPath);\n      }\n\n      // ─── Phase 7: LEAF — no continuation ───\n      return stageOutput;\n    } finally {\n      this._executeDepth--;\n    }\n  }\n\n  // ─────────────────────── Private Helpers ───────────────────────\n\n  private captureDynamicChildrenResult(node: StageNode<TOut, TScope>, context: StageContext): void {\n    const parentStageId = context.getStageId();\n\n    const childStructure: any = {\n      id: `${node.id}-children`,\n      name: 'Dynamic Children',\n      type: 'fork',\n      children: node.children!.map((c) => ({\n        id: c.id,\n        name: c.name,\n        type: 'stage',\n      })),\n    };\n\n    const childStages: Record<string, unknown> = {};\n    if (context.children) {\n      for (const childCtx of context.children) {\n        const snapshot = childCtx.getSnapshot();\n        childStages[snapshot.name || snapshot.id] = {\n          name: snapshot.name,\n          output: snapshot.logs,\n          errors: snapshot.errors,\n          metrics: snapshot.metrics,\n          status: snapshot.errors && Object.keys(snapshot.errors).length > 0 ? 'error' : 'success',\n        };\n      }\n    }\n\n    this.subflowResults.set(node.id, {\n      subflowId: node.id,\n      subflowName: node.name,\n      treeContext: {\n        globalContext: {},\n        stageContexts: childStages as unknown as Record<string, unknown>,\n        history: [],\n      },\n      parentStageId,\n      pipelineStructure: childStructure,\n    });\n  }\n\n  private computeContextDepth(context: StageContext): number {\n    let depth = 0;\n    let current = context.parent;\n    while (current) {\n      depth++;\n      current = current.parent;\n    }\n    return depth;\n  }\n\n  private prefixNodeTree(node: StageNode<TOut, TScope>, prefix: string): StageNode<TOut, TScope> {\n    if (!node) return node;\n    const clone: StageNode<TOut, TScope> = { ...node };\n    clone.name = `${prefix}/${node.name}`;\n    if (clone.subflowId) clone.subflowId = `${prefix}/${clone.subflowId}`;\n    if (clone.next) clone.next = this.prefixNodeTree(clone.next, prefix);\n    if (clone.children) {\n      clone.children = clone.children.map((c) => this.prefixNodeTree(c, prefix));\n    }\n    return clone;\n  }\n\n  private autoRegisterSubflowDef(\n    subflowId: string,\n    subflowDef: NonNullable<StageNode['subflowDef']>,\n    mountNodeId?: string,\n  ): void {\n    // this.subflows is always initialized in the constructor; the null guard below is unreachable.\n    const subflowsDict = this.subflows;\n\n    // First-write-wins\n    const isNewRegistration = !subflowsDict[subflowId];\n    if (isNewRegistration && subflowDef.root) {\n      subflowsDict[subflowId] = {\n        root: subflowDef.root as StageNode<TOut, TScope>,\n        ...(subflowDef.buildTimeStructure ? { buildTimeStructure: subflowDef.buildTimeStructure } : {}),\n      } as any;\n    }\n\n    // Merge stageMap entries (parent entries preserved)\n    if (subflowDef.stageMap) {\n      for (const [key, fn] of Array.from(subflowDef.stageMap.entries())) {\n        if (!this.stageMap.has(key)) {\n          this.stageMap.set(key, fn as StageFunction<TOut, TScope>);\n        }\n      }\n    }\n\n    // Merge nested subflows\n    if (subflowDef.subflows) {\n      for (const [key, def] of Object.entries(subflowDef.subflows)) {\n        if (!subflowsDict[key]) {\n          subflowsDict[key] = def as { root: StageNode<TOut, TScope> };\n        }\n      }\n    }\n\n    if (mountNodeId) {\n      this.structureManager.updateDynamicSubflow(\n        mountNodeId,\n        subflowId,\n        subflowDef.root?.subflowName || subflowDef.root?.name,\n        subflowDef.buildTimeStructure,\n      );\n    }\n\n    // Notify FlowRecorders only on first registration (matches first-write-wins)\n    if (isNewRegistration) {\n      const subflowName = subflowDef.root?.subflowName || subflowDef.root?.name || subflowId;\n      this.narrativeGenerator.onSubflowRegistered(\n        subflowId,\n        subflowName,\n        subflowDef.root?.description,\n        subflowDef.buildTimeStructure,\n      );\n    }\n  }\n}\n"]}