footprintjs 4.12.1 → 4.12.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/DeciderHandler.js +14 -3
- package/dist/esm/lib/engine/handlers/SelectorHandler.js +13 -3
- package/dist/esm/lib/pause/types.js +20 -1
- package/dist/esm/lib/runner/FlowChartExecutor.js +15 -3
- package/dist/lib/engine/handlers/DeciderHandler.js +14 -3
- package/dist/lib/engine/handlers/SelectorHandler.js +13 -3
- package/dist/lib/pause/types.js +20 -1
- package/dist/lib/runner/FlowChartExecutor.js +15 -3
- package/dist/types/lib/pause/types.d.ts +27 -0
- package/package.json +1 -1
|
@@ -16,7 +16,7 @@ export class DeciderHandler {
|
|
|
16
16
|
* Execution order: runStage(fn) → commit → resolve child → log → executeNode(child).
|
|
17
17
|
*/
|
|
18
18
|
async handleScopeBased(node, stageFunc, context, breakFlag, branchPath, runStage, executeNode, callExtractor, getStagePath, traversalContext) {
|
|
19
|
-
var _a, _b, _c;
|
|
19
|
+
var _a, _b, _c, _d;
|
|
20
20
|
const breakFn = () => (breakFlag.shouldBreak = true);
|
|
21
21
|
let branchId;
|
|
22
22
|
let decisionEvidence;
|
|
@@ -88,7 +88,18 @@ export class DeciderHandler {
|
|
|
88
88
|
});
|
|
89
89
|
this.deps.narrativeGenerator.onDecision(node.name, chosen.name, rationale, node.description, traversalContext, decisionEvidence);
|
|
90
90
|
const branchContext = context.createChild(branchPath, chosen.id, chosen.name, chosen.id);
|
|
91
|
-
|
|
91
|
+
try {
|
|
92
|
+
return await executeNode(chosen, branchContext, breakFlag, branchPath);
|
|
93
|
+
}
|
|
94
|
+
catch (error) {
|
|
95
|
+
// Stamp invoker context on PauseSignal during bubble-up.
|
|
96
|
+
// The decider (node) is the invoker; its .next is the continuation target.
|
|
97
|
+
if (isPauseSignal(error)) {
|
|
98
|
+
error.setInvoker(node.id, (_d = node.next) === null || _d === void 0 ? void 0 : _d.id);
|
|
99
|
+
throw error;
|
|
100
|
+
}
|
|
101
|
+
throw error;
|
|
102
|
+
}
|
|
92
103
|
}
|
|
93
104
|
}
|
|
94
|
-
//# 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 { DecisionEvidence } from '../../decide/types.js';\nimport { DECISION_RESULT } from '../../decide/types.js';\nimport type { StageContext } from '../../memory/StageContext.js';\nimport { isPauseSignal } from '../../pause/types.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { TraversalContext } from '../narrative/types.js';\nimport type { HandlerDeps, StageFunction } from '../types.js';\nimport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './types.js';\n\nexport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn };\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    let decisionEvidence: DecisionEvidence | undefined;\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      // Detect DecisionResult from decide() helper via Symbol brand\n      if (stageOutput && typeof stageOutput === 'object' && Reflect.has(stageOutput as object, DECISION_RESULT)) {\n        branchId = (stageOutput as any).branch;\n        decisionEvidence = (stageOutput as any).evidence;\n      } else {\n        branchId = String(stageOutput);\n      }\n    } catch (error: any) {\n      // PauseSignal is expected control flow — commit and re-throw without error logging.\n      if (isPauseSignal(error)) {\n        context.commit();\n        throw error;\n      }\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    // Match branchId first (original unprefixed ID), fall back to id for backward compat.\n    const children = node.children as StageNode<TOut, TScope>[];\n    let chosen = children.find((child) => (child.branchId ?? child.id) === branchId);\n\n    // Fall back to default branch\n    if (!chosen) {\n      const defaultChild = children.find((child) => (child.branchId ?? 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.branchId ?? 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(\n      node.name,\n      chosen.name,\n      rationale,\n      node.description,\n      traversalContext,\n      decisionEvidence,\n    );\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"]}
|
|
105
|
+
//# 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 { DecisionEvidence } from '../../decide/types.js';\nimport { DECISION_RESULT } from '../../decide/types.js';\nimport type { StageContext } from '../../memory/StageContext.js';\nimport { isPauseSignal } from '../../pause/types.js';\nimport type { StageNode } from '../graph/StageNode.js';\nimport type { TraversalContext } from '../narrative/types.js';\nimport type { HandlerDeps, StageFunction } from '../types.js';\nimport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './types.js';\n\nexport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn };\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    let decisionEvidence: DecisionEvidence | undefined;\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      // Detect DecisionResult from decide() helper via Symbol brand\n      if (stageOutput && typeof stageOutput === 'object' && Reflect.has(stageOutput as object, DECISION_RESULT)) {\n        branchId = (stageOutput as any).branch;\n        decisionEvidence = (stageOutput as any).evidence;\n      } else {\n        branchId = String(stageOutput);\n      }\n    } catch (error: any) {\n      // PauseSignal is expected control flow — commit and re-throw without error logging.\n      if (isPauseSignal(error)) {\n        context.commit();\n        throw error;\n      }\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    // Match branchId first (original unprefixed ID), fall back to id for backward compat.\n    const children = node.children as StageNode<TOut, TScope>[];\n    let chosen = children.find((child) => (child.branchId ?? child.id) === branchId);\n\n    // Fall back to default branch\n    if (!chosen) {\n      const defaultChild = children.find((child) => (child.branchId ?? 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.branchId ?? 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(\n      node.name,\n      chosen.name,\n      rationale,\n      node.description,\n      traversalContext,\n      decisionEvidence,\n    );\n\n    const branchContext = context.createChild(branchPath as string, chosen.id, chosen.name, chosen.id);\n\n    try {\n      return await executeNode(chosen, branchContext, breakFlag, branchPath);\n    } catch (error: unknown) {\n      // Stamp invoker context on PauseSignal during bubble-up.\n      // The decider (node) is the invoker; its .next is the continuation target.\n      if (isPauseSignal(error)) {\n        error.setInvoker(node.id!, node.next?.id);\n        throw error;\n      }\n      throw error;\n    }\n  }\n}\n"]}
|
|
@@ -19,7 +19,7 @@ export class SelectorHandler {
|
|
|
19
19
|
* Execution order: runStage(fn) → commit → resolve children → parallel execute.
|
|
20
20
|
*/
|
|
21
21
|
async handleScopeBased(node, stageFunc, context, breakFlag, branchPath, runStage, executeNode, callExtractor, getStagePath, traversalContext) {
|
|
22
|
-
var _a;
|
|
22
|
+
var _a, _b;
|
|
23
23
|
const breakFn = () => (breakFlag.shouldBreak = true);
|
|
24
24
|
let selectedIds;
|
|
25
25
|
let selectionEvidence;
|
|
@@ -98,7 +98,17 @@ export class SelectorHandler {
|
|
|
98
98
|
id: 'selector-temp',
|
|
99
99
|
children: selectedChildren,
|
|
100
100
|
};
|
|
101
|
-
|
|
101
|
+
try {
|
|
102
|
+
return await this.childrenExecutor.executeNodeChildren(tempNode, context, undefined, branchPath, traversalContext);
|
|
103
|
+
}
|
|
104
|
+
catch (error) {
|
|
105
|
+
// Stamp invoker context on PauseSignal during bubble-up.
|
|
106
|
+
if (isPauseSignal(error)) {
|
|
107
|
+
error.setInvoker(node.id, (_b = node.next) === null || _b === void 0 ? void 0 : _b.id);
|
|
108
|
+
throw error;
|
|
109
|
+
}
|
|
110
|
+
throw error;
|
|
111
|
+
}
|
|
102
112
|
}
|
|
103
113
|
}
|
|
104
|
-
//# 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 { SelectionEvidence } from '../../decide/types.js';\nimport { DECISION_RESULT } from '../../decide/types.js';\nimport type { StageContext } from '../../memory/StageContext.js';\nimport { isPauseSignal } from '../../pause/types.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 './types.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    let selectionEvidence: SelectionEvidence | undefined;\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      // Detect SelectionResult from select() helper via Symbol brand\n      if (\n        stageOutput &&\n        typeof stageOutput === 'object' &&\n        Reflect.has(stageOutput as object, DECISION_RESULT) &&\n        Array.isArray((stageOutput as any).branches)\n      ) {\n        selectedIds = (stageOutput as any).branches;\n        selectionEvidence = (stageOutput as any).evidence;\n      } else {\n        selectedIds = Array.isArray(stageOutput) ? stageOutput.map(String) : [String(stageOutput)];\n      }\n    } catch (error: any) {\n      // PauseSignal is expected control flow — commit and re-throw without error logging.\n      if (isPauseSignal(error)) {\n        context.commit();\n        throw error;\n      }\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), 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    // Match branchId first (original unprefixed ID), fall back to id for backward compat.\n    const children = node.children as StageNode<TOut, TScope>[];\n    const selectedChildren = children.filter((c) => selectedIds.includes(c.branchId ?? c.id!));\n\n    // Validate all IDs exist (fail fast)\n    if (selectedChildren.length !== selectedIds.length) {\n      const childIds = children.map((c) => c.branchId ?? 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\n      .filter((c) => !selectedIds.includes(c.branchId ?? c.id!))\n      .map((c) => c.branchId ?? 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(\n      node.name,\n      selectedDisplayNames,\n      children.length,\n      traversalContext,\n      selectionEvidence,\n    );\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"]}
|
|
114
|
+
//# 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 { SelectionEvidence } from '../../decide/types.js';\nimport { DECISION_RESULT } from '../../decide/types.js';\nimport type { StageContext } from '../../memory/StageContext.js';\nimport { isPauseSignal } from '../../pause/types.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 './types.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    let selectionEvidence: SelectionEvidence | undefined;\n    try {\n      const stageOutput = await runStage(node, stageFunc, context, breakFn);\n      // Detect SelectionResult from select() helper via Symbol brand\n      if (\n        stageOutput &&\n        typeof stageOutput === 'object' &&\n        Reflect.has(stageOutput as object, DECISION_RESULT) &&\n        Array.isArray((stageOutput as any).branches)\n      ) {\n        selectedIds = (stageOutput as any).branches;\n        selectionEvidence = (stageOutput as any).evidence;\n      } else {\n        selectedIds = Array.isArray(stageOutput) ? stageOutput.map(String) : [String(stageOutput)];\n      }\n    } catch (error: any) {\n      // PauseSignal is expected control flow — commit and re-throw without error logging.\n      if (isPauseSignal(error)) {\n        context.commit();\n        throw error;\n      }\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), 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    // Match branchId first (original unprefixed ID), fall back to id for backward compat.\n    const children = node.children as StageNode<TOut, TScope>[];\n    const selectedChildren = children.filter((c) => selectedIds.includes(c.branchId ?? c.id!));\n\n    // Validate all IDs exist (fail fast)\n    if (selectedChildren.length !== selectedIds.length) {\n      const childIds = children.map((c) => c.branchId ?? 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\n      .filter((c) => !selectedIds.includes(c.branchId ?? c.id!))\n      .map((c) => c.branchId ?? 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(\n      node.name,\n      selectedDisplayNames,\n      children.length,\n      traversalContext,\n      selectionEvidence,\n    );\n\n    const tempNode: StageNode<TOut, TScope> = {\n      name: 'selector-temp',\n      id: 'selector-temp',\n      children: selectedChildren,\n    };\n    try {\n      return await this.childrenExecutor.executeNodeChildren(\n        tempNode,\n        context,\n        undefined,\n        branchPath,\n        traversalContext,\n      );\n    } catch (error: unknown) {\n      // Stamp invoker context on PauseSignal during bubble-up.\n      if (isPauseSignal(error)) {\n        error.setInvoker(node.id!, node.next?.id);\n        throw error;\n      }\n      throw error;\n    }\n  }\n}\n"]}
|
|
@@ -41,6 +41,25 @@ export class PauseSignal extends Error {
|
|
|
41
41
|
prependSubflow(subflowId) {
|
|
42
42
|
this._subflowPath.unshift(subflowId);
|
|
43
43
|
}
|
|
44
|
+
/** The stage that invoked the paused child (decider, selector, fork). */
|
|
45
|
+
get invokerStageId() {
|
|
46
|
+
return this._invokerStageId;
|
|
47
|
+
}
|
|
48
|
+
/** Where execution should continue after resume (invoker's next node). */
|
|
49
|
+
get continuationStageId() {
|
|
50
|
+
return this._continuationStageId;
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* Stamp the invoker context during bubble-up.
|
|
54
|
+
* Called by decider/selector/fork handlers when catching a child's PauseSignal.
|
|
55
|
+
* First invoker wins (innermost) — subsequent calls are no-ops.
|
|
56
|
+
*/
|
|
57
|
+
setInvoker(invokerStageId, continuationStageId) {
|
|
58
|
+
if (!this._invokerStageId) {
|
|
59
|
+
this._invokerStageId = invokerStageId;
|
|
60
|
+
this._continuationStageId = continuationStageId;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
44
63
|
}
|
|
45
64
|
// ── Type guard ──────────────────────────────────────────────
|
|
46
65
|
/** Check if a value is a PauseResult (stage wants to pause). */
|
|
@@ -55,4 +74,4 @@ export function isPauseSignal(error) {
|
|
|
55
74
|
Object.prototype.hasOwnProperty.call(error, 'pauseData') &&
|
|
56
75
|
Object.prototype.hasOwnProperty.call(error, 'stageId')));
|
|
57
76
|
}
|
|
58
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../src/lib/pause/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;GAiBG;AAEH,+DAA+D;AAE/D;;;;;;GAMG;AACH,MAAM,OAAO,WAAY,SAAQ,KAAK;IAQpC,YAAY,IAAa,EAAE,OAAe;QACxC,KAAK,CAAC,kBAAkB,CAAC,CAAC;QAC1B,IAAI,CAAC,IAAI,GAAG,aAAa,CAAC;QAC1B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,uFAAuF;QACvF,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;IAClB,CAAC;IAED,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED,kEAAkE;IAClE,cAAc,CAAC,SAAiB;QAC9B,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IACvC,CAAC;CACF;AAoJD,+DAA+D;AAE/D,gEAAgE;AAChE,MAAM,UAAU,aAAa,CAAC,KAAc;IAC1C,OAAO,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAK,KAAqB,CAAC,KAAK,KAAK,IAAI,CAAC;AAC9F,CAAC;AAED,wGAAwG;AACxG,MAAM,UAAU,aAAa,CAAC,KAAc;IAC1C,OAAO,CACL,KAAK,YAAY,WAAW;QAC5B,CAAC,KAAK,YAAY,KAAK;YACrB,KAAK,CAAC,IAAI,KAAK,aAAa;YAC5B,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,EAAE,WAAW,CAAC;YACxD,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC,CAC1D,CAAC;AACJ,CAAC","sourcesContent":["/**\n * Pause/Resume — serializable checkpoint for long-running or human-in-the-loop flows.\n *\n * A stage signals pause by calling `scope.$pause(data)` which throws a PauseSignal.\n * The signal bubbles up through SubflowExecutor → FlowchartTraverser → FlowChartExecutor,\n * each level adding its subflow ID to the path.\n *\n * The checkpoint captures:\n *   - pausedPath: full path to the paused stage (e.g., ['sf-payment', 'approve'])\n *   - sharedState: scope at the pause point\n *   - executionTree: completed stages for BTS/narrative\n *   - pauseData: question, reason, or metadata from $pause()\n *\n * Resume rebuilds the flowchart, restores scope, navigates to the paused stage,\n * injects resumeInput, and continues traversal.\n *\n * Supported topologies: linear, subflow, lazy subflow, loop, nested subflow in loop.\n */\n\n// ── PauseSignal ─────────────────────────────────────────────\n\n/**\n * Thrown by `scope.$pause()` to signal that execution should stop\n * and create a serializable checkpoint.\n *\n * Bubbles up through SubflowExecutor (which prepends subflow ID to path)\n * and is caught by FlowchartTraverser/FlowChartExecutor.\n */\nexport class PauseSignal extends Error {\n  /** Data from $pause() — question, reason, metadata. */\n  readonly pauseData: unknown;\n  /** ID of the stage that called $pause(). */\n  readonly stageId: string;\n  /** Path through subflows to the paused stage. Built during bubble-up. */\n  private _subflowPath: string[];\n\n  constructor(data: unknown, stageId: string) {\n    super('Execution paused');\n    this.name = 'PauseSignal';\n    this.pauseData = data;\n    this.stageId = stageId;\n    this._subflowPath = [];\n    // PauseSignal is control flow, not a real error — stack trace has no diagnostic value.\n    this.stack = '';\n  }\n\n  get subflowPath(): readonly string[] {\n    return this._subflowPath;\n  }\n\n  /** Prepend a subflow ID to the path (called during bubble-up). */\n  prependSubflow(subflowId: string): void {\n    this._subflowPath.unshift(subflowId);\n  }\n}\n\n// ── PauseResult ─────────────────────────────────────────────\n\n/**\n * Returned by a pausable stage's execute/resume function to signal pause.\n *\n * @example\n * ```typescript\n * execute: async (scope) => {\n *   scope.orderId = '123';\n *   return { pause: true, data: { question: 'Approve order 123?' } };\n * }\n * ```\n */\nexport interface PauseResult {\n  readonly pause: true;\n  /** Data to include in the checkpoint — question, reason, metadata. */\n  readonly data?: unknown;\n}\n\n// ── FlowchartCheckpoint ─────────────────────────────────────\n\n/**\n * Serializable checkpoint — everything needed to resume a paused flowchart.\n *\n * JSON-safe: no functions, no class instances, no SDK clients.\n * Store anywhere: Redis, Postgres, localStorage, a file.\n *\n * @example\n * ```typescript\n * // Save\n * const checkpoint = executor.getCheckpoint(); // after pause\n * await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n *\n * // Resume (hours later, possibly different server)\n * const checkpoint = JSON.parse(await redis.get(`session:${id}`));\n * const executor = new FlowChartExecutor(chart);\n * await executor.resume(checkpoint, { approved: true });\n * ```\n */\n/**\n * Serializable checkpoint — everything needed to resume a paused flowchart.\n *\n * The execution tree IS the traversed path. The leaf node with status 'paused'\n * IS the cursor. No separate path array needed — the tree structure captures\n * the full nesting (including subflows).\n *\n * JSON-safe: no functions, no class instances, no SDK clients.\n * Store anywhere: Redis, Postgres, localStorage, a file.\n *\n * @example\n * ```typescript\n * const checkpoint = executor.getCheckpoint(); // after pause\n * await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n *\n * // Resume (hours later, possibly different server)\n * const checkpoint = JSON.parse(await redis.get(`session:${id}`));\n * const executor = new FlowChartExecutor(chart);\n * await executor.resume(checkpoint, { approved: true });\n * ```\n */\nexport interface FlowchartCheckpoint {\n  /** Scope state at the pause point — all shared memory key/values. */\n  readonly sharedState: Record<string, unknown>;\n\n  /** Execution tree — the traversed path. The leaf with status 'paused' is the cursor.\n   *  Contains subflow nesting. Used for BTS visualization and to find the resume point. */\n  readonly executionTree: unknown;\n\n  /** ID of the stage that paused. Used by resume() to find the node in the graph. */\n  readonly pausedStageId: string;\n\n  /** Path through subflows to the paused stage (e.g., ['sf-payment', 'sf-validation']).\n   *  Empty array when paused at the top level. */\n  readonly subflowPath: readonly string[];\n\n  /** Data from $pause() — question, reason, metadata. */\n  readonly pauseData?: unknown;\n\n  /** Subflow results collected before the pause. */\n  readonly subflowResults?: Record<string, unknown>;\n\n  /** Timestamp of when the pause occurred. */\n  readonly pausedAt: number;\n}\n\n// ── PausableHandler ─────────────────────────────────────────\n\n/**\n * Handler for a pausable stage — has two phases: execute and resume.\n *\n * `execute` runs the first time. It can return `{ pause: true }` to pause.\n * `resume` runs when the flowchart is resumed. It receives the resume input.\n *\n * Both phases receive the same scope. After execute pauses, the scope state\n * is preserved in the checkpoint. On resume, the scope is restored before\n * calling resume.\n *\n * @example\n * ```typescript\n * .addPausableFunction('ApproveOrder', {\n *   execute: async (scope) => {\n *     scope.orderId = '123';\n *     scope.amount = 299;\n *     return { pause: true, data: { question: `Approve $${scope.amount} refund?` } };\n *   },\n *   resume: async (scope, input) => {\n *     scope.approved = input.approved;\n *     scope.approver = input.approver;\n *   },\n * }, 'approve-order', 'Manager approval gate')\n *\n * // Later — resume with human's answer\n * await executor.resume(checkpoint, { approved: true, approver: 'Jane' });\n * ```\n */\nexport interface PausableHandler<TScope = any, TInput = unknown, TPauseData = unknown> {\n  /**\n   * First-run phase. Return data to pause, or void/undefined to continue normally.\n   *\n   * Any non-void return value becomes the `pauseData` in the checkpoint.\n   * The consumer defines the `TPauseData` type — the FE uses it to render\n   * the right UI (form fields, approval buttons, etc.).\n   *\n   * @example\n   * ```typescript\n   * // TPauseData = { question: string; riskLevel: string }\n   * const handler: PausableHandler<MyState, { approved: boolean }, { question: string; riskLevel: string }> = {\n   *   execute: async (scope) => {\n   *     return { question: `Approve order ${scope.orderId}?`, riskLevel: 'high' };\n   *   },\n   *   resume: async (scope, input) => {\n   *     scope.approved = input.approved;\n   *   },\n   * };\n   * ```\n   */\n  execute: (scope: TScope) => Promise<TPauseData | void> | TPauseData | void;\n  /**\n   * Resume phase. Called with the resume input when execution continues.\n   *\n   * The scope is restored from the checkpoint's `sharedState`. Writes during\n   * `resume` are committed and visible to subsequent stages.\n   */\n  resume: (scope: TScope, input: TInput) => Promise<void> | void;\n}\n\n// ── Type guard ──────────────────────────────────────────────\n\n/** Check if a value is a PauseResult (stage wants to pause). */\nexport function isPauseResult(value: unknown): value is PauseResult {\n  return typeof value === 'object' && value !== null && (value as PauseResult).pause === true;\n}\n\n/** Check if an error is a PauseSignal. Uses instanceof + name brand fallback for cross-realm safety. */\nexport function isPauseSignal(error: unknown): error is PauseSignal {\n  return (\n    error instanceof PauseSignal ||\n    (error instanceof Error &&\n      error.name === 'PauseSignal' &&\n      Object.prototype.hasOwnProperty.call(error, 'pauseData') &&\n      Object.prototype.hasOwnProperty.call(error, 'stageId'))\n  );\n}\n"]}
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../src/lib/pause/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;GAiBG;AAEH,+DAA+D;AAE/D;;;;;;GAMG;AACH,MAAM,OAAO,WAAY,SAAQ,KAAK;IAsBpC,YAAY,IAAa,EAAE,OAAe;QACxC,KAAK,CAAC,kBAAkB,CAAC,CAAC;QAC1B,IAAI,CAAC,IAAI,GAAG,aAAa,CAAC;QAC1B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,uFAAuF;QACvF,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;IAClB,CAAC;IAED,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED,kEAAkE;IAClE,cAAc,CAAC,SAAiB;QAC9B,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IACvC,CAAC;IAED,yEAAyE;IACzE,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED,0EAA0E;IAC1E,IAAI,mBAAmB;QACrB,OAAO,IAAI,CAAC,oBAAoB,CAAC;IACnC,CAAC;IAED;;;;OAIG;IACH,UAAU,CAAC,cAAsB,EAAE,mBAA4B;QAC7D,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC;YAC1B,IAAI,CAAC,eAAe,GAAG,cAAc,CAAC;YACtC,IAAI,CAAC,oBAAoB,GAAG,mBAAmB,CAAC;QAClD,CAAC;IACH,CAAC;CACF;AA0JD,+DAA+D;AAE/D,gEAAgE;AAChE,MAAM,UAAU,aAAa,CAAC,KAAc;IAC1C,OAAO,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAK,KAAqB,CAAC,KAAK,KAAK,IAAI,CAAC;AAC9F,CAAC;AAED,wGAAwG;AACxG,MAAM,UAAU,aAAa,CAAC,KAAc;IAC1C,OAAO,CACL,KAAK,YAAY,WAAW;QAC5B,CAAC,KAAK,YAAY,KAAK;YACrB,KAAK,CAAC,IAAI,KAAK,aAAa;YAC5B,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,EAAE,WAAW,CAAC;YACxD,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC,CAC1D,CAAC;AACJ,CAAC","sourcesContent":["/**\n * Pause/Resume — serializable checkpoint for long-running or human-in-the-loop flows.\n *\n * A stage signals pause by calling `scope.$pause(data)` which throws a PauseSignal.\n * The signal bubbles up through SubflowExecutor → FlowchartTraverser → FlowChartExecutor,\n * each level adding its subflow ID to the path.\n *\n * The checkpoint captures:\n *   - pausedPath: full path to the paused stage (e.g., ['sf-payment', 'approve'])\n *   - sharedState: scope at the pause point\n *   - executionTree: completed stages for BTS/narrative\n *   - pauseData: question, reason, or metadata from $pause()\n *\n * Resume rebuilds the flowchart, restores scope, navigates to the paused stage,\n * injects resumeInput, and continues traversal.\n *\n * Supported topologies: linear, subflow, lazy subflow, loop, nested subflow in loop.\n */\n\n// ── PauseSignal ─────────────────────────────────────────────\n\n/**\n * Thrown by `scope.$pause()` to signal that execution should stop\n * and create a serializable checkpoint.\n *\n * Bubbles up through SubflowExecutor (which prepends subflow ID to path)\n * and is caught by FlowchartTraverser/FlowChartExecutor.\n */\nexport class PauseSignal extends Error {\n  /** Data from $pause() — question, reason, metadata. */\n  readonly pauseData: unknown;\n  /** ID of the stage that called $pause(). */\n  readonly stageId: string;\n  /** Path through subflows to the paused stage. Built during bubble-up. */\n  private _subflowPath: string[];\n\n  /**\n   * Invoker context — enriched during bubble-up when a PauseSignal passes\n   * through a decider, selector, or fork handler.\n   *\n   * The invoker is the stage that called executeNode() on the paused child.\n   * Captured during traversal (not reconstructed from the tree).\n   *\n   * `continuationStageId` is the invoker's `.next` node — where execution\n   * should continue after resume. Without this, branch children have no\n   * `.next` pointer and resume would terminate early.\n   */\n  private _invokerStageId?: string;\n  private _continuationStageId?: string;\n\n  constructor(data: unknown, stageId: string) {\n    super('Execution paused');\n    this.name = 'PauseSignal';\n    this.pauseData = data;\n    this.stageId = stageId;\n    this._subflowPath = [];\n    // PauseSignal is control flow, not a real error — stack trace has no diagnostic value.\n    this.stack = '';\n  }\n\n  get subflowPath(): readonly string[] {\n    return this._subflowPath;\n  }\n\n  /** Prepend a subflow ID to the path (called during bubble-up). */\n  prependSubflow(subflowId: string): void {\n    this._subflowPath.unshift(subflowId);\n  }\n\n  /** The stage that invoked the paused child (decider, selector, fork). */\n  get invokerStageId(): string | undefined {\n    return this._invokerStageId;\n  }\n\n  /** Where execution should continue after resume (invoker's next node). */\n  get continuationStageId(): string | undefined {\n    return this._continuationStageId;\n  }\n\n  /**\n   * Stamp the invoker context during bubble-up.\n   * Called by decider/selector/fork handlers when catching a child's PauseSignal.\n   * First invoker wins (innermost) — subsequent calls are no-ops.\n   */\n  setInvoker(invokerStageId: string, continuationStageId?: string): void {\n    if (!this._invokerStageId) {\n      this._invokerStageId = invokerStageId;\n      this._continuationStageId = continuationStageId;\n    }\n  }\n}\n\n// ── PauseResult ─────────────────────────────────────────────\n\n/**\n * Returned by a pausable stage's execute/resume function to signal pause.\n *\n * @example\n * ```typescript\n * execute: async (scope) => {\n *   scope.orderId = '123';\n *   return { pause: true, data: { question: 'Approve order 123?' } };\n * }\n * ```\n */\nexport interface PauseResult {\n  readonly pause: true;\n  /** Data to include in the checkpoint — question, reason, metadata. */\n  readonly data?: unknown;\n}\n\n// ── FlowchartCheckpoint ─────────────────────────────────────\n\n/**\n * Serializable checkpoint — everything needed to resume a paused flowchart.\n *\n * JSON-safe: no functions, no class instances, no SDK clients.\n * Store anywhere: Redis, Postgres, localStorage, a file.\n *\n * @example\n * ```typescript\n * // Save\n * const checkpoint = executor.getCheckpoint(); // after pause\n * await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n *\n * // Resume (hours later, possibly different server)\n * const checkpoint = JSON.parse(await redis.get(`session:${id}`));\n * const executor = new FlowChartExecutor(chart);\n * await executor.resume(checkpoint, { approved: true });\n * ```\n */\n/**\n * Serializable checkpoint — everything needed to resume a paused flowchart.\n *\n * The execution tree IS the traversed path. The leaf node with status 'paused'\n * IS the cursor. No separate path array needed — the tree structure captures\n * the full nesting (including subflows).\n *\n * JSON-safe: no functions, no class instances, no SDK clients.\n * Store anywhere: Redis, Postgres, localStorage, a file.\n *\n * @example\n * ```typescript\n * const checkpoint = executor.getCheckpoint(); // after pause\n * await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n *\n * // Resume (hours later, possibly different server)\n * const checkpoint = JSON.parse(await redis.get(`session:${id}`));\n * const executor = new FlowChartExecutor(chart);\n * await executor.resume(checkpoint, { approved: true });\n * ```\n */\nexport interface FlowchartCheckpoint {\n  /** Scope state at the pause point — all shared memory key/values. */\n  readonly sharedState: Record<string, unknown>;\n\n  /** Execution tree — the traversed path. The leaf with status 'paused' is the cursor.\n   *  Contains subflow nesting. Used for BTS visualization and to find the resume point. */\n  readonly executionTree: unknown;\n\n  /** ID of the stage that paused. Used by resume() to find the node in the graph. */\n  readonly pausedStageId: string;\n\n  /** Path through subflows to the paused stage (e.g., ['sf-payment', 'sf-validation']).\n   *  Empty array when paused at the top level. */\n  readonly subflowPath: readonly string[];\n\n  /** Data from $pause() — question, reason, metadata. */\n  readonly pauseData?: unknown;\n\n  /** Subflow results collected before the pause. */\n  readonly subflowResults?: Record<string, unknown>;\n\n  /** Stage that invoked the paused child (decider, selector, fork). Absent for linear pauses. */\n  readonly invokerStageId?: string;\n\n  /** Where to continue after resume — the invoker's next node ID. Absent for linear pauses. */\n  readonly continuationStageId?: string;\n\n  /** Timestamp of when the pause occurred. */\n  readonly pausedAt: number;\n}\n\n// ── PausableHandler ─────────────────────────────────────────\n\n/**\n * Handler for a pausable stage — has two phases: execute and resume.\n *\n * `execute` runs the first time. It can return `{ pause: true }` to pause.\n * `resume` runs when the flowchart is resumed. It receives the resume input.\n *\n * Both phases receive the same scope. After execute pauses, the scope state\n * is preserved in the checkpoint. On resume, the scope is restored before\n * calling resume.\n *\n * @example\n * ```typescript\n * .addPausableFunction('ApproveOrder', {\n *   execute: async (scope) => {\n *     scope.orderId = '123';\n *     scope.amount = 299;\n *     return { pause: true, data: { question: `Approve $${scope.amount} refund?` } };\n *   },\n *   resume: async (scope, input) => {\n *     scope.approved = input.approved;\n *     scope.approver = input.approver;\n *   },\n * }, 'approve-order', 'Manager approval gate')\n *\n * // Later — resume with human's answer\n * await executor.resume(checkpoint, { approved: true, approver: 'Jane' });\n * ```\n */\nexport interface PausableHandler<TScope = any, TInput = unknown, TPauseData = unknown> {\n  /**\n   * First-run phase. Return data to pause, or void/undefined to continue normally.\n   *\n   * Any non-void return value becomes the `pauseData` in the checkpoint.\n   * The consumer defines the `TPauseData` type — the FE uses it to render\n   * the right UI (form fields, approval buttons, etc.).\n   *\n   * @example\n   * ```typescript\n   * // TPauseData = { question: string; riskLevel: string }\n   * const handler: PausableHandler<MyState, { approved: boolean }, { question: string; riskLevel: string }> = {\n   *   execute: async (scope) => {\n   *     return { question: `Approve order ${scope.orderId}?`, riskLevel: 'high' };\n   *   },\n   *   resume: async (scope, input) => {\n   *     scope.approved = input.approved;\n   *   },\n   * };\n   * ```\n   */\n  execute: (scope: TScope) => Promise<TPauseData | void> | TPauseData | void;\n  /**\n   * Resume phase. Called with the resume input when execution continues.\n   *\n   * The scope is restored from the checkpoint's `sharedState`. Writes during\n   * `resume` are committed and visible to subsequent stages.\n   */\n  resume: (scope: TScope, input: TInput) => Promise<void> | void;\n}\n\n// ── Type guard ──────────────────────────────────────────────\n\n/** Check if a value is a PauseResult (stage wants to pause). */\nexport function isPauseResult(value: unknown): value is PauseResult {\n  return typeof value === 'object' && value !== null && (value as PauseResult).pause === true;\n}\n\n/** Check if an error is a PauseSignal. Uses instanceof + name brand fallback for cross-realm safety. */\nexport function isPauseSignal(error: unknown): error is PauseSignal {\n  return (\n    error instanceof PauseSignal ||\n    (error instanceof Error &&\n      error.name === 'PauseSignal' &&\n      Object.prototype.hasOwnProperty.call(error, 'pauseData') &&\n      Object.prototype.hasOwnProperty.call(error, 'stageId'))\n  );\n}\n"]}
|