footprintjs 0.17.0 → 0.17.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/esm/lib/engine/handlers/ChildrenExecutor.js +6 -6
- package/dist/esm/lib/engine/handlers/ContinuationResolver.js +8 -8
- package/dist/esm/lib/engine/handlers/DeciderHandler.js +4 -4
- package/dist/esm/lib/engine/handlers/SelectorHandler.js +5 -5
- package/dist/esm/lib/engine/handlers/SubflowExecutor.js +4 -4
- package/dist/esm/lib/engine/narrative/CombinedNarrativeBuilder.js +1 -1
- package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +45 -22
- package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +7 -7
- package/dist/lib/engine/handlers/ChildrenExecutor.js +6 -6
- package/dist/lib/engine/handlers/ContinuationResolver.js +8 -8
- package/dist/lib/engine/handlers/DeciderHandler.js +4 -4
- package/dist/lib/engine/handlers/SelectorHandler.js +5 -5
- package/dist/lib/engine/handlers/SubflowExecutor.js +4 -4
- package/dist/lib/engine/narrative/CombinedNarrativeBuilder.js +1 -1
- package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +45 -22
- package/dist/lib/engine/traversal/FlowchartTraverser.js +7 -7
- package/dist/types/lib/engine/handlers/ChildrenExecutor.d.ts +3 -2
- package/dist/types/lib/engine/handlers/ContinuationResolver.d.ts +2 -1
- package/dist/types/lib/engine/handlers/DeciderHandler.d.ts +2 -1
- package/dist/types/lib/engine/handlers/SelectorHandler.d.ts +2 -1
- package/dist/types/lib/engine/handlers/SubflowExecutor.d.ts +2 -1
- package/dist/types/lib/engine/narrative/CombinedNarrativeBuilder.d.ts +2 -0
- package/package.json +1 -1
|
@@ -16,14 +16,14 @@ export class ChildrenExecutor {
|
|
|
16
16
|
* Execute all children in parallel. Each child commits on settle.
|
|
17
17
|
* Uses Promise.allSettled to ensure all children complete even if some fail.
|
|
18
18
|
*/
|
|
19
|
-
async executeNodeChildren(node, context, parentBreakFlag, branchPath) {
|
|
19
|
+
async executeNodeChildren(node, context, parentBreakFlag, branchPath, traversalContext) {
|
|
20
20
|
var _a, _b, _c;
|
|
21
21
|
let breakCount = 0;
|
|
22
22
|
const totalChildren = (_b = (_a = node.children) === null || _a === void 0 ? void 0 : _a.length) !== null && _b !== void 0 ? _b : 0;
|
|
23
23
|
const allChildren = (_c = node.children) !== null && _c !== void 0 ? _c : [];
|
|
24
24
|
// Narrative: capture the fan-out
|
|
25
25
|
const childDisplayNames = allChildren.map((c) => c.name);
|
|
26
|
-
this.deps.narrativeGenerator.onFork(node.name, childDisplayNames);
|
|
26
|
+
this.deps.narrativeGenerator.onFork(node.name, childDisplayNames, traversalContext);
|
|
27
27
|
const childPromises = allChildren.map((child) => {
|
|
28
28
|
const childBranchPath = branchPath || child.id;
|
|
29
29
|
const childContext = context.createChild(childBranchPath, child.id, child.name, child.id);
|
|
@@ -81,7 +81,7 @@ export class ChildrenExecutor {
|
|
|
81
81
|
* Execute selected children based on selector result.
|
|
82
82
|
* Validates IDs, records selection info, then delegates to executeNodeChildren.
|
|
83
83
|
*/
|
|
84
|
-
async executeSelectedChildren(selector, children, input, context, branchPath) {
|
|
84
|
+
async executeSelectedChildren(selector, children, input, context, branchPath, traversalContext) {
|
|
85
85
|
const selectorResult = await selector(input);
|
|
86
86
|
const selectedIds = Array.isArray(selectorResult) ? selectorResult : [selectorResult];
|
|
87
87
|
context.addLog('selectedChildIds', selectedIds);
|
|
@@ -108,13 +108,13 @@ export class ChildrenExecutor {
|
|
|
108
108
|
context.addFlowDebugMessage('selected', `Running ${selectedNames} (${selectedChildren.length} of ${children.length} matched)`, { count: selectedChildren.length, targetStage: selectedChildren.map((c) => c.name) });
|
|
109
109
|
// Narrative: capture the selection
|
|
110
110
|
const selectedDisplayNames = selectedChildren.map((c) => c.name);
|
|
111
|
-
this.deps.narrativeGenerator.onSelected(context.stageName || 'selector', selectedDisplayNames, children.length);
|
|
111
|
+
this.deps.narrativeGenerator.onSelected(context.stageName || 'selector', selectedDisplayNames, children.length, traversalContext);
|
|
112
112
|
const tempNode = {
|
|
113
113
|
name: 'selector-temp',
|
|
114
114
|
id: 'selector-temp',
|
|
115
115
|
children: selectedChildren,
|
|
116
116
|
};
|
|
117
|
-
return await this.executeNodeChildren(tempNode, context, undefined, branchPath);
|
|
117
|
+
return await this.executeNodeChildren(tempNode, context, undefined, branchPath, traversalContext);
|
|
118
118
|
}
|
|
119
119
|
}
|
|
120
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ChildrenExecutor.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/ChildrenExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ChildrenExecutor — Parallel fan-out via Promise.allSettled.\n *\n * Responsibilities:\n * - Execute all children in parallel (fork pattern)\n * - Execute selected children based on selector output (multi-choice)\n * - Handle throttling error flagging for rate-limited operations\n * - Aggregate results into { childId: { result, isError } }\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { Selector, StageNode } from '../graph/StageNode.js';\nimport type { HandlerDeps, NodeResultType } from '../types.js';\n\n/** Callback for recursive node execution. Avoids circular dependency with traverser. */\nexport type ExecuteNodeFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  breakFlag: { shouldBreak: boolean },\n  branchPath?: string,\n) => Promise<any>;\n\nexport class ChildrenExecutor<TOut = any, TScope = any> {\n  constructor(private deps: HandlerDeps<TOut, TScope>, private executeNode: ExecuteNodeFn<TOut, TScope>) {}\n\n  /**\n   * Execute all children in parallel. Each child commits on settle.\n   * Uses Promise.allSettled to ensure all children complete even if some fail.\n   */\n  async executeNodeChildren(\n    node: StageNode<TOut, TScope>,\n    context: StageContext,\n    parentBreakFlag?: { shouldBreak: boolean },\n    branchPath?: string,\n  ): Promise<Record<string, NodeResultType>> {\n    let breakCount = 0;\n    const totalChildren = node.children?.length ?? 0;\n    const allChildren = node.children ?? [];\n\n    // Narrative: capture the fan-out\n    const childDisplayNames = allChildren.map((c) => c.name);\n    this.deps.narrativeGenerator.onFork(node.name, childDisplayNames);\n\n    const childPromises: Promise<NodeResultType>[] = allChildren.map((child) => {\n      const childBranchPath = branchPath || child.id;\n      const childContext = context.createChild(childBranchPath as string, child.id as string, child.name, child.id);\n      const childBreakFlag = { shouldBreak: false };\n\n      const updateParentBreakFlag = () => {\n        if (childBreakFlag.shouldBreak) breakCount += 1;\n        if (parentBreakFlag && breakCount === totalChildren) parentBreakFlag.shouldBreak = true;\n      };\n\n      return this.executeNode(child, childContext, childBreakFlag, childBranchPath)\n        .then((result) => {\n          childContext.commit();\n          updateParentBreakFlag();\n          return { id: child.id!, result, isError: false };\n        })\n        .catch((error) => {\n          childContext.commit();\n          updateParentBreakFlag();\n          this.deps.logger.info(`TREE PIPELINE: executeNodeChildren - Error for id: ${child?.id}`, { error });\n          if (this.deps.throttlingErrorChecker && this.deps.throttlingErrorChecker(error)) {\n            childContext.updateObject(['monitor'], 'isThrottled', true);\n          }\n          return { id: child.id!, result: error, isError: true };\n        });\n    });\n\n    const childrenResults: Record<string, NodeResultType> = {};\n\n    if (node.failFast) {\n      // Fail-fast: first child error rejects immediately (unwrapped)\n      const results = await Promise.all(\n        allChildren.map((child, i) =>\n          childPromises[i].then((r) => {\n            if (r.isError) throw r.result;\n            return r;\n          }),\n        ),\n      );\n      for (const { id, result, isError } of results) {\n        childrenResults[id] = { id, result, isError: isError ?? false };\n      }\n    } else {\n      // Default: run all children to completion even if some fail\n      const settled = await Promise.allSettled(childPromises);\n      settled.forEach((s) => {\n        if (s.status === 'fulfilled') {\n          const { id, result, isError } = s.value;\n          childrenResults[id] = { id, result, isError: isError ?? false };\n        } else {\n          this.deps.logger.error(`Execution failed: ${s.reason}`);\n        }\n      });\n    }\n\n    return childrenResults;\n  }\n\n  /**\n   * Execute selected children based on selector result.\n   * Validates IDs, records selection info, then delegates to executeNodeChildren.\n   */\n  async executeSelectedChildren(\n    selector: Selector,\n    children: StageNode<TOut, TScope>[],\n    input: any,\n    context: StageContext,\n    branchPath: string,\n  ): Promise<Record<string, NodeResultType>> {\n    const selectorResult = await selector(input);\n    const selectedIds = Array.isArray(selectorResult) ? selectorResult : [selectorResult];\n\n    context.addLog('selectedChildIds', selectedIds);\n    context.addLog('selectorPattern', 'multi-choice');\n\n    if (selectedIds.length === 0) {\n      context.addLog('skippedAllChildren', true);\n      return {};\n    }\n\n    const selectedChildren = children.filter((c) => selectedIds.includes(c.id!));\n\n    // Validate all IDs exist (fail fast)\n    if (selectedChildren.length !== selectedIds.length) {\n      const childIds = children.map((c) => c.id);\n      const missing = selectedIds.filter((id) => !childIds.includes(id));\n      const errorMessage = `Selector returned unknown child IDs: ${missing.join(', ')}. Available: ${childIds.join(\n        ', ',\n      )}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}):`, { error: errorMessage });\n      context.addError('selectorError', errorMessage);\n      throw new Error(errorMessage);\n    }\n\n    const skippedIds = children.filter((c) => !selectedIds.includes(c.id!)).map((c) => c.id);\n    if (skippedIds.length > 0) {\n      context.addLog('skippedChildIds', skippedIds);\n    }\n\n    const selectedNames = selectedChildren.map((c) => c.name).join(', ');\n    context.addFlowDebugMessage(\n      'selected',\n      `Running ${selectedNames} (${selectedChildren.length} of ${children.length} matched)`,\n      { count: selectedChildren.length, targetStage: selectedChildren.map((c) => c.name) },\n    );\n\n    // Narrative: capture the selection\n    const selectedDisplayNames = selectedChildren.map((c) => c.name);\n    this.deps.narrativeGenerator.onSelected(context.stageName || 'selector', selectedDisplayNames, children.length);\n\n    const tempNode: StageNode<TOut, TScope> = {\n      name: 'selector-temp',\n      id: 'selector-temp',\n      children: selectedChildren,\n    };\n    return await this.executeNodeChildren(tempNode, context, undefined, branchPath);\n  }\n}\n"]}
|
|
120
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ChildrenExecutor.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/ChildrenExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ChildrenExecutor — Parallel fan-out via Promise.allSettled.\n *\n * Responsibilities:\n * - Execute all children in parallel (fork pattern)\n * - Execute selected children based on selector output (multi-choice)\n * - Handle throttling error flagging for rate-limited operations\n * - Aggregate results into { childId: { result, isError } }\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { Selector, StageNode } from '../graph/StageNode.js';\nimport type { TraversalContext } from '../narrative/types.js';\nimport type { HandlerDeps, NodeResultType } from '../types.js';\n\n/** Callback for recursive node execution. Avoids circular dependency with traverser. */\nexport type ExecuteNodeFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  breakFlag: { shouldBreak: boolean },\n  branchPath?: string,\n) => Promise<any>;\n\nexport class ChildrenExecutor<TOut = any, TScope = any> {\n  constructor(private deps: HandlerDeps<TOut, TScope>, private executeNode: ExecuteNodeFn<TOut, TScope>) {}\n\n  /**\n   * Execute all children in parallel. Each child commits on settle.\n   * Uses Promise.allSettled to ensure all children complete even if some fail.\n   */\n  async executeNodeChildren(\n    node: StageNode<TOut, TScope>,\n    context: StageContext,\n    parentBreakFlag?: { shouldBreak: boolean },\n    branchPath?: string,\n    traversalContext?: TraversalContext,\n  ): Promise<Record<string, NodeResultType>> {\n    let breakCount = 0;\n    const totalChildren = node.children?.length ?? 0;\n    const allChildren = node.children ?? [];\n\n    // Narrative: capture the fan-out\n    const childDisplayNames = allChildren.map((c) => c.name);\n    this.deps.narrativeGenerator.onFork(node.name, childDisplayNames, traversalContext);\n\n    const childPromises: Promise<NodeResultType>[] = allChildren.map((child) => {\n      const childBranchPath = branchPath || child.id;\n      const childContext = context.createChild(childBranchPath as string, child.id as string, child.name, child.id);\n      const childBreakFlag = { shouldBreak: false };\n\n      const updateParentBreakFlag = () => {\n        if (childBreakFlag.shouldBreak) breakCount += 1;\n        if (parentBreakFlag && breakCount === totalChildren) parentBreakFlag.shouldBreak = true;\n      };\n\n      return this.executeNode(child, childContext, childBreakFlag, childBranchPath)\n        .then((result) => {\n          childContext.commit();\n          updateParentBreakFlag();\n          return { id: child.id!, result, isError: false };\n        })\n        .catch((error) => {\n          childContext.commit();\n          updateParentBreakFlag();\n          this.deps.logger.info(`TREE PIPELINE: executeNodeChildren - Error for id: ${child?.id}`, { error });\n          if (this.deps.throttlingErrorChecker && this.deps.throttlingErrorChecker(error)) {\n            childContext.updateObject(['monitor'], 'isThrottled', true);\n          }\n          return { id: child.id!, result: error, isError: true };\n        });\n    });\n\n    const childrenResults: Record<string, NodeResultType> = {};\n\n    if (node.failFast) {\n      // Fail-fast: first child error rejects immediately (unwrapped)\n      const results = await Promise.all(\n        allChildren.map((child, i) =>\n          childPromises[i].then((r) => {\n            if (r.isError) throw r.result;\n            return r;\n          }),\n        ),\n      );\n      for (const { id, result, isError } of results) {\n        childrenResults[id] = { id, result, isError: isError ?? false };\n      }\n    } else {\n      // Default: run all children to completion even if some fail\n      const settled = await Promise.allSettled(childPromises);\n      settled.forEach((s) => {\n        if (s.status === 'fulfilled') {\n          const { id, result, isError } = s.value;\n          childrenResults[id] = { id, result, isError: isError ?? false };\n        } else {\n          this.deps.logger.error(`Execution failed: ${s.reason}`);\n        }\n      });\n    }\n\n    return childrenResults;\n  }\n\n  /**\n   * Execute selected children based on selector result.\n   * Validates IDs, records selection info, then delegates to executeNodeChildren.\n   */\n  async executeSelectedChildren(\n    selector: Selector,\n    children: StageNode<TOut, TScope>[],\n    input: any,\n    context: StageContext,\n    branchPath: string,\n    traversalContext?: TraversalContext,\n  ): Promise<Record<string, NodeResultType>> {\n    const selectorResult = await selector(input);\n    const selectedIds = Array.isArray(selectorResult) ? selectorResult : [selectorResult];\n\n    context.addLog('selectedChildIds', selectedIds);\n    context.addLog('selectorPattern', 'multi-choice');\n\n    if (selectedIds.length === 0) {\n      context.addLog('skippedAllChildren', true);\n      return {};\n    }\n\n    const selectedChildren = children.filter((c) => selectedIds.includes(c.id!));\n\n    // Validate all IDs exist (fail fast)\n    if (selectedChildren.length !== selectedIds.length) {\n      const childIds = children.map((c) => c.id);\n      const missing = selectedIds.filter((id) => !childIds.includes(id));\n      const errorMessage = `Selector returned unknown child IDs: ${missing.join(', ')}. Available: ${childIds.join(\n        ', ',\n      )}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}):`, { error: errorMessage });\n      context.addError('selectorError', errorMessage);\n      throw new Error(errorMessage);\n    }\n\n    const skippedIds = children.filter((c) => !selectedIds.includes(c.id!)).map((c) => c.id);\n    if (skippedIds.length > 0) {\n      context.addLog('skippedChildIds', skippedIds);\n    }\n\n    const selectedNames = selectedChildren.map((c) => c.name).join(', ');\n    context.addFlowDebugMessage(\n      'selected',\n      `Running ${selectedNames} (${selectedChildren.length} of ${children.length} matched)`,\n      { count: selectedChildren.length, targetStage: selectedChildren.map((c) => c.name) },\n    );\n\n    // Narrative: capture the selection\n    const selectedDisplayNames = selectedChildren.map((c) => c.name);\n    this.deps.narrativeGenerator.onSelected(\n      context.stageName || 'selector',\n      selectedDisplayNames,\n      children.length,\n      traversalContext,\n    );\n\n    const tempNode: StageNode<TOut, TScope> = {\n      name: 'selector-temp',\n      id: 'selector-temp',\n      children: selectedChildren,\n    };\n    return await this.executeNodeChildren(tempNode, context, undefined, branchPath, traversalContext);\n  }\n}\n"]}
|
|
@@ -27,17 +27,17 @@ export class ContinuationResolver {
|
|
|
27
27
|
* Dispatches to handleStringReference, handleDirectNode, or handleNodeReference
|
|
28
28
|
* based on the dynamicNext type.
|
|
29
29
|
*/
|
|
30
|
-
async resolve(dynamicNext, node, context, breakFlag, branchPath, executeNode) {
|
|
30
|
+
async resolve(dynamicNext, node, context, breakFlag, branchPath, executeNode, traversalContext) {
|
|
31
31
|
if (typeof dynamicNext === 'string') {
|
|
32
|
-
return this.handleStringReference(dynamicNext, node, context, breakFlag, branchPath, executeNode);
|
|
32
|
+
return this.handleStringReference(dynamicNext, node, context, breakFlag, branchPath, executeNode, traversalContext);
|
|
33
33
|
}
|
|
34
34
|
if (dynamicNext.fn) {
|
|
35
35
|
return this.handleDirectNode(dynamicNext, context, breakFlag, branchPath, executeNode);
|
|
36
36
|
}
|
|
37
|
-
return this.handleNodeReference(dynamicNext, node, context, breakFlag, branchPath, executeNode);
|
|
37
|
+
return this.handleNodeReference(dynamicNext, node, context, breakFlag, branchPath, executeNode, traversalContext);
|
|
38
38
|
}
|
|
39
39
|
/** dynamicNext is a string ID → resolve from graph, track iteration. */
|
|
40
|
-
async handleStringReference(nodeId, currentNode, context, breakFlag, branchPath, executeNode) {
|
|
40
|
+
async handleStringReference(nodeId, currentNode, context, breakFlag, branchPath, executeNode, traversalContext) {
|
|
41
41
|
const targetNode = this.nodeResolver.findNodeById(nodeId);
|
|
42
42
|
if (!targetNode) {
|
|
43
43
|
const errorMessage = `dynamicNext target node not found: ${nodeId}`;
|
|
@@ -52,7 +52,7 @@ export class ContinuationResolver {
|
|
|
52
52
|
targetStage: targetNode.name,
|
|
53
53
|
iteration: iteration + 1,
|
|
54
54
|
});
|
|
55
|
-
this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description);
|
|
55
|
+
this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description, traversalContext);
|
|
56
56
|
const nextStageContext = context.createNext(branchPath, iteratedStageName, targetNode.id);
|
|
57
57
|
return executeNode(targetNode, nextStageContext, breakFlag, branchPath);
|
|
58
58
|
}
|
|
@@ -67,7 +67,7 @@ export class ContinuationResolver {
|
|
|
67
67
|
return executeNode(dynamicNode, nextStageContext, breakFlag, branchPath);
|
|
68
68
|
}
|
|
69
69
|
/** dynamicNext is a StageNode without fn → reference by ID, resolve + track iteration. */
|
|
70
|
-
async handleNodeReference(dynamicNode, currentNode, context, breakFlag, branchPath, executeNode) {
|
|
70
|
+
async handleNodeReference(dynamicNode, currentNode, context, breakFlag, branchPath, executeNode, traversalContext) {
|
|
71
71
|
const nextNodeId = dynamicNode.id;
|
|
72
72
|
if (!nextNodeId) {
|
|
73
73
|
const errorMessage = 'dynamicNext node must have an id when used as reference';
|
|
@@ -88,7 +88,7 @@ export class ContinuationResolver {
|
|
|
88
88
|
targetStage: targetNode.name,
|
|
89
89
|
iteration: iteration + 1,
|
|
90
90
|
});
|
|
91
|
-
this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description);
|
|
91
|
+
this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description, traversalContext);
|
|
92
92
|
const nextStageContext = context.createNext(branchPath, iteratedStageName, targetNode.id);
|
|
93
93
|
return executeNode(targetNode, nextStageContext, breakFlag, branchPath);
|
|
94
94
|
}
|
|
@@ -118,4 +118,4 @@ export class ContinuationResolver {
|
|
|
118
118
|
return iteration === 0 ? baseName : `${baseName}.${iteration}`;
|
|
119
119
|
}
|
|
120
120
|
}
|
|
121
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ContinuationResolver.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/ContinuationResolver.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAeH,MAAM,CAAC,MAAM,sBAAsB,GAAG,IAAI,CAAC;AAE3C,MAAM,OAAO,oBAAoB;IAU/B,YACmB,IAA+B,EAC/B,YAAwC,EACzD,iBAA2D,EAC3D,aAAsB;QAHL,SAAI,GAAJ,IAAI,CAA2B;QAC/B,iBAAY,GAAZ,YAAY,CAA4B;QAX3D;;;WAGG;QACK,sBAAiB,GAAwB,IAAI,GAAG,EAAE,CAAC;QAWzD,IAAI,CAAC,iBAAiB,GAAG,iBAAiB,CAAC;QAC3C,IAAI,CAAC,aAAa,GAAG,aAAa,aAAb,aAAa,cAAb,aAAa,GAAI,sBAAsB,CAAC;IAC/D,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,OAAO,CACX,WAA6C,EAC7C,IAA6B,EAC7B,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC;QAExC,IAAI,OAAO,WAAW,KAAK,QAAQ,EAAE,CAAC;YACpC,OAAO,IAAI,CAAC,qBAAqB,CAAC,WAAW,EAAE,IAAI,EAAE,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;QACpG,CAAC;QAED,IAAI,WAAW,CAAC,EAAE,EAAE,CAAC;YACnB,OAAO,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;QACzF,CAAC;QAED,OAAO,IAAI,CAAC,mBAAmB,CAAC,WAAW,EAAE,IAAI,EAAE,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;IAClG,CAAC;IAED,wEAAwE;IAChE,KAAK,CAAC,qBAAqB,CACjC,MAAc,EACd,WAAoC,EACpC,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC;QAExC,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAC1D,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,YAAY,GAAG,sCAAsC,MAAM,EAAE,CAAC;YACpE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,sBAAsB,UAAU,YAAY,WAAW,CAAC,IAAI,IAAI,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAClH,MAAM,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAChC,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC;QACxD,MAAM,iBAAiB,GAAG,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;QAChF,OAAO,CAAC,MAAM,CAAC,mBAAmB,EAAE,MAAM,CAAC,CAAC;QAC5C,OAAO,CAAC,MAAM,CAAC,sBAAsB,EAAE,SAAS,CAAC,CAAC;QAElD,OAAO,CAAC,mBAAmB,CAAC,MAAM,EAAE,mBAAmB,UAAU,CAAC,IAAI,eAAe,SAAS,GAAG,CAAC,GAAG,EAAE;YACrG,WAAW,EAAE,UAAU,CAAC,IAAI;YAC5B,SAAS,EAAE,SAAS,GAAG,CAAC;SACzB,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,GAAG,CAAC,EAAE,UAAU,CAAC,WAAW,CAAC,CAAC;QAE5F,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAoB,EAAE,iBAAiB,EAAE,UAAU,CAAC,EAAE,CAAC,CAAC;QACpG,OAAO,WAAW,CAAC,UAAU,EAAE,gBAAgB,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IAC1E,CAAC;IAED,6EAA6E;IACrE,KAAK,CAAC,gBAAgB,CAC5B,WAAoC,EACpC,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC;QAExC,OAAO,CAAC,MAAM,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;QAC1C,OAAO,CAAC,MAAM,CAAC,iBAAiB,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC;QAEpD,OAAO,CAAC,mBAAmB,CAAC,MAAM,EAAE,aAAa,WAAW,CAAC,IAAI,kBAAkB,EAAE;YACnF,WAAW,EAAE,WAAW,CAAC,IAAI;SAC9B,CAAC,CAAC;QAEH,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAoB,EAAE,WAAW,CAAC,IAAI,EAAE,WAAW,CAAC,EAAE,CAAC,CAAC;QACpG,OAAO,WAAW,CAAC,WAAW,EAAE,gBAAgB,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IAC3E,CAAC;IAED,0FAA0F;IAClF,KAAK,CAAC,mBAAmB,CAC/B,WAAoC,EACpC,WAAoC,EACpC,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC;QAExC,MAAM,UAAU,GAAG,WAAW,CAAC,EAAE,CAAC;QAClC,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,YAAY,GAAG,yDAAyD,CAAC;YAC/E,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,sBAAsB,UAAU,YAAY,WAAW,CAAC,IAAI,IAAI,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAClH,MAAM,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAChC,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;QAC9D,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,YAAY,GAAG,sCAAsC,UAAU,EAAE,CAAC;YACxE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,sBAAsB,UAAU,YAAY,WAAW,CAAC,IAAI,IAAI,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAClH,MAAM,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAChC,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,wBAAwB,CAAC,UAAU,CAAC,CAAC;QAC5D,MAAM,iBAAiB,GAAG,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;QAChF,OAAO,CAAC,MAAM,CAAC,mBAAmB,EAAE,UAAU,CAAC,CAAC;QAChD,OAAO,CAAC,MAAM,CAAC,sBAAsB,EAAE,SAAS,CAAC,CAAC;QAElD,OAAO,CAAC,mBAAmB,CAAC,MAAM,EAAE,mBAAmB,UAAU,CAAC,IAAI,eAAe,SAAS,GAAG,CAAC,GAAG,EAAE;YACrG,WAAW,EAAE,UAAU,CAAC,IAAI;YAC5B,SAAS,EAAE,SAAS,GAAG,CAAC;SACzB,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,GAAG,CAAC,EAAE,UAAU,CAAC,WAAW,CAAC,CAAC;QAE5F,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAoB,EAAE,iBAAiB,EAAE,UAAU,CAAC,EAAE,CAAC,CAAC;QACpG,OAAO,WAAW,CAAC,UAAU,EAAE,gBAAgB,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IAC1E,CAAC;IAED;;;;OAIG;IACH,wBAAwB,CAAC,MAAc;;QACrC,MAAM,OAAO,GAAG,MAAA,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,MAAM,CAAC,mCAAI,CAAC,CAAC;QACxD,IAAI,OAAO,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YAClC,MAAM,IAAI,KAAK,CACb,4BAA4B,IAAI,CAAC,aAAa,wBAAwB,MAAM,KAAK;gBAC/E,0CAA0C,CAC7C,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,GAAG,CAAC,CAAC,CAAC;QAEhD,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,OAAO,GAAG,CAAC,CAAC,CAAC;QAC9C,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,QAAgB,EAAE,SAAiB;QACtD,OAAO,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,QAAQ,IAAI,SAAS,EAAE,CAAC;IACjE,CAAC;CACF","sourcesContent":["/**\n * ContinuationResolver — Back-edge resolution + iteration counting.\n *\n * Resolves dynamic continuations (loop-backs, dynamic next) and tracks\n * per-node iteration counts for context tree naming.\n *\n * Supports three dynamicNext patterns:\n * - String ID → reference to existing node (resolve via NodeResolver)\n * - StageNode with fn → truly dynamic node (execute directly)\n * - StageNode without fn → reference by ID (resolve via NodeResolver)\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { HandlerDeps } from '../types.js';\nimport type { NodeResolver } from './NodeResolver.js';\n\n/** Callback for recursive node execution. Avoids circular dep with traverser. */\nexport type ExecuteNodeFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  breakFlag: { shouldBreak: boolean },\n  branchPath?: string,\n) => Promise<any>;\n\nexport const DEFAULT_MAX_ITERATIONS = 1000;\n\nexport class ContinuationResolver<TOut = any, TScope = any> {\n  /**\n   * Iteration counter per node ID.\n   * Key: node.id, Value: visit count (0 = first visit).\n   */\n  private iterationCounters: Map<string, number> = new Map();\n\n  private readonly onIterationUpdate?: (nodeId: string, count: number) => void;\n  private readonly maxIterations: number;\n\n  constructor(\n    private readonly deps: HandlerDeps<TOut, TScope>,\n    private readonly nodeResolver: NodeResolver<TOut, TScope>,\n    onIterationUpdate?: (nodeId: string, count: number) => void,\n    maxIterations?: number,\n  ) {\n    this.onIterationUpdate = onIterationUpdate;\n    this.maxIterations = maxIterations ?? DEFAULT_MAX_ITERATIONS;\n  }\n\n  /**\n   * Resolve a dynamic continuation.\n   * Dispatches to handleStringReference, handleDirectNode, or handleNodeReference\n   * based on the dynamicNext type.\n   */\n  async resolve(\n    dynamicNext: string | StageNode<TOut, TScope>,\n    node: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n  ): Promise<any> {\n    if (typeof dynamicNext === 'string') {\n      return this.handleStringReference(dynamicNext, node, context, breakFlag, branchPath, executeNode);\n    }\n\n    if (dynamicNext.fn) {\n      return this.handleDirectNode(dynamicNext, context, breakFlag, branchPath, executeNode);\n    }\n\n    return this.handleNodeReference(dynamicNext, node, context, breakFlag, branchPath, executeNode);\n  }\n\n  /** dynamicNext is a string ID → resolve from graph, track iteration. */\n  private async handleStringReference(\n    nodeId: string,\n    currentNode: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n  ): Promise<any> {\n    const targetNode = this.nodeResolver.findNodeById(nodeId);\n    if (!targetNode) {\n      const errorMessage = `dynamicNext target node not found: ${nodeId}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${currentNode.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n\n    const iteration = this.getAndIncrementIteration(nodeId);\n    const iteratedStageName = this.getIteratedStageName(targetNode.name, iteration);\n    context.addLog('dynamicNextTarget', nodeId);\n    context.addLog('dynamicNextIteration', iteration);\n\n    context.addFlowDebugMessage('loop', `Looping back to ${targetNode.name} (iteration ${iteration + 1})`, {\n      targetStage: targetNode.name,\n      iteration: iteration + 1,\n    });\n\n    this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description);\n\n    const nextStageContext = context.createNext(branchPath as string, iteratedStageName, targetNode.id);\n    return executeNode(targetNode, nextStageContext, breakFlag, branchPath);\n  }\n\n  /** dynamicNext is a StageNode with fn → execute directly (truly dynamic). */\n  private async handleDirectNode(\n    dynamicNode: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n  ): Promise<any> {\n    context.addLog('dynamicNextDirect', true);\n    context.addLog('dynamicNextName', dynamicNode.name);\n\n    context.addFlowDebugMessage('next', `Moving to ${dynamicNode.name} stage (dynamic)`, {\n      targetStage: dynamicNode.name,\n    });\n\n    const nextStageContext = context.createNext(branchPath as string, dynamicNode.name, dynamicNode.id);\n    return executeNode(dynamicNode, nextStageContext, breakFlag, branchPath);\n  }\n\n  /** dynamicNext is a StageNode without fn → reference by ID, resolve + track iteration. */\n  private async handleNodeReference(\n    dynamicNode: StageNode<TOut, TScope>,\n    currentNode: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n  ): Promise<any> {\n    const nextNodeId = dynamicNode.id;\n    if (!nextNodeId) {\n      const errorMessage = 'dynamicNext node must have an id when used as reference';\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${currentNode.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n\n    const targetNode = this.nodeResolver.findNodeById(nextNodeId);\n    if (!targetNode) {\n      const errorMessage = `dynamicNext target node not found: ${nextNodeId}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${currentNode.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n\n    const iteration = this.getAndIncrementIteration(nextNodeId);\n    const iteratedStageName = this.getIteratedStageName(targetNode.name, iteration);\n    context.addLog('dynamicNextTarget', nextNodeId);\n    context.addLog('dynamicNextIteration', iteration);\n\n    context.addFlowDebugMessage('loop', `Looping back to ${targetNode.name} (iteration ${iteration + 1})`, {\n      targetStage: targetNode.name,\n      iteration: iteration + 1,\n    });\n\n    this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description);\n\n    const nextStageContext = context.createNext(branchPath as string, iteratedStageName, targetNode.id);\n    return executeNode(targetNode, nextStageContext, breakFlag, branchPath);\n  }\n\n  /**\n   * Get the next iteration number for a node and increment.\n   * Returns 0 for first visit, 1 for second, etc.\n   * Throws if maxIterations exceeded (infinite loop guard).\n   */\n  getAndIncrementIteration(nodeId: string): number {\n    const current = this.iterationCounters.get(nodeId) ?? 0;\n    if (current >= this.maxIterations) {\n      throw new Error(\n        `Maximum loop iterations (${this.maxIterations}) exceeded for node '${nodeId}'. ` +\n          'Set maxIterations to increase the limit.',\n      );\n    }\n    this.iterationCounters.set(nodeId, current + 1);\n\n    if (this.onIterationUpdate) {\n      this.onIterationUpdate(nodeId, current + 1);\n    }\n\n    return current;\n  }\n\n  /**\n   * Generate an iterated stage name for context tree.\n   * First visit: \"askLLM\", second: \"askLLM.1\", third: \"askLLM.2\".\n   */\n  getIteratedStageName(baseName: string, iteration: number): string {\n    return iteration === 0 ? baseName : `${baseName}.${iteration}`;\n  }\n}\n"]}
|
|
121
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ContinuationResolver.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/ContinuationResolver.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAgBH,MAAM,CAAC,MAAM,sBAAsB,GAAG,IAAI,CAAC;AAE3C,MAAM,OAAO,oBAAoB;IAU/B,YACmB,IAA+B,EAC/B,YAAwC,EACzD,iBAA2D,EAC3D,aAAsB;QAHL,SAAI,GAAJ,IAAI,CAA2B;QAC/B,iBAAY,GAAZ,YAAY,CAA4B;QAX3D;;;WAGG;QACK,sBAAiB,GAAwB,IAAI,GAAG,EAAE,CAAC;QAWzD,IAAI,CAAC,iBAAiB,GAAG,iBAAiB,CAAC;QAC3C,IAAI,CAAC,aAAa,GAAG,aAAa,aAAb,aAAa,cAAb,aAAa,GAAI,sBAAsB,CAAC;IAC/D,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,OAAO,CACX,WAA6C,EAC7C,IAA6B,EAC7B,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC,EACxC,gBAAmC;QAEnC,IAAI,OAAO,WAAW,KAAK,QAAQ,EAAE,CAAC;YACpC,OAAO,IAAI,CAAC,qBAAqB,CAC/B,WAAW,EACX,IAAI,EACJ,OAAO,EACP,SAAS,EACT,UAAU,EACV,WAAW,EACX,gBAAgB,CACjB,CAAC;QACJ,CAAC;QAED,IAAI,WAAW,CAAC,EAAE,EAAE,CAAC;YACnB,OAAO,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;QACzF,CAAC;QAED,OAAO,IAAI,CAAC,mBAAmB,CAAC,WAAW,EAAE,IAAI,EAAE,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,WAAW,EAAE,gBAAgB,CAAC,CAAC;IACpH,CAAC;IAED,wEAAwE;IAChE,KAAK,CAAC,qBAAqB,CACjC,MAAc,EACd,WAAoC,EACpC,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC,EACxC,gBAAmC;QAEnC,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAC1D,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,YAAY,GAAG,sCAAsC,MAAM,EAAE,CAAC;YACpE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,sBAAsB,UAAU,YAAY,WAAW,CAAC,IAAI,IAAI,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAClH,MAAM,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAChC,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC;QACxD,MAAM,iBAAiB,GAAG,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;QAChF,OAAO,CAAC,MAAM,CAAC,mBAAmB,EAAE,MAAM,CAAC,CAAC;QAC5C,OAAO,CAAC,MAAM,CAAC,sBAAsB,EAAE,SAAS,CAAC,CAAC;QAElD,OAAO,CAAC,mBAAmB,CAAC,MAAM,EAAE,mBAAmB,UAAU,CAAC,IAAI,eAAe,SAAS,GAAG,CAAC,GAAG,EAAE;YACrG,WAAW,EAAE,UAAU,CAAC,IAAI;YAC5B,SAAS,EAAE,SAAS,GAAG,CAAC;SACzB,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,GAAG,CAAC,EAAE,UAAU,CAAC,WAAW,EAAE,gBAAgB,CAAC,CAAC;QAE9G,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAoB,EAAE,iBAAiB,EAAE,UAAU,CAAC,EAAE,CAAC,CAAC;QACpG,OAAO,WAAW,CAAC,UAAU,EAAE,gBAAgB,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IAC1E,CAAC;IAED,6EAA6E;IACrE,KAAK,CAAC,gBAAgB,CAC5B,WAAoC,EACpC,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC;QAExC,OAAO,CAAC,MAAM,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;QAC1C,OAAO,CAAC,MAAM,CAAC,iBAAiB,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC;QAEpD,OAAO,CAAC,mBAAmB,CAAC,MAAM,EAAE,aAAa,WAAW,CAAC,IAAI,kBAAkB,EAAE;YACnF,WAAW,EAAE,WAAW,CAAC,IAAI;SAC9B,CAAC,CAAC;QAEH,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAoB,EAAE,WAAW,CAAC,IAAI,EAAE,WAAW,CAAC,EAAE,CAAC,CAAC;QACpG,OAAO,WAAW,CAAC,WAAW,EAAE,gBAAgB,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IAC3E,CAAC;IAED,0FAA0F;IAClF,KAAK,CAAC,mBAAmB,CAC/B,WAAoC,EACpC,WAAoC,EACpC,OAAqB,EACrB,SAAmC,EACnC,UAA8B,EAC9B,WAAwC,EACxC,gBAAmC;QAEnC,MAAM,UAAU,GAAG,WAAW,CAAC,EAAE,CAAC;QAClC,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,YAAY,GAAG,yDAAyD,CAAC;YAC/E,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,sBAAsB,UAAU,YAAY,WAAW,CAAC,IAAI,IAAI,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAClH,MAAM,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAChC,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;QAC9D,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,MAAM,YAAY,GAAG,sCAAsC,UAAU,EAAE,CAAC;YACxE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,sBAAsB,UAAU,YAAY,WAAW,CAAC,IAAI,IAAI,EAAE,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAClH,MAAM,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC;QAChC,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,wBAAwB,CAAC,UAAU,CAAC,CAAC;QAC5D,MAAM,iBAAiB,GAAG,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;QAChF,OAAO,CAAC,MAAM,CAAC,mBAAmB,EAAE,UAAU,CAAC,CAAC;QAChD,OAAO,CAAC,MAAM,CAAC,sBAAsB,EAAE,SAAS,CAAC,CAAC;QAElD,OAAO,CAAC,mBAAmB,CAAC,MAAM,EAAE,mBAAmB,UAAU,CAAC,IAAI,eAAe,SAAS,GAAG,CAAC,GAAG,EAAE;YACrG,WAAW,EAAE,UAAU,CAAC,IAAI;YAC5B,SAAS,EAAE,SAAS,GAAG,CAAC;SACzB,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,GAAG,CAAC,EAAE,UAAU,CAAC,WAAW,EAAE,gBAAgB,CAAC,CAAC;QAE9G,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAoB,EAAE,iBAAiB,EAAE,UAAU,CAAC,EAAE,CAAC,CAAC;QACpG,OAAO,WAAW,CAAC,UAAU,EAAE,gBAAgB,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IAC1E,CAAC;IAED;;;;OAIG;IACH,wBAAwB,CAAC,MAAc;;QACrC,MAAM,OAAO,GAAG,MAAA,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,MAAM,CAAC,mCAAI,CAAC,CAAC;QACxD,IAAI,OAAO,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YAClC,MAAM,IAAI,KAAK,CACb,4BAA4B,IAAI,CAAC,aAAa,wBAAwB,MAAM,KAAK;gBAC/E,0CAA0C,CAC7C,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,GAAG,CAAC,CAAC,CAAC;QAEhD,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,OAAO,GAAG,CAAC,CAAC,CAAC;QAC9C,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,QAAgB,EAAE,SAAiB;QACtD,OAAO,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,QAAQ,IAAI,SAAS,EAAE,CAAC;IACjE,CAAC;CACF","sourcesContent":["/**\n * ContinuationResolver — Back-edge resolution + iteration counting.\n *\n * Resolves dynamic continuations (loop-backs, dynamic next) and tracks\n * per-node iteration counts for context tree naming.\n *\n * Supports three dynamicNext patterns:\n * - String ID → reference to existing node (resolve via NodeResolver)\n * - StageNode with fn → truly dynamic node (execute directly)\n * - StageNode without fn → reference by ID (resolve via NodeResolver)\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { TraversalContext } from '../narrative/types.js';\nimport type { HandlerDeps } from '../types.js';\nimport type { NodeResolver } from './NodeResolver.js';\n\n/** Callback for recursive node execution. Avoids circular dep with traverser. */\nexport type ExecuteNodeFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  breakFlag: { shouldBreak: boolean },\n  branchPath?: string,\n) => Promise<any>;\n\nexport const DEFAULT_MAX_ITERATIONS = 1000;\n\nexport class ContinuationResolver<TOut = any, TScope = any> {\n  /**\n   * Iteration counter per node ID.\n   * Key: node.id, Value: visit count (0 = first visit).\n   */\n  private iterationCounters: Map<string, number> = new Map();\n\n  private readonly onIterationUpdate?: (nodeId: string, count: number) => void;\n  private readonly maxIterations: number;\n\n  constructor(\n    private readonly deps: HandlerDeps<TOut, TScope>,\n    private readonly nodeResolver: NodeResolver<TOut, TScope>,\n    onIterationUpdate?: (nodeId: string, count: number) => void,\n    maxIterations?: number,\n  ) {\n    this.onIterationUpdate = onIterationUpdate;\n    this.maxIterations = maxIterations ?? DEFAULT_MAX_ITERATIONS;\n  }\n\n  /**\n   * Resolve a dynamic continuation.\n   * Dispatches to handleStringReference, handleDirectNode, or handleNodeReference\n   * based on the dynamicNext type.\n   */\n  async resolve(\n    dynamicNext: string | StageNode<TOut, TScope>,\n    node: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n    traversalContext?: TraversalContext,\n  ): Promise<any> {\n    if (typeof dynamicNext === 'string') {\n      return this.handleStringReference(\n        dynamicNext,\n        node,\n        context,\n        breakFlag,\n        branchPath,\n        executeNode,\n        traversalContext,\n      );\n    }\n\n    if (dynamicNext.fn) {\n      return this.handleDirectNode(dynamicNext, context, breakFlag, branchPath, executeNode);\n    }\n\n    return this.handleNodeReference(dynamicNext, node, context, breakFlag, branchPath, executeNode, traversalContext);\n  }\n\n  /** dynamicNext is a string ID → resolve from graph, track iteration. */\n  private async handleStringReference(\n    nodeId: string,\n    currentNode: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n    traversalContext?: TraversalContext,\n  ): Promise<any> {\n    const targetNode = this.nodeResolver.findNodeById(nodeId);\n    if (!targetNode) {\n      const errorMessage = `dynamicNext target node not found: ${nodeId}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${currentNode.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n\n    const iteration = this.getAndIncrementIteration(nodeId);\n    const iteratedStageName = this.getIteratedStageName(targetNode.name, iteration);\n    context.addLog('dynamicNextTarget', nodeId);\n    context.addLog('dynamicNextIteration', iteration);\n\n    context.addFlowDebugMessage('loop', `Looping back to ${targetNode.name} (iteration ${iteration + 1})`, {\n      targetStage: targetNode.name,\n      iteration: iteration + 1,\n    });\n\n    this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description, traversalContext);\n\n    const nextStageContext = context.createNext(branchPath as string, iteratedStageName, targetNode.id);\n    return executeNode(targetNode, nextStageContext, breakFlag, branchPath);\n  }\n\n  /** dynamicNext is a StageNode with fn → execute directly (truly dynamic). */\n  private async handleDirectNode(\n    dynamicNode: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n  ): Promise<any> {\n    context.addLog('dynamicNextDirect', true);\n    context.addLog('dynamicNextName', dynamicNode.name);\n\n    context.addFlowDebugMessage('next', `Moving to ${dynamicNode.name} stage (dynamic)`, {\n      targetStage: dynamicNode.name,\n    });\n\n    const nextStageContext = context.createNext(branchPath as string, dynamicNode.name, dynamicNode.id);\n    return executeNode(dynamicNode, nextStageContext, breakFlag, branchPath);\n  }\n\n  /** dynamicNext is a StageNode without fn → reference by ID, resolve + track iteration. */\n  private async handleNodeReference(\n    dynamicNode: StageNode<TOut, TScope>,\n    currentNode: StageNode<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n    traversalContext?: TraversalContext,\n  ): Promise<any> {\n    const nextNodeId = dynamicNode.id;\n    if (!nextNodeId) {\n      const errorMessage = 'dynamicNext node must have an id when used as reference';\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${currentNode.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n\n    const targetNode = this.nodeResolver.findNodeById(nextNodeId);\n    if (!targetNode) {\n      const errorMessage = `dynamicNext target node not found: ${nextNodeId}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${currentNode.name}]:`, { error: errorMessage });\n      throw new Error(errorMessage);\n    }\n\n    const iteration = this.getAndIncrementIteration(nextNodeId);\n    const iteratedStageName = this.getIteratedStageName(targetNode.name, iteration);\n    context.addLog('dynamicNextTarget', nextNodeId);\n    context.addLog('dynamicNextIteration', iteration);\n\n    context.addFlowDebugMessage('loop', `Looping back to ${targetNode.name} (iteration ${iteration + 1})`, {\n      targetStage: targetNode.name,\n      iteration: iteration + 1,\n    });\n\n    this.deps.narrativeGenerator.onLoop(targetNode.name, iteration + 1, targetNode.description, traversalContext);\n\n    const nextStageContext = context.createNext(branchPath as string, iteratedStageName, targetNode.id);\n    return executeNode(targetNode, nextStageContext, breakFlag, branchPath);\n  }\n\n  /**\n   * Get the next iteration number for a node and increment.\n   * Returns 0 for first visit, 1 for second, etc.\n   * Throws if maxIterations exceeded (infinite loop guard).\n   */\n  getAndIncrementIteration(nodeId: string): number {\n    const current = this.iterationCounters.get(nodeId) ?? 0;\n    if (current >= this.maxIterations) {\n      throw new Error(\n        `Maximum loop iterations (${this.maxIterations}) exceeded for node '${nodeId}'. ` +\n          'Set maxIterations to increase the limit.',\n      );\n    }\n    this.iterationCounters.set(nodeId, current + 1);\n\n    if (this.onIterationUpdate) {\n      this.onIterationUpdate(nodeId, current + 1);\n    }\n\n    return current;\n  }\n\n  /**\n   * Generate an iterated stage name for context tree.\n   * First visit: \"askLLM\", second: \"askLLM.1\", third: \"askLLM.2\".\n   */\n  getIteratedStageName(baseName: string, iteration: number): string {\n    return iteration === 0 ? baseName : `${baseName}.${iteration}`;\n  }\n}\n"]}
|
|
@@ -13,7 +13,7 @@ export class DeciderHandler {
|
|
|
13
13
|
* The stage function IS the decider — its return value is the branch ID.
|
|
14
14
|
* Execution order: runStage(fn) → commit → resolve child → log → executeNode(child).
|
|
15
15
|
*/
|
|
16
|
-
async handleScopeBased(node, stageFunc, context, breakFlag, branchPath, runStage, executeNode, callExtractor, getStagePath) {
|
|
16
|
+
async handleScopeBased(node, stageFunc, context, breakFlag, branchPath, runStage, executeNode, callExtractor, getStagePath, traversalContext) {
|
|
17
17
|
var _a, _b;
|
|
18
18
|
const breakFn = () => (breakFlag.shouldBreak = true);
|
|
19
19
|
let branchId;
|
|
@@ -29,7 +29,7 @@ export class DeciderHandler {
|
|
|
29
29
|
});
|
|
30
30
|
this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });
|
|
31
31
|
context.addError('stageExecutionError', error.toString());
|
|
32
|
-
this.deps.narrativeGenerator.onError(node.name, error.toString(), error);
|
|
32
|
+
this.deps.narrativeGenerator.onError(node.name, error.toString(), error, traversalContext);
|
|
33
33
|
throw error;
|
|
34
34
|
}
|
|
35
35
|
context.commit();
|
|
@@ -70,9 +70,9 @@ export class DeciderHandler {
|
|
|
70
70
|
targetStage: chosen.name,
|
|
71
71
|
rationale: rationale || `returned branchId: ${branchId}`,
|
|
72
72
|
});
|
|
73
|
-
this.deps.narrativeGenerator.onDecision(node.name, chosen.name, rationale, node.description);
|
|
73
|
+
this.deps.narrativeGenerator.onDecision(node.name, chosen.name, rationale, node.description, traversalContext);
|
|
74
74
|
const branchContext = context.createChild(branchPath, chosen.id, chosen.name, chosen.id);
|
|
75
75
|
return executeNode(chosen, branchContext, breakFlag, branchPath);
|
|
76
76
|
}
|
|
77
77
|
}
|
|
78
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DeciderHandler.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/DeciderHandler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DeciderHandler — Single-choice conditional branching.\n *\n * Handles scope-based deciders (stage IS the decider, returns branch ID).\n * Logs flow control decisions and narrative sentences.\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { HandlerDeps, StageFunction } from '../types.js';\n\n/** Callback for running a stage with commit + extractor. Avoids circular dep with traverser. */\nexport type RunStageFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  stageFunc: StageFunction<TOut, TScope>,\n  context: StageContext,\n  breakFn: () => void,\n) => Promise<TOut>;\n\n/** Callback for recursive node execution. Avoids circular dep with traverser. */\nexport type ExecuteNodeFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  breakFlag: { shouldBreak: boolean },\n  branchPath?: string,\n) => Promise<any>;\n\n/** Callback for calling the extractor after stage execution. */\nexport type CallExtractorFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  stagePath: string,\n  stageOutput?: unknown,\n  errorInfo?: { type: string; message: string },\n) => void;\n\n/** Callback for computing the stage path for extractor. */\nexport type GetStagePathFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  branchPath?: string,\n  contextStageName?: string,\n) => string;\n\nexport class DeciderHandler<TOut = any, TScope = any> {\n  constructor(private readonly deps: HandlerDeps<TOut, TScope>) {}\n\n  /**\n   * Handle a scope-based decider (created via addDeciderFunction).\n   * The stage function IS the decider — its return value is the branch ID.\n   * Execution order: runStage(fn) → commit → resolve child → log → executeNode(child).\n   */\n  async handleScopeBased(\n    node: StageNode<TOut, TScope>,\n    stageFunc: StageFunction<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    runStage: RunStageFn<TOut, TScope>,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n    callExtractor: CallExtractorFn<TOut, TScope>,\n    getStagePath: GetStagePathFn<TOut, TScope>,\n  ): Promise<any> {\n    const breakFn = () => (breakFlag.shouldBreak = true);\n\n    let branchId: string;\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      branchId = String(stageOutput);\n    } catch (error: any) {\n      context.commit();\n      callExtractor(node, context, getStagePath(node, branchPath, context.stageName), undefined, {\n        type: 'stageExecutionError',\n        message: error.toString(),\n      });\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });\n      context.addError('stageExecutionError', error.toString());\n      this.deps.narrativeGenerator.onError(node.name, error.toString(), error);\n      throw error;\n    }\n\n    context.commit();\n    callExtractor(node, context, getStagePath(node, branchPath, context.stageName), branchId);\n\n    if (breakFlag.shouldBreak) {\n      this.deps.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);\n      return branchId;\n    }\n\n    // Resolve child by matching branch ID against node.children\n    const children = node.children as StageNode<TOut, TScope>[];\n    let chosen = children.find((child) => child.id === branchId);\n\n    // Fall back to default branch\n    if (!chosen) {\n      const defaultChild = children.find((child) => child.id === 'default');\n      if (defaultChild) {\n        chosen = defaultChild;\n      } else {\n        const errorMessage = `Scope-based decider '${node.name}' returned branch ID '${branchId}' which doesn't match any child and no default branch is set`;\n        context.addError('deciderError', errorMessage);\n        throw new Error(errorMessage);\n      }\n    }\n\n    const chosenName = chosen.name;\n    const wasDefault = chosen.id !== branchId;\n    const rationale = context.debug?.logContext?.deciderRationale as string | undefined;\n    let branchReason: string;\n    if (wasDefault) {\n      branchReason = `Returned '${branchId}' (no match), fell back to default → ${chosenName} path.`;\n    } else if (rationale) {\n      branchReason = `Based on: ${rationale} → chose ${chosenName} path.`;\n    } else {\n      branchReason = `Evaluated scope and returned '${branchId}' → chose ${chosenName} path.`;\n    }\n    context.addFlowDebugMessage('branch', branchReason, {\n      targetStage: chosen.name,\n      rationale: rationale || `returned branchId: ${branchId}`,\n    });\n\n    this.deps.narrativeGenerator.onDecision(node.name, chosen.name, rationale, node.description);\n\n    const branchContext = context.createChild(branchPath as string, chosen.id, chosen.name, chosen.id);\n    return executeNode(chosen, branchContext, breakFlag, branchPath);\n  }\n}\n"]}
|
|
78
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DeciderHandler.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/DeciderHandler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DeciderHandler — Single-choice conditional branching.\n *\n * Handles scope-based deciders (stage IS the decider, returns branch ID).\n * Logs flow control decisions and narrative sentences.\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { TraversalContext } from '../narrative/types.js';\nimport type { HandlerDeps, StageFunction } from '../types.js';\n\n/** Callback for running a stage with commit + extractor. Avoids circular dep with traverser. */\nexport type RunStageFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  stageFunc: StageFunction<TOut, TScope>,\n  context: StageContext,\n  breakFn: () => void,\n) => Promise<TOut>;\n\n/** Callback for recursive node execution. Avoids circular dep with traverser. */\nexport type ExecuteNodeFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  breakFlag: { shouldBreak: boolean },\n  branchPath?: string,\n) => Promise<any>;\n\n/** Callback for calling the extractor after stage execution. */\nexport type CallExtractorFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  context: StageContext,\n  stagePath: string,\n  stageOutput?: unknown,\n  errorInfo?: { type: string; message: string },\n) => void;\n\n/** Callback for computing the stage path for extractor. */\nexport type GetStagePathFn<TOut = any, TScope = any> = (\n  node: StageNode<TOut, TScope>,\n  branchPath?: string,\n  contextStageName?: string,\n) => string;\n\nexport class DeciderHandler<TOut = any, TScope = any> {\n  constructor(private readonly deps: HandlerDeps<TOut, TScope>) {}\n\n  /**\n   * Handle a scope-based decider (created via addDeciderFunction).\n   * The stage function IS the decider — its return value is the branch ID.\n   * Execution order: runStage(fn) → commit → resolve child → log → executeNode(child).\n   */\n  async handleScopeBased(\n    node: StageNode<TOut, TScope>,\n    stageFunc: StageFunction<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    runStage: RunStageFn<TOut, TScope>,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n    callExtractor: CallExtractorFn<TOut, TScope>,\n    getStagePath: GetStagePathFn<TOut, TScope>,\n    traversalContext?: TraversalContext,\n  ): Promise<any> {\n    const breakFn = () => (breakFlag.shouldBreak = true);\n\n    let branchId: string;\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      branchId = String(stageOutput);\n    } catch (error: any) {\n      context.commit();\n      callExtractor(node, context, getStagePath(node, branchPath, context.stageName), undefined, {\n        type: 'stageExecutionError',\n        message: error.toString(),\n      });\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });\n      context.addError('stageExecutionError', error.toString());\n      this.deps.narrativeGenerator.onError(node.name, error.toString(), error, traversalContext);\n      throw error;\n    }\n\n    context.commit();\n    callExtractor(node, context, getStagePath(node, branchPath, context.stageName), branchId);\n\n    if (breakFlag.shouldBreak) {\n      this.deps.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);\n      return branchId;\n    }\n\n    // Resolve child by matching branch ID against node.children\n    const children = node.children as StageNode<TOut, TScope>[];\n    let chosen = children.find((child) => child.id === branchId);\n\n    // Fall back to default branch\n    if (!chosen) {\n      const defaultChild = children.find((child) => child.id === 'default');\n      if (defaultChild) {\n        chosen = defaultChild;\n      } else {\n        const errorMessage = `Scope-based decider '${node.name}' returned branch ID '${branchId}' which doesn't match any child and no default branch is set`;\n        context.addError('deciderError', errorMessage);\n        throw new Error(errorMessage);\n      }\n    }\n\n    const chosenName = chosen.name;\n    const wasDefault = chosen.id !== branchId;\n    const rationale = context.debug?.logContext?.deciderRationale as string | undefined;\n    let branchReason: string;\n    if (wasDefault) {\n      branchReason = `Returned '${branchId}' (no match), fell back to default → ${chosenName} path.`;\n    } else if (rationale) {\n      branchReason = `Based on: ${rationale} → chose ${chosenName} path.`;\n    } else {\n      branchReason = `Evaluated scope and returned '${branchId}' → chose ${chosenName} path.`;\n    }\n    context.addFlowDebugMessage('branch', branchReason, {\n      targetStage: chosen.name,\n      rationale: rationale || `returned branchId: ${branchId}`,\n    });\n\n    this.deps.narrativeGenerator.onDecision(node.name, chosen.name, rationale, node.description, traversalContext);\n\n    const branchContext = context.createChild(branchPath as string, chosen.id, chosen.name, chosen.id);\n    return executeNode(chosen, branchContext, breakFlag, branchPath);\n  }\n}\n"]}
|
|
@@ -16,7 +16,7 @@ export class SelectorHandler {
|
|
|
16
16
|
* The stage function IS the selector — its return value contains branch IDs.
|
|
17
17
|
* Execution order: runStage(fn) → commit → resolve children → parallel execute.
|
|
18
18
|
*/
|
|
19
|
-
async handleScopeBased(node, stageFunc, context, breakFlag, branchPath, runStage, executeNode, callExtractor, getStagePath) {
|
|
19
|
+
async handleScopeBased(node, stageFunc, context, breakFlag, branchPath, runStage, executeNode, callExtractor, getStagePath, traversalContext) {
|
|
20
20
|
var _a;
|
|
21
21
|
const breakFn = () => (breakFlag.shouldBreak = true);
|
|
22
22
|
let selectedIds;
|
|
@@ -49,7 +49,7 @@ export class SelectorHandler {
|
|
|
49
49
|
count: 0,
|
|
50
50
|
targetStage: [],
|
|
51
51
|
});
|
|
52
|
-
this.deps.narrativeGenerator.onSelected(node.name, [], ((_a = node.children) !== null && _a !== void 0 ? _a : []).length);
|
|
52
|
+
this.deps.narrativeGenerator.onSelected(node.name, [], ((_a = node.children) !== null && _a !== void 0 ? _a : []).length, traversalContext);
|
|
53
53
|
return {};
|
|
54
54
|
}
|
|
55
55
|
// Resolve children by matching selected IDs against node.children
|
|
@@ -71,13 +71,13 @@ export class SelectorHandler {
|
|
|
71
71
|
const selectedNames = selectedChildren.map((c) => c.name).join(', ');
|
|
72
72
|
context.addFlowDebugMessage('selected', `Running ${selectedNames} (${selectedChildren.length} of ${children.length} matched)`, { count: selectedChildren.length, targetStage: selectedChildren.map((c) => c.name) });
|
|
73
73
|
const selectedDisplayNames = selectedChildren.map((c) => c.name);
|
|
74
|
-
this.deps.narrativeGenerator.onSelected(node.name, selectedDisplayNames, children.length);
|
|
74
|
+
this.deps.narrativeGenerator.onSelected(node.name, selectedDisplayNames, children.length, traversalContext);
|
|
75
75
|
const tempNode = {
|
|
76
76
|
name: 'selector-temp',
|
|
77
77
|
id: 'selector-temp',
|
|
78
78
|
children: selectedChildren,
|
|
79
79
|
};
|
|
80
|
-
return await this.childrenExecutor.executeNodeChildren(tempNode, context, undefined, branchPath);
|
|
80
|
+
return await this.childrenExecutor.executeNodeChildren(tempNode, context, undefined, branchPath, traversalContext);
|
|
81
81
|
}
|
|
82
82
|
}
|
|
83
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SelectorHandler.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/SelectorHandler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SelectorHandler — Multi-choice filtered fan-out.\n *\n * Responsibilities:\n * - Execute scope-based selector nodes (stage → commit → resolve children → parallel execution)\n * - The selector function IS a stage: reads scope, returns string[] of branch IDs\n * - Delegates parallel execution of selected children to ChildrenExecutor\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { HandlerDeps, NodeResultType, StageFunction } from '../types.js';\nimport type { ChildrenExecutor } from './ChildrenExecutor.js';\nimport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './DeciderHandler.js';\n\nexport class SelectorHandler<TOut = any, TScope = any> {\n  constructor(\n    private readonly deps: HandlerDeps<TOut, TScope>,\n    private readonly childrenExecutor: ChildrenExecutor<TOut, TScope>,\n  ) {}\n\n  /**\n   * Handle a scope-based selector node (created via addSelectorFunction).\n   * The stage function IS the selector — its return value contains branch IDs.\n   * Execution order: runStage(fn) → commit → resolve children → parallel execute.\n   */\n  async handleScopeBased(\n    node: StageNode<TOut, TScope>,\n    stageFunc: StageFunction<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    runStage: RunStageFn<TOut, TScope>,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n    callExtractor: CallExtractorFn<TOut, TScope>,\n    getStagePath: GetStagePathFn<TOut, TScope>,\n  ): Promise<Record<string, NodeResultType>> {\n    const breakFn = () => (breakFlag.shouldBreak = true);\n\n    let selectedIds: string[];\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      selectedIds = Array.isArray(stageOutput) ? stageOutput.map(String) : [String(stageOutput)];\n    } catch (error: any) {\n      context.commit();\n      callExtractor(node, context, getStagePath(node, branchPath, context.stageName), undefined, {\n        type: 'stageExecutionError',\n        message: error.toString(),\n      });\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });\n      context.addError('stageExecutionError', error.toString());\n      this.deps.narrativeGenerator.onError(node.name, error.toString(), error);\n      throw error;\n    }\n\n    context.commit();\n    callExtractor(node, context, getStagePath(node, branchPath, context.stageName), selectedIds);\n\n    if (breakFlag.shouldBreak) {\n      this.deps.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);\n      return {};\n    }\n\n    context.addLog('selectedChildIds', selectedIds);\n    context.addLog('selectorPattern', 'scope-based-multi-choice');\n\n    if (selectedIds.length === 0) {\n      context.addLog('skippedAllChildren', true);\n      context.addFlowDebugMessage('selected', 'No children selected — skipping all branches.', {\n        count: 0,\n        targetStage: [],\n      });\n      this.deps.narrativeGenerator.onSelected(node.name, [], (node.children ?? []).length);\n      return {};\n    }\n\n    // Resolve children by matching selected IDs against node.children\n    const children = node.children as StageNode<TOut, TScope>[];\n    const selectedChildren = children.filter((c) => selectedIds.includes(c.id!));\n\n    // Validate all IDs exist (fail fast)\n    if (selectedChildren.length !== selectedIds.length) {\n      const childIds = children.map((c) => c.id);\n      const missing = selectedIds.filter((id) => !childIds.includes(id));\n      const errorMessage = `Scope-based selector '${node.name}' returned unknown child IDs: ${missing.join(\n        ', ',\n      )}. Available: ${childIds.join(', ')}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}):`, { error: errorMessage });\n      context.addError('selectorError', errorMessage);\n      throw new Error(errorMessage);\n    }\n\n    const skippedIds = children.filter((c) => !selectedIds.includes(c.id!)).map((c) => c.id);\n    if (skippedIds.length > 0) {\n      context.addLog('skippedChildIds', skippedIds);\n    }\n\n    const selectedNames = selectedChildren.map((c) => c.name).join(', ');\n    context.addFlowDebugMessage(\n      'selected',\n      `Running ${selectedNames} (${selectedChildren.length} of ${children.length} matched)`,\n      { count: selectedChildren.length, targetStage: selectedChildren.map((c) => c.name) },\n    );\n\n    const selectedDisplayNames = selectedChildren.map((c) => c.name);\n    this.deps.narrativeGenerator.onSelected(node.name, selectedDisplayNames, children.length);\n\n    const tempNode: StageNode<TOut, TScope> = {\n      name: 'selector-temp',\n      id: 'selector-temp',\n      children: selectedChildren,\n    };\n    return await this.childrenExecutor.executeNodeChildren(tempNode, context, undefined, branchPath);\n  }\n}\n"]}
|
|
83
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SelectorHandler.js","sourceRoot":"","sources":["../../../../../src/lib/engine/handlers/SelectorHandler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SelectorHandler — Multi-choice filtered fan-out.\n *\n * Responsibilities:\n * - Execute scope-based selector nodes (stage → commit → resolve children → parallel execution)\n * - The selector function IS a stage: reads scope, returns string[] of branch IDs\n * - Delegates parallel execution of selected children to ChildrenExecutor\n */\n\nimport type { StageContext } from '../../memory/StageContext.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { TraversalContext } from '../narrative/types.js';\nimport type { HandlerDeps, NodeResultType, StageFunction } from '../types.js';\nimport type { ChildrenExecutor } from './ChildrenExecutor.js';\nimport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './DeciderHandler.js';\n\nexport class SelectorHandler<TOut = any, TScope = any> {\n  constructor(\n    private readonly deps: HandlerDeps<TOut, TScope>,\n    private readonly childrenExecutor: ChildrenExecutor<TOut, TScope>,\n  ) {}\n\n  /**\n   * Handle a scope-based selector node (created via addSelectorFunction).\n   * The stage function IS the selector — its return value contains branch IDs.\n   * Execution order: runStage(fn) → commit → resolve children → parallel execute.\n   */\n  async handleScopeBased(\n    node: StageNode<TOut, TScope>,\n    stageFunc: StageFunction<TOut, TScope>,\n    context: StageContext,\n    breakFlag: { shouldBreak: boolean },\n    branchPath: string | undefined,\n    runStage: RunStageFn<TOut, TScope>,\n    executeNode: ExecuteNodeFn<TOut, TScope>,\n    callExtractor: CallExtractorFn<TOut, TScope>,\n    getStagePath: GetStagePathFn<TOut, TScope>,\n    traversalContext?: TraversalContext,\n  ): Promise<Record<string, NodeResultType>> {\n    const breakFn = () => (breakFlag.shouldBreak = true);\n\n    let selectedIds: string[];\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      selectedIds = Array.isArray(stageOutput) ? stageOutput.map(String) : [String(stageOutput)];\n    } catch (error: any) {\n      context.commit();\n      callExtractor(node, context, getStagePath(node, branchPath, context.stageName), undefined, {\n        type: 'stageExecutionError',\n        message: error.toString(),\n      });\n      this.deps.logger.error(`Error in pipeline (${branchPath}) stage [${node.name}]:`, { error });\n      context.addError('stageExecutionError', error.toString());\n      this.deps.narrativeGenerator.onError(node.name, error.toString(), error);\n      throw error;\n    }\n\n    context.commit();\n    callExtractor(node, context, getStagePath(node, branchPath, context.stageName), selectedIds);\n\n    if (breakFlag.shouldBreak) {\n      this.deps.logger.info(`Execution stopped in pipeline (${branchPath}) after ${node.name} due to break condition.`);\n      return {};\n    }\n\n    context.addLog('selectedChildIds', selectedIds);\n    context.addLog('selectorPattern', 'scope-based-multi-choice');\n\n    if (selectedIds.length === 0) {\n      context.addLog('skippedAllChildren', true);\n      context.addFlowDebugMessage('selected', 'No children selected — skipping all branches.', {\n        count: 0,\n        targetStage: [],\n      });\n      this.deps.narrativeGenerator.onSelected(node.name, [], (node.children ?? []).length, traversalContext);\n      return {};\n    }\n\n    // Resolve children by matching selected IDs against node.children\n    const children = node.children as StageNode<TOut, TScope>[];\n    const selectedChildren = children.filter((c) => selectedIds.includes(c.id!));\n\n    // Validate all IDs exist (fail fast)\n    if (selectedChildren.length !== selectedIds.length) {\n      const childIds = children.map((c) => c.id);\n      const missing = selectedIds.filter((id) => !childIds.includes(id));\n      const errorMessage = `Scope-based selector '${node.name}' returned unknown child IDs: ${missing.join(\n        ', ',\n      )}. Available: ${childIds.join(', ')}`;\n      this.deps.logger.error(`Error in pipeline (${branchPath}):`, { error: errorMessage });\n      context.addError('selectorError', errorMessage);\n      throw new Error(errorMessage);\n    }\n\n    const skippedIds = children.filter((c) => !selectedIds.includes(c.id!)).map((c) => c.id);\n    if (skippedIds.length > 0) {\n      context.addLog('skippedChildIds', skippedIds);\n    }\n\n    const selectedNames = selectedChildren.map((c) => c.name).join(', ');\n    context.addFlowDebugMessage(\n      'selected',\n      `Running ${selectedNames} (${selectedChildren.length} of ${children.length} matched)`,\n      { count: selectedChildren.length, targetStage: selectedChildren.map((c) => c.name) },\n    );\n\n    const selectedDisplayNames = selectedChildren.map((c) => c.name);\n    this.deps.narrativeGenerator.onSelected(node.name, selectedDisplayNames, children.length, traversalContext);\n\n    const tempNode: StageNode<TOut, TScope> = {\n      name: 'selector-temp',\n      id: 'selector-temp',\n      children: selectedChildren,\n    };\n    return await this.childrenExecutor.executeNodeChildren(tempNode, context, undefined, branchPath, traversalContext);\n  }\n}\n"]}
|