footprintjs 0.17.2 → 0.18.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CLAUDE.md CHANGED
@@ -84,10 +84,11 @@ await executor.run({
84
84
  env: { traceId: 'req-123', signal: abortSignal, timeoutMs: 5000 },
85
85
  });
86
86
 
87
+ executor.attachRecorder(recorder) // plug scope observer (one-liner, no scopeFactory needed)
87
88
  executor.getNarrative() // combined flow + data narrative
88
- executor.getNarrativeEntries() // structured entries with type/depth/stageName
89
+ executor.getNarrativeEntries() // structured entries with type/depth/stageName/stageId
89
90
  executor.getFlowNarrative() // flow-only (no data ops)
90
- executor.getSnapshot() // full memory state
91
+ executor.getSnapshot() // full memory state (includes recorder snapshots)
91
92
  executor.attachFlowRecorder(r) // plug flow observer
92
93
  executor.setRedactionPolicy({}) // PII protection
93
94
  ```
@@ -127,9 +128,13 @@ Both use `{ id, hooks } → dispatcher → error isolation → attach/detach`. I
127
128
 
128
129
  **FlowRecorder** (control flow — fires AFTER stage execution):
129
130
  - `onStageExecuted`, `onNext`, `onDecision`, `onFork`, `onSelected`, `onSubflowEntry/Exit`, `onLoop`, `onBreak`, `onError`
131
+ - All events carry `traversalContext: TraversalContext` (stageId, stageName, parentStageId, subflowId, subflowPath, depth, loopIteration, forkBranch)
132
+ - Optional `toSnapshot(): { name, data }` — expose collected data for inclusion in `getSnapshot().recorders`
130
133
  - Built-in: 8 strategies (Narrative, Adaptive, Windowed, RLE, Milestone, Progressive, Separate, Manifest, Silent)
134
+ - All event types exported: `FlowStageEvent`, `FlowNextEvent`, `FlowDecisionEvent`, `FlowForkEvent`, `FlowSelectedEvent`, `FlowSubflowEvent`, `FlowSubflowRegisteredEvent`, `FlowLoopEvent`, `FlowBreakEvent`, `FlowErrorEvent`, `TraversalContext`
131
135
 
132
136
  **CombinedNarrativeRecorder** implements BOTH interfaces. Auto-attached by `setEnableNarrative()`.
137
+ - All `CombinedNarrativeEntry` objects carry `stageId?: string` for UI sync (matches spec node id).
133
138
 
134
139
  ## Event Ordering
135
140
 
@@ -126,7 +126,7 @@ export class CombinedNarrativeRecorder {
126
126
  const names = event.children.join(', ');
127
127
  this.entries.push({
128
128
  type: 'fork',
129
- text: `[Parallel]: ${event.children.length} paths were executed in parallel: ${names}.`,
129
+ text: `[Parallel]: Forking into ${event.children.length} parallel paths: ${names}.`,
130
130
  depth: 0,
131
131
  stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
132
132
  subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
@@ -137,7 +137,7 @@ export class CombinedNarrativeRecorder {
137
137
  const names = event.selected.join(', ');
138
138
  this.entries.push({
139
139
  type: 'fork',
140
- text: `[Selected]: ${event.selected.length} of ${event.total} paths were selected: ${names}.`,
140
+ text: `[Selected]: ${event.selected.length} of ${event.total} paths selected for execution: ${names}.`,
141
141
  depth: 0,
142
142
  stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
143
143
  subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
@@ -344,4 +344,4 @@ function summarizeValue(value, maxLen) {
344
344
  }
345
345
  return String(value);
346
346
  }
347
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/CombinedNarrativeRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeRecorder — Inline narrative builder that merges flow + data during traversal.\n *\n * Replaces the post-processing CombinedNarrativeBuilder by implementing BOTH\n * FlowRecorder (control-flow events) and Recorder (scope data events).\n *\n * Event ordering guarantees this works:\n *   1. Scope events (onRead, onWrite) fire DURING stage execution\n *   2. Flow events (onStageExecuted, onDecision) fire AFTER stage execution\n *   3. Both carry the same `stageName` — no matching ambiguity\n *\n * So we buffer scope ops per-stage, then when the flow event arrives,\n * emit the stage entry + flush the buffered ops in one pass.\n */\n\nimport type { ReadEvent, Recorder, WriteEvent } from '../../scope/types.js';\nimport type { CombinedNarrativeEntry } from './CombinedNarrativeBuilder.js';\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\n// ── Types ──────────────────────────────────────────────────────────────────\n\ninterface BufferedOp {\n  type: 'read' | 'write';\n  key: string;\n  valueSummary: string;\n  operation?: 'set' | 'update' | 'delete';\n  stepNumber: number;\n}\n\nexport interface CombinedNarrativeRecorderOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  maxValueLength?: number;\n}\n\n// ── Recorder ───────────────────────────────────────────────────────────────\n\nexport class CombinedNarrativeRecorder implements FlowRecorder, Recorder {\n  readonly id: string;\n\n  private entries: CombinedNarrativeEntry[] = [];\n  private pendingOps = new Map<string, BufferedOp[]>();\n  /** Per-subflow stage counters. Key '' = root flow. */\n  private stageCounters = new Map<string, number>();\n  /** Per-subflow first-stage flags. Key '' = root flow. */\n  private firstStageFlags = new Map<string, boolean>();\n\n  private includeStepNumbers: boolean;\n  private includeValues: boolean;\n  private maxValueLength: number;\n\n  constructor(options?: CombinedNarrativeRecorderOptions & { id?: string }) {\n    this.id = options?.id ?? 'combined-narrative';\n    this.includeStepNumbers = options?.includeStepNumbers ?? true;\n    this.includeValues = options?.includeValues ?? true;\n    this.maxValueLength = options?.maxValueLength ?? 80;\n  }\n\n  // ── Scope channel (fires first, during stage execution) ───────────────\n\n  onRead(event: ReadEvent): void {\n    if (!event.key) return;\n    this.bufferOp(event.stageName, {\n      type: 'read',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n    });\n  }\n\n  onWrite(event: WriteEvent): void {\n    this.bufferOp(event.stageName, {\n      type: 'write',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n      operation: event.operation,\n    });\n  }\n\n  // ── Flow channel (fires after stage execution) ────────────────────────\n\n  onStageExecuted(event: FlowStageEvent): void {\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const text = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.stageName}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.stageName}.`;\n\n    const sfId = event.traversalContext?.subflowId;\n    const stageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId, stageId);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const stageText = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.decider}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.decider}.`;\n\n    const deciderStageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider, event.traversalContext?.subflowId, deciderStageId);\n\n    // Emit the condition entry\n    const branchName = event.chosen;\n    let conditionText: string;\n    if (event.description && event.rationale) {\n      conditionText = `It ${event.description}: ${event.rationale}, so it chose ${branchName}.`;\n    } else if (event.description) {\n      conditionText = `It ${event.description} and chose ${branchName}.`;\n    } else if (event.rationale) {\n      conditionText = `A decision was made: ${event.rationale}, so the path taken was ${branchName}.`;\n    } else {\n      conditionText = `A decision was made, and the path taken was ${branchName}.`;\n    }\n    this.entries.push({\n      type: 'condition',\n      text: `[Condition]: ${conditionText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onNext(): void {\n    // No-op. onStageExecuted already has the description for the next stage.\n    // For deciders (no onStageExecuted), onDecision handles the announcement.\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Parallel]: ${event.children.length} paths were executed in parallel: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Selected]: ${event.selected.length} of ${event.total} paths were selected: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    // Reset stage counter for this subflow so stages start at \"Stage 1\" on re-entry\n    const sfKey = event.subflowId ?? '';\n    this.stageCounters.delete(sfKey);\n    this.firstStageFlags.delete(sfKey);\n\n    const text = event.description\n      ? `Entering the ${event.name} subflow: ${event.description}.`\n      : `Entering the ${event.name} subflow.`;\n    this.entries.push({\n      type: 'subflow',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.entries.push({\n      type: 'subflow',\n      text: `Exiting the ${event.name} subflow.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    const text = event.description\n      ? `On pass ${event.iteration}: ${event.description} again.`\n      : `On pass ${event.iteration} through ${event.target}.`;\n    this.entries.push({\n      type: 'loop',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.entries.push({\n      type: 'break',\n      text: `Execution stopped at ${event.stageName}.`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  /**\n   * Handles errors from both channels:\n   * - FlowRecorder.onError (FlowErrorEvent with message + structuredError)\n   * - Recorder.onError (ErrorEvent from scope system — ignored for narrative)\n   */\n  onError(event: FlowErrorEvent | { stageName?: string; message?: string }): void {\n    // Only handle flow errors (which have `message` and `structuredError`)\n    if (typeof (event as FlowErrorEvent).message !== 'string') return;\n    const flowEvent = event as FlowErrorEvent;\n\n    let text = `An error occurred at ${flowEvent.stageName}: ${flowEvent.message}.`;\n    if (flowEvent.structuredError?.issues?.length) {\n      const details = flowEvent.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      text += ` Validation issues: ${details}.`;\n    }\n    this.entries.push({\n      type: 'error',\n      text: `[Error]: ${text}`,\n      depth: 0,\n      stageName: flowEvent.stageName,\n      stageId: flowEvent.traversalContext?.stageId,\n      subflowId: flowEvent.traversalContext?.subflowId,\n    });\n  }\n\n  // ── Output ────────────────────────────────────────────────────────────\n\n  /** Returns structured entries for programmatic consumption. */\n  getEntries(): CombinedNarrativeEntry[] {\n    return [...this.entries];\n  }\n\n  /** Returns formatted narrative lines (same output as CombinedNarrativeBuilder.build). */\n  getNarrative(indent = '  '): string[] {\n    return this.entries.map((entry) => `${indent.repeat(entry.depth)}${entry.text}`);\n  }\n\n  /**\n   * Returns entries grouped by subflowId for structured access.\n   * Root-level entries have subflowId = undefined.\n   */\n  getEntriesBySubflow(): Record<string, CombinedNarrativeEntry[]> {\n    const result: Record<string, CombinedNarrativeEntry[]> = { '': [] };\n    for (const entry of this.entries) {\n      const key = entry.subflowId ?? '';\n      if (!result[key]) result[key] = [];\n      result[key].push(entry);\n    }\n    return result;\n  }\n\n  /** Clears all state. Called automatically before each run. */\n  clear(): void {\n    this.entries = [];\n    this.pendingOps.clear();\n    this.stageCounters.clear();\n    this.firstStageFlags.clear();\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\n\n  /** Increment and return the stage counter for a given subflow ('' = root). */\n  private incrementStageCounter(subflowKey: string): number {\n    const current = this.stageCounters.get(subflowKey) ?? 0;\n    const next = current + 1;\n    this.stageCounters.set(subflowKey, next);\n    return next;\n  }\n\n  /** Returns true if this is the first stage for the given subflow, consuming the flag. */\n  private consumeFirstStageFlag(subflowKey: string): boolean {\n    if (!this.firstStageFlags.has(subflowKey)) {\n      this.firstStageFlags.set(subflowKey, false);\n      return true;\n    }\n    return false;\n  }\n\n  private bufferOp(stageName: string, op: Omit<BufferedOp, 'stepNumber'>): void {\n    let ops = this.pendingOps.get(stageName);\n    if (!ops) {\n      ops = [];\n      this.pendingOps.set(stageName, ops);\n    }\n    ops.push({ ...op, stepNumber: ops.length + 1 });\n  }\n\n  private flushOps(stageName: string, subflowId?: string, stageId?: string): void {\n    const ops = this.pendingOps.get(stageName);\n    if (!ops || ops.length === 0) return;\n\n    for (const op of ops) {\n      const stepPrefix = this.includeStepNumbers ? `Step ${op.stepNumber}: ` : '';\n\n      let text: string;\n      if (op.type === 'read') {\n        text =\n          this.includeValues && op.valueSummary\n            ? `${stepPrefix}Read ${op.key} = ${op.valueSummary}`\n            : `${stepPrefix}Read ${op.key}`;\n      } else if (op.operation === 'delete') {\n        text = `${stepPrefix}Delete ${op.key}`;\n      } else if (op.operation === 'update') {\n        text = this.includeValues\n          ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Update ${op.key}`;\n      } else {\n        text = this.includeValues\n          ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Write ${op.key}`;\n      }\n\n      this.entries.push({\n        type: 'step',\n        text,\n        depth: 1,\n        stageName,\n        stageId,\n        stepNumber: op.stepNumber,\n        subflowId,\n      });\n    }\n\n    this.pendingOps.delete(stageName);\n  }\n}\n\n// ── Value summarizer (same logic as NarrativeRecorder) ─────────────────────\n\nfunction summarizeValue(value: unknown, maxLen: number): string {\n  if (value === undefined) return 'undefined';\n  if (value === null) return 'null';\n  if (typeof value === 'string') {\n    return value.length <= maxLen ? `\"${value}\"` : `\"${value.slice(0, maxLen - 3)}...\"`;\n  }\n  if (typeof value === 'number' || typeof value === 'boolean') return String(value);\n  if (Array.isArray(value)) {\n    return value.length === 0 ? '[]' : `(${value.length} item${value.length > 1 ? 's' : ''})`;\n  }\n  if (typeof value === 'object') {\n    const keys = Object.keys(value as Record<string, unknown>);\n    if (keys.length === 0) return '{}';\n    const preview = keys.slice(0, 4).join(', ');\n    const suffix = keys.length > 4 ? `, ... (${keys.length} keys)` : '';\n    const result = `{${preview}${suffix}}`;\n    return result.length <= maxLen ? result : `{${keys.length} keys}`;\n  }\n  return String(value);\n}\n"]}
347
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/CombinedNarrativeRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeRecorder — Inline narrative builder that merges flow + data during traversal.\n *\n * Replaces the post-processing CombinedNarrativeBuilder by implementing BOTH\n * FlowRecorder (control-flow events) and Recorder (scope data events).\n *\n * Event ordering guarantees this works:\n *   1. Scope events (onRead, onWrite) fire DURING stage execution\n *   2. Flow events (onStageExecuted, onDecision) fire AFTER stage execution\n *   3. Both carry the same `stageName` — no matching ambiguity\n *\n * So we buffer scope ops per-stage, then when the flow event arrives,\n * emit the stage entry + flush the buffered ops in one pass.\n */\n\nimport type { ReadEvent, Recorder, WriteEvent } from '../../scope/types.js';\nimport type { CombinedNarrativeEntry } from './CombinedNarrativeBuilder.js';\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\n// ── Types ──────────────────────────────────────────────────────────────────\n\ninterface BufferedOp {\n  type: 'read' | 'write';\n  key: string;\n  valueSummary: string;\n  operation?: 'set' | 'update' | 'delete';\n  stepNumber: number;\n}\n\nexport interface CombinedNarrativeRecorderOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  maxValueLength?: number;\n}\n\n// ── Recorder ───────────────────────────────────────────────────────────────\n\nexport class CombinedNarrativeRecorder implements FlowRecorder, Recorder {\n  readonly id: string;\n\n  private entries: CombinedNarrativeEntry[] = [];\n  private pendingOps = new Map<string, BufferedOp[]>();\n  /** Per-subflow stage counters. Key '' = root flow. */\n  private stageCounters = new Map<string, number>();\n  /** Per-subflow first-stage flags. Key '' = root flow. */\n  private firstStageFlags = new Map<string, boolean>();\n\n  private includeStepNumbers: boolean;\n  private includeValues: boolean;\n  private maxValueLength: number;\n\n  constructor(options?: CombinedNarrativeRecorderOptions & { id?: string }) {\n    this.id = options?.id ?? 'combined-narrative';\n    this.includeStepNumbers = options?.includeStepNumbers ?? true;\n    this.includeValues = options?.includeValues ?? true;\n    this.maxValueLength = options?.maxValueLength ?? 80;\n  }\n\n  // ── Scope channel (fires first, during stage execution) ───────────────\n\n  onRead(event: ReadEvent): void {\n    if (!event.key) return;\n    this.bufferOp(event.stageName, {\n      type: 'read',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n    });\n  }\n\n  onWrite(event: WriteEvent): void {\n    this.bufferOp(event.stageName, {\n      type: 'write',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n      operation: event.operation,\n    });\n  }\n\n  // ── Flow channel (fires after stage execution) ────────────────────────\n\n  onStageExecuted(event: FlowStageEvent): void {\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const text = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.stageName}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.stageName}.`;\n\n    const sfId = event.traversalContext?.subflowId;\n    const stageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId, stageId);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const stageText = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.decider}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.decider}.`;\n\n    const deciderStageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider, event.traversalContext?.subflowId, deciderStageId);\n\n    // Emit the condition entry\n    const branchName = event.chosen;\n    let conditionText: string;\n    if (event.description && event.rationale) {\n      conditionText = `It ${event.description}: ${event.rationale}, so it chose ${branchName}.`;\n    } else if (event.description) {\n      conditionText = `It ${event.description} and chose ${branchName}.`;\n    } else if (event.rationale) {\n      conditionText = `A decision was made: ${event.rationale}, so the path taken was ${branchName}.`;\n    } else {\n      conditionText = `A decision was made, and the path taken was ${branchName}.`;\n    }\n    this.entries.push({\n      type: 'condition',\n      text: `[Condition]: ${conditionText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onNext(): void {\n    // No-op. onStageExecuted already has the description for the next stage.\n    // For deciders (no onStageExecuted), onDecision handles the announcement.\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Parallel]: Forking into ${event.children.length} parallel paths: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Selected]: ${event.selected.length} of ${event.total} paths selected for execution: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    // Reset stage counter for this subflow so stages start at \"Stage 1\" on re-entry\n    const sfKey = event.subflowId ?? '';\n    this.stageCounters.delete(sfKey);\n    this.firstStageFlags.delete(sfKey);\n\n    const text = event.description\n      ? `Entering the ${event.name} subflow: ${event.description}.`\n      : `Entering the ${event.name} subflow.`;\n    this.entries.push({\n      type: 'subflow',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.entries.push({\n      type: 'subflow',\n      text: `Exiting the ${event.name} subflow.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    const text = event.description\n      ? `On pass ${event.iteration}: ${event.description} again.`\n      : `On pass ${event.iteration} through ${event.target}.`;\n    this.entries.push({\n      type: 'loop',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.entries.push({\n      type: 'break',\n      text: `Execution stopped at ${event.stageName}.`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  /**\n   * Handles errors from both channels:\n   * - FlowRecorder.onError (FlowErrorEvent with message + structuredError)\n   * - Recorder.onError (ErrorEvent from scope system — ignored for narrative)\n   */\n  onError(event: FlowErrorEvent | { stageName?: string; message?: string }): void {\n    // Only handle flow errors (which have `message` and `structuredError`)\n    if (typeof (event as FlowErrorEvent).message !== 'string') return;\n    const flowEvent = event as FlowErrorEvent;\n\n    let text = `An error occurred at ${flowEvent.stageName}: ${flowEvent.message}.`;\n    if (flowEvent.structuredError?.issues?.length) {\n      const details = flowEvent.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      text += ` Validation issues: ${details}.`;\n    }\n    this.entries.push({\n      type: 'error',\n      text: `[Error]: ${text}`,\n      depth: 0,\n      stageName: flowEvent.stageName,\n      stageId: flowEvent.traversalContext?.stageId,\n      subflowId: flowEvent.traversalContext?.subflowId,\n    });\n  }\n\n  // ── Output ────────────────────────────────────────────────────────────\n\n  /** Returns structured entries for programmatic consumption. */\n  getEntries(): CombinedNarrativeEntry[] {\n    return [...this.entries];\n  }\n\n  /** Returns formatted narrative lines (same output as CombinedNarrativeBuilder.build). */\n  getNarrative(indent = '  '): string[] {\n    return this.entries.map((entry) => `${indent.repeat(entry.depth)}${entry.text}`);\n  }\n\n  /**\n   * Returns entries grouped by subflowId for structured access.\n   * Root-level entries have subflowId = undefined.\n   */\n  getEntriesBySubflow(): Record<string, CombinedNarrativeEntry[]> {\n    const result: Record<string, CombinedNarrativeEntry[]> = { '': [] };\n    for (const entry of this.entries) {\n      const key = entry.subflowId ?? '';\n      if (!result[key]) result[key] = [];\n      result[key].push(entry);\n    }\n    return result;\n  }\n\n  /** Clears all state. Called automatically before each run. */\n  clear(): void {\n    this.entries = [];\n    this.pendingOps.clear();\n    this.stageCounters.clear();\n    this.firstStageFlags.clear();\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\n\n  /** Increment and return the stage counter for a given subflow ('' = root). */\n  private incrementStageCounter(subflowKey: string): number {\n    const current = this.stageCounters.get(subflowKey) ?? 0;\n    const next = current + 1;\n    this.stageCounters.set(subflowKey, next);\n    return next;\n  }\n\n  /** Returns true if this is the first stage for the given subflow, consuming the flag. */\n  private consumeFirstStageFlag(subflowKey: string): boolean {\n    if (!this.firstStageFlags.has(subflowKey)) {\n      this.firstStageFlags.set(subflowKey, false);\n      return true;\n    }\n    return false;\n  }\n\n  private bufferOp(stageName: string, op: Omit<BufferedOp, 'stepNumber'>): void {\n    let ops = this.pendingOps.get(stageName);\n    if (!ops) {\n      ops = [];\n      this.pendingOps.set(stageName, ops);\n    }\n    ops.push({ ...op, stepNumber: ops.length + 1 });\n  }\n\n  private flushOps(stageName: string, subflowId?: string, stageId?: string): void {\n    const ops = this.pendingOps.get(stageName);\n    if (!ops || ops.length === 0) return;\n\n    for (const op of ops) {\n      const stepPrefix = this.includeStepNumbers ? `Step ${op.stepNumber}: ` : '';\n\n      let text: string;\n      if (op.type === 'read') {\n        text =\n          this.includeValues && op.valueSummary\n            ? `${stepPrefix}Read ${op.key} = ${op.valueSummary}`\n            : `${stepPrefix}Read ${op.key}`;\n      } else if (op.operation === 'delete') {\n        text = `${stepPrefix}Delete ${op.key}`;\n      } else if (op.operation === 'update') {\n        text = this.includeValues\n          ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Update ${op.key}`;\n      } else {\n        text = this.includeValues\n          ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Write ${op.key}`;\n      }\n\n      this.entries.push({\n        type: 'step',\n        text,\n        depth: 1,\n        stageName,\n        stageId,\n        stepNumber: op.stepNumber,\n        subflowId,\n      });\n    }\n\n    this.pendingOps.delete(stageName);\n  }\n}\n\n// ── Value summarizer (same logic as NarrativeRecorder) ─────────────────────\n\nfunction summarizeValue(value: unknown, maxLen: number): string {\n  if (value === undefined) return 'undefined';\n  if (value === null) return 'null';\n  if (typeof value === 'string') {\n    return value.length <= maxLen ? `\"${value}\"` : `\"${value.slice(0, maxLen - 3)}...\"`;\n  }\n  if (typeof value === 'number' || typeof value === 'boolean') return String(value);\n  if (Array.isArray(value)) {\n    return value.length === 0 ? '[]' : `(${value.length} item${value.length > 1 ? 's' : ''})`;\n  }\n  if (typeof value === 'object') {\n    const keys = Object.keys(value as Record<string, unknown>);\n    if (keys.length === 0) return '{}';\n    const preview = keys.slice(0, 4).join(', ');\n    const suffix = keys.length > 4 ? `, ... (${keys.length} keys)` : '';\n    const result = `{${preview}${suffix}}`;\n    return result.length <= maxLen ? result : `{${keys.length} keys}`;\n  }\n  return String(value);\n}\n"]}
@@ -51,7 +51,7 @@ export class ControlFlowNarrativeGenerator {
51
51
  }
52
52
  onFork(parentStage, childNames) {
53
53
  const names = childNames.join(', ');
54
- this.sentences.push(`${childNames.length} paths were executed in parallel: ${names}.`);
54
+ this.sentences.push(`Forking into ${childNames.length} parallel paths: ${names}.`);
55
55
  }
56
56
  onSelected(parentStage, selectedNames, totalCount) {
57
57
  const names = selectedNames.join(', ');
@@ -90,4 +90,4 @@ export class ControlFlowNarrativeGenerator {
90
90
  return [...this.sentences];
91
91
  }
92
92
  }
93
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ControlFlowNarrativeGenerator.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/ControlFlowNarrativeGenerator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ControlFlowNarrativeGenerator — Active implementation of IControlFlowNarrative.\n *\n * Converts traversal events into plain-English sentences at traversal time.\n * Produces a human-readable story as a first-class output, enabling any consumer\n * (cheaper LLM, follow-up agent, logging system) to understand what happened\n * without parsing technical structures.\n *\n * This is the FLOW narrative — it captures control flow decisions.\n * The DATA narrative comes from scope/recorders/NarrativeRecorder.\n * CombinedNarrativeBuilder merges both into one story.\n */\n\nimport type { IControlFlowNarrative } from './types.js';\n\nexport class ControlFlowNarrativeGenerator implements IControlFlowNarrative {\n  private sentences: string[] = [];\n  private isFirstStage = true;\n\n  onStageExecuted(stageName: string, description?: string): void {\n    if (this.isFirstStage) {\n      if (description) {\n        this.sentences.push(`The process began: ${description}.`);\n      } else {\n        this.sentences.push(`The process began with ${stageName}.`);\n      }\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(fromStage: string, toStage: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Next step: ${description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${toStage}.`);\n    }\n  }\n\n  onDecision(deciderName: string, chosenBranch: string, rationale?: string, deciderDescription?: string): void {\n    const branchName = chosenBranch;\n    if (deciderDescription && rationale) {\n      this.sentences.push(`It ${deciderDescription}: ${rationale}, so it chose ${branchName}.`);\n    } else if (deciderDescription) {\n      this.sentences.push(`It ${deciderDescription} and chose ${branchName}.`);\n    } else if (rationale) {\n      this.sentences.push(`A decision was made: ${rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n  }\n\n  onFork(parentStage: string, childNames: string[]): void {\n    const names = childNames.join(', ');\n    this.sentences.push(`${childNames.length} paths were executed in parallel: ${names}.`);\n  }\n\n  onSelected(parentStage: string, selectedNames: string[], totalCount: number): void {\n    const names = selectedNames.join(', ');\n    this.sentences.push(`${selectedNames.length} of ${totalCount} paths were selected: ${names}.`);\n  }\n\n  onSubflowEntry(subflowName: string, _subflowId?: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Entering the ${subflowName} subflow: ${description}.`);\n    } else {\n      this.sentences.push(`Entering the ${subflowName} subflow.`);\n    }\n  }\n\n  onSubflowExit(subflowName: string, _subflowId?: string): void {\n    this.sentences.push(`Exiting the ${subflowName} subflow.`);\n  }\n\n  onSubflowRegistered(_subflowId: string, _name: string, _description?: string, _specStructure?: unknown): void {\n    // No narrative output for registration events\n  }\n\n  onLoop(targetStage: string, iteration: number, description?: string): void {\n    if (description) {\n      this.sentences.push(`On pass ${iteration}: ${description} again.`);\n    } else {\n      this.sentences.push(`On pass ${iteration} through ${targetStage}.`);\n    }\n  }\n\n  onBreak(stageName: string): void {\n    this.sentences.push(`Execution stopped at ${stageName}.`);\n  }\n\n  onError(stageName: string, errorMessage: string, _error: unknown): void {\n    this.sentences.push(`An error occurred at ${stageName}: ${errorMessage}.`);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n}\n"]}
93
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ControlFlowNarrativeGenerator.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/ControlFlowNarrativeGenerator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ControlFlowNarrativeGenerator — Active implementation of IControlFlowNarrative.\n *\n * Converts traversal events into plain-English sentences at traversal time.\n * Produces a human-readable story as a first-class output, enabling any consumer\n * (cheaper LLM, follow-up agent, logging system) to understand what happened\n * without parsing technical structures.\n *\n * This is the FLOW narrative — it captures control flow decisions.\n * The DATA narrative comes from scope/recorders/NarrativeRecorder.\n * CombinedNarrativeBuilder merges both into one story.\n */\n\nimport type { IControlFlowNarrative } from './types.js';\n\nexport class ControlFlowNarrativeGenerator implements IControlFlowNarrative {\n  private sentences: string[] = [];\n  private isFirstStage = true;\n\n  onStageExecuted(stageName: string, description?: string): void {\n    if (this.isFirstStage) {\n      if (description) {\n        this.sentences.push(`The process began: ${description}.`);\n      } else {\n        this.sentences.push(`The process began with ${stageName}.`);\n      }\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(fromStage: string, toStage: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Next step: ${description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${toStage}.`);\n    }\n  }\n\n  onDecision(deciderName: string, chosenBranch: string, rationale?: string, deciderDescription?: string): void {\n    const branchName = chosenBranch;\n    if (deciderDescription && rationale) {\n      this.sentences.push(`It ${deciderDescription}: ${rationale}, so it chose ${branchName}.`);\n    } else if (deciderDescription) {\n      this.sentences.push(`It ${deciderDescription} and chose ${branchName}.`);\n    } else if (rationale) {\n      this.sentences.push(`A decision was made: ${rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n  }\n\n  onFork(parentStage: string, childNames: string[]): void {\n    const names = childNames.join(', ');\n    this.sentences.push(`Forking into ${childNames.length} parallel paths: ${names}.`);\n  }\n\n  onSelected(parentStage: string, selectedNames: string[], totalCount: number): void {\n    const names = selectedNames.join(', ');\n    this.sentences.push(`${selectedNames.length} of ${totalCount} paths were selected: ${names}.`);\n  }\n\n  onSubflowEntry(subflowName: string, _subflowId?: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Entering the ${subflowName} subflow: ${description}.`);\n    } else {\n      this.sentences.push(`Entering the ${subflowName} subflow.`);\n    }\n  }\n\n  onSubflowExit(subflowName: string, _subflowId?: string): void {\n    this.sentences.push(`Exiting the ${subflowName} subflow.`);\n  }\n\n  onSubflowRegistered(_subflowId: string, _name: string, _description?: string, _specStructure?: unknown): void {\n    // No narrative output for registration events\n  }\n\n  onLoop(targetStage: string, iteration: number, description?: string): void {\n    if (description) {\n      this.sentences.push(`On pass ${iteration}: ${description} again.`);\n    } else {\n      this.sentences.push(`On pass ${iteration} through ${targetStage}.`);\n    }\n  }\n\n  onBreak(stageName: string): void {\n    this.sentences.push(`Execution stopped at ${stageName}.`);\n  }\n\n  onError(stageName: string, errorMessage: string, _error: unknown): void {\n    this.sentences.push(`An error occurred at ${stageName}: ${errorMessage}.`);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n}\n"]}
@@ -55,7 +55,7 @@ export class NarrativeFlowRecorder {
55
55
  }
56
56
  onFork(event) {
57
57
  const names = event.children.join(', ');
58
- this.sentences.push(`${event.children.length} paths were executed in parallel: ${names}.`);
58
+ this.sentences.push(`Forking into ${event.children.length} parallel paths: ${names}.`);
59
59
  this.stageNames.push(undefined);
60
60
  }
61
61
  onSelected(event) {
@@ -126,4 +126,4 @@ export class NarrativeFlowRecorder {
126
126
  this.isFirstStage = true;
127
127
  }
128
128
  }
129
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NarrativeFlowRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/NarrativeFlowRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * NarrativeFlowRecorder — Default FlowRecorder that generates plain-English narrative.\n *\n * This is the FlowRecorder equivalent of ControlFlowNarrativeGenerator.\n * Produces the same sentences, same format, same behavior — but as a\n * pluggable FlowRecorder that can be swapped, extended, or composed.\n *\n * Consumers who want different narrative behavior (windowed loops, adaptive\n * summarization, etc.) can replace this with a different FlowRecorder.\n */\n\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowNextEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\nexport class NarrativeFlowRecorder implements FlowRecorder {\n  readonly id: string;\n  private sentences: string[] = [];\n  /** Parallel array: the actual stage name that produced each sentence. */\n  private stageNames: (string | undefined)[] = [];\n  private isFirstStage = true;\n\n  constructor(id?: string) {\n    this.id = id ?? 'narrative';\n  }\n\n  onStageExecuted(event: FlowStageEvent): void {\n    if (this.isFirstStage) {\n      if (event.description) {\n        this.sentences.push(`The process began: ${event.description}.`);\n      } else {\n        this.sentences.push(`The process began with ${event.stageName}.`);\n      }\n      this.stageNames.push(event.stageName);\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(event: FlowNextEvent): void {\n    if (event.description) {\n      this.sentences.push(`Next step: ${event.description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${event.to}.`);\n    }\n    this.stageNames.push(event.to);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const branchName = event.chosen;\n    if (event.description && event.rationale) {\n      this.sentences.push(`It ${event.description}: ${event.rationale}, so it chose ${branchName}.`);\n    } else if (event.description) {\n      this.sentences.push(`It ${event.description} and chose ${branchName}.`);\n    } else if (event.rationale) {\n      this.sentences.push(`A decision was made: ${event.rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n    this.stageNames.push(event.decider);\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.sentences.push(`${event.children.length} paths were executed in parallel: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.sentences.push(`${event.selected.length} of ${event.total} paths were selected: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    if (event.description) {\n      this.sentences.push(`Entering the ${event.name} subflow: ${event.description}.`);\n    } else {\n      this.sentences.push(`Entering the ${event.name} subflow.`);\n    }\n    this.stageNames.push(event.name);\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.sentences.push(`Exiting the ${event.name} subflow.`);\n    this.stageNames.push(event.name);\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    if (event.description) {\n      this.sentences.push(`On pass ${event.iteration}: ${event.description} again.`);\n    } else {\n      this.sentences.push(`On pass ${event.iteration} through ${event.target}.`);\n    }\n    this.stageNames.push(event.target);\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.sentences.push(`Execution stopped at ${event.stageName}.`);\n    this.stageNames.push(event.stageName);\n  }\n\n  onError(event: FlowErrorEvent): void {\n    let sentence = `An error occurred at ${event.stageName}: ${event.message}.`;\n\n    // Enrich with field-level issues when available\n    if (event.structuredError.issues && event.structuredError.issues.length > 0) {\n      const issueDetails = event.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      sentence += ` Validation issues: ${issueDetails}.`;\n    }\n\n    this.sentences.push(sentence);\n    this.stageNames.push(event.stageName);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n\n  /**\n   * Returns a parallel array of stage names corresponding to each sentence.\n   * Used by CombinedNarrativeBuilder to match flow sentences to data-level stage operations.\n   *\n   * @deprecated Since 0.9.x — Only needed by the deprecated {@link CombinedNarrativeBuilder}.\n   * The default executor path now uses {@link CombinedNarrativeRecorder} which tracks stage\n   * names inline. Will be removed in v1.0.\n   */\n  getSentenceStageNames(): (string | undefined)[] {\n    return [...this.stageNames];\n  }\n\n  /** Clears accumulated sentences. Useful for reuse across runs. */\n  clear(): void {\n    this.sentences = [];\n    this.stageNames = [];\n    this.isFirstStage = true;\n  }\n}\n"]}
129
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NarrativeFlowRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/NarrativeFlowRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * NarrativeFlowRecorder — Default FlowRecorder that generates plain-English narrative.\n *\n * This is the FlowRecorder equivalent of ControlFlowNarrativeGenerator.\n * Produces the same sentences, same format, same behavior — but as a\n * pluggable FlowRecorder that can be swapped, extended, or composed.\n *\n * Consumers who want different narrative behavior (windowed loops, adaptive\n * summarization, etc.) can replace this with a different FlowRecorder.\n */\n\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowNextEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\nexport class NarrativeFlowRecorder implements FlowRecorder {\n  readonly id: string;\n  private sentences: string[] = [];\n  /** Parallel array: the actual stage name that produced each sentence. */\n  private stageNames: (string | undefined)[] = [];\n  private isFirstStage = true;\n\n  constructor(id?: string) {\n    this.id = id ?? 'narrative';\n  }\n\n  onStageExecuted(event: FlowStageEvent): void {\n    if (this.isFirstStage) {\n      if (event.description) {\n        this.sentences.push(`The process began: ${event.description}.`);\n      } else {\n        this.sentences.push(`The process began with ${event.stageName}.`);\n      }\n      this.stageNames.push(event.stageName);\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(event: FlowNextEvent): void {\n    if (event.description) {\n      this.sentences.push(`Next step: ${event.description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${event.to}.`);\n    }\n    this.stageNames.push(event.to);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const branchName = event.chosen;\n    if (event.description && event.rationale) {\n      this.sentences.push(`It ${event.description}: ${event.rationale}, so it chose ${branchName}.`);\n    } else if (event.description) {\n      this.sentences.push(`It ${event.description} and chose ${branchName}.`);\n    } else if (event.rationale) {\n      this.sentences.push(`A decision was made: ${event.rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n    this.stageNames.push(event.decider);\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.sentences.push(`Forking into ${event.children.length} parallel paths: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.sentences.push(`${event.selected.length} of ${event.total} paths were selected: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    if (event.description) {\n      this.sentences.push(`Entering the ${event.name} subflow: ${event.description}.`);\n    } else {\n      this.sentences.push(`Entering the ${event.name} subflow.`);\n    }\n    this.stageNames.push(event.name);\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.sentences.push(`Exiting the ${event.name} subflow.`);\n    this.stageNames.push(event.name);\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    if (event.description) {\n      this.sentences.push(`On pass ${event.iteration}: ${event.description} again.`);\n    } else {\n      this.sentences.push(`On pass ${event.iteration} through ${event.target}.`);\n    }\n    this.stageNames.push(event.target);\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.sentences.push(`Execution stopped at ${event.stageName}.`);\n    this.stageNames.push(event.stageName);\n  }\n\n  onError(event: FlowErrorEvent): void {\n    let sentence = `An error occurred at ${event.stageName}: ${event.message}.`;\n\n    // Enrich with field-level issues when available\n    if (event.structuredError.issues && event.structuredError.issues.length > 0) {\n      const issueDetails = event.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      sentence += ` Validation issues: ${issueDetails}.`;\n    }\n\n    this.sentences.push(sentence);\n    this.stageNames.push(event.stageName);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n\n  /**\n   * Returns a parallel array of stage names corresponding to each sentence.\n   * Used by CombinedNarrativeBuilder to match flow sentences to data-level stage operations.\n   *\n   * @deprecated Since 0.9.x — Only needed by the deprecated {@link CombinedNarrativeBuilder}.\n   * The default executor path now uses {@link CombinedNarrativeRecorder} which tracks stage\n   * names inline. Will be removed in v1.0.\n   */\n  getSentenceStageNames(): (string | undefined)[] {\n    return [...this.stageNames];\n  }\n\n  /** Clears accumulated sentences. Useful for reuse across runs. */\n  clear(): void {\n    this.sentences = [];\n    this.stageNames = [];\n    this.isFirstStage = true;\n  }\n}\n"]}
@@ -21,6 +21,7 @@ export class FlowChartExecutor {
21
21
  constructor(flowChart, scopeFactory = defaultScopeFactory, defaultValuesForContext, initialContext, readOnlyContext, throttlingErrorChecker, streamHandlers, scopeProtectionMode, enrichSnapshots) {
22
22
  this.narrativeEnabled = false;
23
23
  this.flowRecorders = [];
24
+ this.scopeRecorders = [];
24
25
  this.sharedRedactedKeys = new Set();
25
26
  this.sharedRedactedFieldsByKey = new Map();
26
27
  this.flowChartArgs = {
@@ -60,6 +61,20 @@ export class FlowChartExecutor {
60
61
  else {
61
62
  this.combinedRecorder = undefined;
62
63
  }
64
+ // Attach user-provided scope recorders (from attachRecorder())
65
+ if (this.scopeRecorders.length > 0) {
66
+ const recorders = this.scopeRecorders;
67
+ const prevFactory = scopeFactory;
68
+ scopeFactory = ((ctx, stageName, readOnly, envArg) => {
69
+ const scope = prevFactory(ctx, stageName, readOnly, envArg);
70
+ if (scope && typeof scope.attachRecorder === 'function') {
71
+ for (const r of recorders) {
72
+ scope.attachRecorder(r);
73
+ }
74
+ }
75
+ return scope;
76
+ });
77
+ }
63
78
  // Share redacted keys across all scope instances in this pipeline run.
64
79
  // This ensures that once a key is marked as redacted in one stage,
65
80
  // subsequent stages' recorders also see it as redacted.
@@ -128,6 +143,23 @@ export class FlowChartExecutor {
128
143
  patterns: ((_b = (_a = this.redactionPolicy) === null || _a === void 0 ? void 0 : _a.patterns) !== null && _b !== void 0 ? _b : []).map((p) => p.source),
129
144
  };
130
145
  }
146
+ // ─── Recorder Management ───
147
+ /**
148
+ * Attach a scope Recorder to observe data operations (reads, writes, commits).
149
+ * Automatically attached to every ScopeFacade created during traversal.
150
+ * Must be called before run().
151
+ */
152
+ attachRecorder(recorder) {
153
+ this.scopeRecorders.push(recorder);
154
+ }
155
+ /** Detach all scope Recorders with the given ID. */
156
+ detachRecorder(id) {
157
+ this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);
158
+ }
159
+ /** Returns a defensive copy of attached scope Recorders. */
160
+ getRecorders() {
161
+ return [...this.scopeRecorders];
162
+ }
131
163
  // ─── FlowRecorder Management ───
132
164
  /**
133
165
  * Attach a FlowRecorder to observe control flow events.
@@ -288,4 +320,4 @@ export class FlowChartExecutor {
288
320
  return recorder === null || recorder === void 0 ? void 0 : recorder.getSpec(subflowId);
289
321
  }
290
322
  }
291
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Pairs with FlowChartBuilder:\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *   const executor = new FlowChartExecutor(chart);          // uses default ScopeFacade\n *   const executor = new FlowChartExecutor(chart, myFactory); // custom scope factory\n *   const result = await executor.run();\n */\n\nimport type { CombinedNarrativeEntry } from '../engine/narrative/CombinedNarrativeBuilder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExtractorError,\n  type FlowChart,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    scopeFactory: ScopeFactory<TScope> = defaultScopeFactory as ScopeFactory<TScope>,\n    defaultValuesForContext?: unknown,\n    initialContext?: unknown,\n    readOnlyContext?: unknown,\n    throttlingErrorChecker?: (error: unknown) => boolean,\n    streamHandlers?: StreamHandlers,\n    scopeProtectionMode?: ScopeProtectionMode,\n    enrichSnapshots?: boolean,\n  ) {\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // When narrative is enabled, create a CombinedNarrativeRecorder that implements\n    // BOTH FlowRecorder (control flow) and Recorder (scope data). It builds the\n    // combined narrative inline during traversal — no post-processing merge needed.\n    let scopeFactory = args.scopeFactory;\n    if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder();\n      const recorder = this.combinedRecorder;\n      const originalFactory = args.scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = originalFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).attachRecorder === 'function') {\n          (scope as any).attachRecorder(recorder);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    // Share redacted keys across all scope instances in this pipeline run.\n    // This ensures that once a key is marked as redacted in one stage,\n    // subsequent stages' recorders also see it as redacted.\n    // Also injects the RedactionPolicy if one has been set.\n    {\n      this.sharedRedactedKeys = new Set<string>();\n      this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n      const sharedRedactedKeys = this.sharedRedactedKeys;\n      const policy = this.redactionPolicy;\n      const prevFactory = scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = prevFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).useSharedRedactedKeys === 'function') {\n          (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n        }\n        if (policy && scope && typeof (scope as any).useRedactionPolicy === 'function') {\n          (scope as any).useRedactionPolicy(policy);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    }\n\n    const runtime = new ExecutionRuntime(fc.root.name, fc.root.id, args.defaultValuesForContext, args.initialContext);\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: fc.root,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n    });\n  }\n\n  enableNarrative(): void {\n    this.narrativeEnabled = true;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<TraversalResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // Clear stateful recorders before re-run to prevent cross-run accumulation\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env);\n    try {\n      return await this.traverser.execute();\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getEnrichedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}
323
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Pairs with FlowChartBuilder:\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *   const executor = new FlowChartExecutor(chart);          // uses default ScopeFacade\n *   const executor = new FlowChartExecutor(chart, myFactory); // custom scope factory\n *   const result = await executor.run();\n */\n\nimport type { CombinedNarrativeEntry } from '../engine/narrative/CombinedNarrativeBuilder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExtractorError,\n  type FlowChart,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { Recorder, RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private scopeRecorders: Recorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    scopeFactory: ScopeFactory<TScope> = defaultScopeFactory as ScopeFactory<TScope>,\n    defaultValuesForContext?: unknown,\n    initialContext?: unknown,\n    readOnlyContext?: unknown,\n    throttlingErrorChecker?: (error: unknown) => boolean,\n    streamHandlers?: StreamHandlers,\n    scopeProtectionMode?: ScopeProtectionMode,\n    enrichSnapshots?: boolean,\n  ) {\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // When narrative is enabled, create a CombinedNarrativeRecorder that implements\n    // BOTH FlowRecorder (control flow) and Recorder (scope data). It builds the\n    // combined narrative inline during traversal — no post-processing merge needed.\n    let scopeFactory = args.scopeFactory;\n    if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder();\n      const recorder = this.combinedRecorder;\n      const originalFactory = args.scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = originalFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).attachRecorder === 'function') {\n          (scope as any).attachRecorder(recorder);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    // Attach user-provided scope recorders (from attachRecorder())\n    if (this.scopeRecorders.length > 0) {\n      const recorders = this.scopeRecorders;\n      const prevFactory = scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = prevFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).attachRecorder === 'function') {\n          for (const r of recorders) {\n            (scope as any).attachRecorder(r);\n          }\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    }\n\n    // Share redacted keys across all scope instances in this pipeline run.\n    // This ensures that once a key is marked as redacted in one stage,\n    // subsequent stages' recorders also see it as redacted.\n    // Also injects the RedactionPolicy if one has been set.\n    {\n      this.sharedRedactedKeys = new Set<string>();\n      this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n      const sharedRedactedKeys = this.sharedRedactedKeys;\n      const policy = this.redactionPolicy;\n      const prevFactory = scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = prevFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).useSharedRedactedKeys === 'function') {\n          (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n        }\n        if (policy && scope && typeof (scope as any).useRedactionPolicy === 'function') {\n          (scope as any).useRedactionPolicy(policy);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    }\n\n    const runtime = new ExecutionRuntime(fc.root.name, fc.root.id, args.defaultValuesForContext, args.initialContext);\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: fc.root,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n    });\n  }\n\n  enableNarrative(): void {\n    this.narrativeEnabled = true;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── Recorder Management ───\n\n  /**\n   * Attach a scope Recorder to observe data operations (reads, writes, commits).\n   * Automatically attached to every ScopeFacade created during traversal.\n   * Must be called before run().\n   */\n  attachRecorder(recorder: Recorder): void {\n    this.scopeRecorders.push(recorder);\n  }\n\n  /** Detach all scope Recorders with the given ID. */\n  detachRecorder(id: string): void {\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached scope Recorders. */\n  getRecorders(): Recorder[] {\n    return [...this.scopeRecorders];\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<TraversalResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // Clear stateful recorders before re-run to prevent cross-run accumulation\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env);\n    try {\n      return await this.traverser.execute();\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getEnrichedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}
@@ -129,7 +129,7 @@ class CombinedNarrativeRecorder {
129
129
  const names = event.children.join(', ');
130
130
  this.entries.push({
131
131
  type: 'fork',
132
- text: `[Parallel]: ${event.children.length} paths were executed in parallel: ${names}.`,
132
+ text: `[Parallel]: Forking into ${event.children.length} parallel paths: ${names}.`,
133
133
  depth: 0,
134
134
  stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
135
135
  subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
@@ -140,7 +140,7 @@ class CombinedNarrativeRecorder {
140
140
  const names = event.selected.join(', ');
141
141
  this.entries.push({
142
142
  type: 'fork',
143
- text: `[Selected]: ${event.selected.length} of ${event.total} paths were selected: ${names}.`,
143
+ text: `[Selected]: ${event.selected.length} of ${event.total} paths selected for execution: ${names}.`,
144
144
  depth: 0,
145
145
  stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
146
146
  subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
@@ -348,4 +348,4 @@ function summarizeValue(value, maxLen) {
348
348
  }
349
349
  return String(value);
350
350
  }
351
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeRecorder.js","sourceRoot":"","sources":["../../../../src/lib/engine/narrative/CombinedNarrativeRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeRecorder — Inline narrative builder that merges flow + data during traversal.\n *\n * Replaces the post-processing CombinedNarrativeBuilder by implementing BOTH\n * FlowRecorder (control-flow events) and Recorder (scope data events).\n *\n * Event ordering guarantees this works:\n *   1. Scope events (onRead, onWrite) fire DURING stage execution\n *   2. Flow events (onStageExecuted, onDecision) fire AFTER stage execution\n *   3. Both carry the same `stageName` — no matching ambiguity\n *\n * So we buffer scope ops per-stage, then when the flow event arrives,\n * emit the stage entry + flush the buffered ops in one pass.\n */\n\nimport type { ReadEvent, Recorder, WriteEvent } from '../../scope/types.js';\nimport type { CombinedNarrativeEntry } from './CombinedNarrativeBuilder.js';\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\n// ── Types ──────────────────────────────────────────────────────────────────\n\ninterface BufferedOp {\n  type: 'read' | 'write';\n  key: string;\n  valueSummary: string;\n  operation?: 'set' | 'update' | 'delete';\n  stepNumber: number;\n}\n\nexport interface CombinedNarrativeRecorderOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  maxValueLength?: number;\n}\n\n// ── Recorder ───────────────────────────────────────────────────────────────\n\nexport class CombinedNarrativeRecorder implements FlowRecorder, Recorder {\n  readonly id: string;\n\n  private entries: CombinedNarrativeEntry[] = [];\n  private pendingOps = new Map<string, BufferedOp[]>();\n  /** Per-subflow stage counters. Key '' = root flow. */\n  private stageCounters = new Map<string, number>();\n  /** Per-subflow first-stage flags. Key '' = root flow. */\n  private firstStageFlags = new Map<string, boolean>();\n\n  private includeStepNumbers: boolean;\n  private includeValues: boolean;\n  private maxValueLength: number;\n\n  constructor(options?: CombinedNarrativeRecorderOptions & { id?: string }) {\n    this.id = options?.id ?? 'combined-narrative';\n    this.includeStepNumbers = options?.includeStepNumbers ?? true;\n    this.includeValues = options?.includeValues ?? true;\n    this.maxValueLength = options?.maxValueLength ?? 80;\n  }\n\n  // ── Scope channel (fires first, during stage execution) ───────────────\n\n  onRead(event: ReadEvent): void {\n    if (!event.key) return;\n    this.bufferOp(event.stageName, {\n      type: 'read',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n    });\n  }\n\n  onWrite(event: WriteEvent): void {\n    this.bufferOp(event.stageName, {\n      type: 'write',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n      operation: event.operation,\n    });\n  }\n\n  // ── Flow channel (fires after stage execution) ────────────────────────\n\n  onStageExecuted(event: FlowStageEvent): void {\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const text = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.stageName}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.stageName}.`;\n\n    const sfId = event.traversalContext?.subflowId;\n    const stageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId, stageId);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const stageText = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.decider}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.decider}.`;\n\n    const deciderStageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider, event.traversalContext?.subflowId, deciderStageId);\n\n    // Emit the condition entry\n    const branchName = event.chosen;\n    let conditionText: string;\n    if (event.description && event.rationale) {\n      conditionText = `It ${event.description}: ${event.rationale}, so it chose ${branchName}.`;\n    } else if (event.description) {\n      conditionText = `It ${event.description} and chose ${branchName}.`;\n    } else if (event.rationale) {\n      conditionText = `A decision was made: ${event.rationale}, so the path taken was ${branchName}.`;\n    } else {\n      conditionText = `A decision was made, and the path taken was ${branchName}.`;\n    }\n    this.entries.push({\n      type: 'condition',\n      text: `[Condition]: ${conditionText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onNext(): void {\n    // No-op. onStageExecuted already has the description for the next stage.\n    // For deciders (no onStageExecuted), onDecision handles the announcement.\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Parallel]: ${event.children.length} paths were executed in parallel: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Selected]: ${event.selected.length} of ${event.total} paths were selected: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    // Reset stage counter for this subflow so stages start at \"Stage 1\" on re-entry\n    const sfKey = event.subflowId ?? '';\n    this.stageCounters.delete(sfKey);\n    this.firstStageFlags.delete(sfKey);\n\n    const text = event.description\n      ? `Entering the ${event.name} subflow: ${event.description}.`\n      : `Entering the ${event.name} subflow.`;\n    this.entries.push({\n      type: 'subflow',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.entries.push({\n      type: 'subflow',\n      text: `Exiting the ${event.name} subflow.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    const text = event.description\n      ? `On pass ${event.iteration}: ${event.description} again.`\n      : `On pass ${event.iteration} through ${event.target}.`;\n    this.entries.push({\n      type: 'loop',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.entries.push({\n      type: 'break',\n      text: `Execution stopped at ${event.stageName}.`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  /**\n   * Handles errors from both channels:\n   * - FlowRecorder.onError (FlowErrorEvent with message + structuredError)\n   * - Recorder.onError (ErrorEvent from scope system — ignored for narrative)\n   */\n  onError(event: FlowErrorEvent | { stageName?: string; message?: string }): void {\n    // Only handle flow errors (which have `message` and `structuredError`)\n    if (typeof (event as FlowErrorEvent).message !== 'string') return;\n    const flowEvent = event as FlowErrorEvent;\n\n    let text = `An error occurred at ${flowEvent.stageName}: ${flowEvent.message}.`;\n    if (flowEvent.structuredError?.issues?.length) {\n      const details = flowEvent.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      text += ` Validation issues: ${details}.`;\n    }\n    this.entries.push({\n      type: 'error',\n      text: `[Error]: ${text}`,\n      depth: 0,\n      stageName: flowEvent.stageName,\n      stageId: flowEvent.traversalContext?.stageId,\n      subflowId: flowEvent.traversalContext?.subflowId,\n    });\n  }\n\n  // ── Output ────────────────────────────────────────────────────────────\n\n  /** Returns structured entries for programmatic consumption. */\n  getEntries(): CombinedNarrativeEntry[] {\n    return [...this.entries];\n  }\n\n  /** Returns formatted narrative lines (same output as CombinedNarrativeBuilder.build). */\n  getNarrative(indent = '  '): string[] {\n    return this.entries.map((entry) => `${indent.repeat(entry.depth)}${entry.text}`);\n  }\n\n  /**\n   * Returns entries grouped by subflowId for structured access.\n   * Root-level entries have subflowId = undefined.\n   */\n  getEntriesBySubflow(): Record<string, CombinedNarrativeEntry[]> {\n    const result: Record<string, CombinedNarrativeEntry[]> = { '': [] };\n    for (const entry of this.entries) {\n      const key = entry.subflowId ?? '';\n      if (!result[key]) result[key] = [];\n      result[key].push(entry);\n    }\n    return result;\n  }\n\n  /** Clears all state. Called automatically before each run. */\n  clear(): void {\n    this.entries = [];\n    this.pendingOps.clear();\n    this.stageCounters.clear();\n    this.firstStageFlags.clear();\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\n\n  /** Increment and return the stage counter for a given subflow ('' = root). */\n  private incrementStageCounter(subflowKey: string): number {\n    const current = this.stageCounters.get(subflowKey) ?? 0;\n    const next = current + 1;\n    this.stageCounters.set(subflowKey, next);\n    return next;\n  }\n\n  /** Returns true if this is the first stage for the given subflow, consuming the flag. */\n  private consumeFirstStageFlag(subflowKey: string): boolean {\n    if (!this.firstStageFlags.has(subflowKey)) {\n      this.firstStageFlags.set(subflowKey, false);\n      return true;\n    }\n    return false;\n  }\n\n  private bufferOp(stageName: string, op: Omit<BufferedOp, 'stepNumber'>): void {\n    let ops = this.pendingOps.get(stageName);\n    if (!ops) {\n      ops = [];\n      this.pendingOps.set(stageName, ops);\n    }\n    ops.push({ ...op, stepNumber: ops.length + 1 });\n  }\n\n  private flushOps(stageName: string, subflowId?: string, stageId?: string): void {\n    const ops = this.pendingOps.get(stageName);\n    if (!ops || ops.length === 0) return;\n\n    for (const op of ops) {\n      const stepPrefix = this.includeStepNumbers ? `Step ${op.stepNumber}: ` : '';\n\n      let text: string;\n      if (op.type === 'read') {\n        text =\n          this.includeValues && op.valueSummary\n            ? `${stepPrefix}Read ${op.key} = ${op.valueSummary}`\n            : `${stepPrefix}Read ${op.key}`;\n      } else if (op.operation === 'delete') {\n        text = `${stepPrefix}Delete ${op.key}`;\n      } else if (op.operation === 'update') {\n        text = this.includeValues\n          ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Update ${op.key}`;\n      } else {\n        text = this.includeValues\n          ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Write ${op.key}`;\n      }\n\n      this.entries.push({\n        type: 'step',\n        text,\n        depth: 1,\n        stageName,\n        stageId,\n        stepNumber: op.stepNumber,\n        subflowId,\n      });\n    }\n\n    this.pendingOps.delete(stageName);\n  }\n}\n\n// ── Value summarizer (same logic as NarrativeRecorder) ─────────────────────\n\nfunction summarizeValue(value: unknown, maxLen: number): string {\n  if (value === undefined) return 'undefined';\n  if (value === null) return 'null';\n  if (typeof value === 'string') {\n    return value.length <= maxLen ? `\"${value}\"` : `\"${value.slice(0, maxLen - 3)}...\"`;\n  }\n  if (typeof value === 'number' || typeof value === 'boolean') return String(value);\n  if (Array.isArray(value)) {\n    return value.length === 0 ? '[]' : `(${value.length} item${value.length > 1 ? 's' : ''})`;\n  }\n  if (typeof value === 'object') {\n    const keys = Object.keys(value as Record<string, unknown>);\n    if (keys.length === 0) return '{}';\n    const preview = keys.slice(0, 4).join(', ');\n    const suffix = keys.length > 4 ? `, ... (${keys.length} keys)` : '';\n    const result = `{${preview}${suffix}}`;\n    return result.length <= maxLen ? result : `{${keys.length} keys}`;\n  }\n  return String(value);\n}\n"]}
351
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeRecorder.js","sourceRoot":"","sources":["../../../../src/lib/engine/narrative/CombinedNarrativeRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeRecorder — Inline narrative builder that merges flow + data during traversal.\n *\n * Replaces the post-processing CombinedNarrativeBuilder by implementing BOTH\n * FlowRecorder (control-flow events) and Recorder (scope data events).\n *\n * Event ordering guarantees this works:\n *   1. Scope events (onRead, onWrite) fire DURING stage execution\n *   2. Flow events (onStageExecuted, onDecision) fire AFTER stage execution\n *   3. Both carry the same `stageName` — no matching ambiguity\n *\n * So we buffer scope ops per-stage, then when the flow event arrives,\n * emit the stage entry + flush the buffered ops in one pass.\n */\n\nimport type { ReadEvent, Recorder, WriteEvent } from '../../scope/types.js';\nimport type { CombinedNarrativeEntry } from './CombinedNarrativeBuilder.js';\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\n// ── Types ──────────────────────────────────────────────────────────────────\n\ninterface BufferedOp {\n  type: 'read' | 'write';\n  key: string;\n  valueSummary: string;\n  operation?: 'set' | 'update' | 'delete';\n  stepNumber: number;\n}\n\nexport interface CombinedNarrativeRecorderOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  maxValueLength?: number;\n}\n\n// ── Recorder ───────────────────────────────────────────────────────────────\n\nexport class CombinedNarrativeRecorder implements FlowRecorder, Recorder {\n  readonly id: string;\n\n  private entries: CombinedNarrativeEntry[] = [];\n  private pendingOps = new Map<string, BufferedOp[]>();\n  /** Per-subflow stage counters. Key '' = root flow. */\n  private stageCounters = new Map<string, number>();\n  /** Per-subflow first-stage flags. Key '' = root flow. */\n  private firstStageFlags = new Map<string, boolean>();\n\n  private includeStepNumbers: boolean;\n  private includeValues: boolean;\n  private maxValueLength: number;\n\n  constructor(options?: CombinedNarrativeRecorderOptions & { id?: string }) {\n    this.id = options?.id ?? 'combined-narrative';\n    this.includeStepNumbers = options?.includeStepNumbers ?? true;\n    this.includeValues = options?.includeValues ?? true;\n    this.maxValueLength = options?.maxValueLength ?? 80;\n  }\n\n  // ── Scope channel (fires first, during stage execution) ───────────────\n\n  onRead(event: ReadEvent): void {\n    if (!event.key) return;\n    this.bufferOp(event.stageName, {\n      type: 'read',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n    });\n  }\n\n  onWrite(event: WriteEvent): void {\n    this.bufferOp(event.stageName, {\n      type: 'write',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n      operation: event.operation,\n    });\n  }\n\n  // ── Flow channel (fires after stage execution) ────────────────────────\n\n  onStageExecuted(event: FlowStageEvent): void {\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const text = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.stageName}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.stageName}.`;\n\n    const sfId = event.traversalContext?.subflowId;\n    const stageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId, stageId);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const stageText = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.decider}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.decider}.`;\n\n    const deciderStageId = event.traversalContext?.stageId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider, event.traversalContext?.subflowId, deciderStageId);\n\n    // Emit the condition entry\n    const branchName = event.chosen;\n    let conditionText: string;\n    if (event.description && event.rationale) {\n      conditionText = `It ${event.description}: ${event.rationale}, so it chose ${branchName}.`;\n    } else if (event.description) {\n      conditionText = `It ${event.description} and chose ${branchName}.`;\n    } else if (event.rationale) {\n      conditionText = `A decision was made: ${event.rationale}, so the path taken was ${branchName}.`;\n    } else {\n      conditionText = `A decision was made, and the path taken was ${branchName}.`;\n    }\n    this.entries.push({\n      type: 'condition',\n      text: `[Condition]: ${conditionText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onNext(): void {\n    // No-op. onStageExecuted already has the description for the next stage.\n    // For deciders (no onStageExecuted), onDecision handles the announcement.\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Parallel]: Forking into ${event.children.length} parallel paths: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Selected]: ${event.selected.length} of ${event.total} paths selected for execution: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    // Reset stage counter for this subflow so stages start at \"Stage 1\" on re-entry\n    const sfKey = event.subflowId ?? '';\n    this.stageCounters.delete(sfKey);\n    this.firstStageFlags.delete(sfKey);\n\n    const text = event.description\n      ? `Entering the ${event.name} subflow: ${event.description}.`\n      : `Entering the ${event.name} subflow.`;\n    this.entries.push({\n      type: 'subflow',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.entries.push({\n      type: 'subflow',\n      text: `Exiting the ${event.name} subflow.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    const text = event.description\n      ? `On pass ${event.iteration}: ${event.description} again.`\n      : `On pass ${event.iteration} through ${event.target}.`;\n    this.entries.push({\n      type: 'loop',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.entries.push({\n      type: 'break',\n      text: `Execution stopped at ${event.stageName}.`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  /**\n   * Handles errors from both channels:\n   * - FlowRecorder.onError (FlowErrorEvent with message + structuredError)\n   * - Recorder.onError (ErrorEvent from scope system — ignored for narrative)\n   */\n  onError(event: FlowErrorEvent | { stageName?: string; message?: string }): void {\n    // Only handle flow errors (which have `message` and `structuredError`)\n    if (typeof (event as FlowErrorEvent).message !== 'string') return;\n    const flowEvent = event as FlowErrorEvent;\n\n    let text = `An error occurred at ${flowEvent.stageName}: ${flowEvent.message}.`;\n    if (flowEvent.structuredError?.issues?.length) {\n      const details = flowEvent.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      text += ` Validation issues: ${details}.`;\n    }\n    this.entries.push({\n      type: 'error',\n      text: `[Error]: ${text}`,\n      depth: 0,\n      stageName: flowEvent.stageName,\n      stageId: flowEvent.traversalContext?.stageId,\n      subflowId: flowEvent.traversalContext?.subflowId,\n    });\n  }\n\n  // ── Output ────────────────────────────────────────────────────────────\n\n  /** Returns structured entries for programmatic consumption. */\n  getEntries(): CombinedNarrativeEntry[] {\n    return [...this.entries];\n  }\n\n  /** Returns formatted narrative lines (same output as CombinedNarrativeBuilder.build). */\n  getNarrative(indent = '  '): string[] {\n    return this.entries.map((entry) => `${indent.repeat(entry.depth)}${entry.text}`);\n  }\n\n  /**\n   * Returns entries grouped by subflowId for structured access.\n   * Root-level entries have subflowId = undefined.\n   */\n  getEntriesBySubflow(): Record<string, CombinedNarrativeEntry[]> {\n    const result: Record<string, CombinedNarrativeEntry[]> = { '': [] };\n    for (const entry of this.entries) {\n      const key = entry.subflowId ?? '';\n      if (!result[key]) result[key] = [];\n      result[key].push(entry);\n    }\n    return result;\n  }\n\n  /** Clears all state. Called automatically before each run. */\n  clear(): void {\n    this.entries = [];\n    this.pendingOps.clear();\n    this.stageCounters.clear();\n    this.firstStageFlags.clear();\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\n\n  /** Increment and return the stage counter for a given subflow ('' = root). */\n  private incrementStageCounter(subflowKey: string): number {\n    const current = this.stageCounters.get(subflowKey) ?? 0;\n    const next = current + 1;\n    this.stageCounters.set(subflowKey, next);\n    return next;\n  }\n\n  /** Returns true if this is the first stage for the given subflow, consuming the flag. */\n  private consumeFirstStageFlag(subflowKey: string): boolean {\n    if (!this.firstStageFlags.has(subflowKey)) {\n      this.firstStageFlags.set(subflowKey, false);\n      return true;\n    }\n    return false;\n  }\n\n  private bufferOp(stageName: string, op: Omit<BufferedOp, 'stepNumber'>): void {\n    let ops = this.pendingOps.get(stageName);\n    if (!ops) {\n      ops = [];\n      this.pendingOps.set(stageName, ops);\n    }\n    ops.push({ ...op, stepNumber: ops.length + 1 });\n  }\n\n  private flushOps(stageName: string, subflowId?: string, stageId?: string): void {\n    const ops = this.pendingOps.get(stageName);\n    if (!ops || ops.length === 0) return;\n\n    for (const op of ops) {\n      const stepPrefix = this.includeStepNumbers ? `Step ${op.stepNumber}: ` : '';\n\n      let text: string;\n      if (op.type === 'read') {\n        text =\n          this.includeValues && op.valueSummary\n            ? `${stepPrefix}Read ${op.key} = ${op.valueSummary}`\n            : `${stepPrefix}Read ${op.key}`;\n      } else if (op.operation === 'delete') {\n        text = `${stepPrefix}Delete ${op.key}`;\n      } else if (op.operation === 'update') {\n        text = this.includeValues\n          ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Update ${op.key}`;\n      } else {\n        text = this.includeValues\n          ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Write ${op.key}`;\n      }\n\n      this.entries.push({\n        type: 'step',\n        text,\n        depth: 1,\n        stageName,\n        stageId,\n        stepNumber: op.stepNumber,\n        subflowId,\n      });\n    }\n\n    this.pendingOps.delete(stageName);\n  }\n}\n\n// ── Value summarizer (same logic as NarrativeRecorder) ─────────────────────\n\nfunction summarizeValue(value: unknown, maxLen: number): string {\n  if (value === undefined) return 'undefined';\n  if (value === null) return 'null';\n  if (typeof value === 'string') {\n    return value.length <= maxLen ? `\"${value}\"` : `\"${value.slice(0, maxLen - 3)}...\"`;\n  }\n  if (typeof value === 'number' || typeof value === 'boolean') return String(value);\n  if (Array.isArray(value)) {\n    return value.length === 0 ? '[]' : `(${value.length} item${value.length > 1 ? 's' : ''})`;\n  }\n  if (typeof value === 'object') {\n    const keys = Object.keys(value as Record<string, unknown>);\n    if (keys.length === 0) return '{}';\n    const preview = keys.slice(0, 4).join(', ');\n    const suffix = keys.length > 4 ? `, ... (${keys.length} keys)` : '';\n    const result = `{${preview}${suffix}}`;\n    return result.length <= maxLen ? result : `{${keys.length} keys}`;\n  }\n  return String(value);\n}\n"]}
@@ -54,7 +54,7 @@ class ControlFlowNarrativeGenerator {
54
54
  }
55
55
  onFork(parentStage, childNames) {
56
56
  const names = childNames.join(', ');
57
- this.sentences.push(`${childNames.length} paths were executed in parallel: ${names}.`);
57
+ this.sentences.push(`Forking into ${childNames.length} parallel paths: ${names}.`);
58
58
  }
59
59
  onSelected(parentStage, selectedNames, totalCount) {
60
60
  const names = selectedNames.join(', ');
@@ -94,4 +94,4 @@ class ControlFlowNarrativeGenerator {
94
94
  }
95
95
  }
96
96
  exports.ControlFlowNarrativeGenerator = ControlFlowNarrativeGenerator;
97
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ControlFlowNarrativeGenerator.js","sourceRoot":"","sources":["../../../../src/lib/engine/narrative/ControlFlowNarrativeGenerator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ControlFlowNarrativeGenerator — Active implementation of IControlFlowNarrative.\n *\n * Converts traversal events into plain-English sentences at traversal time.\n * Produces a human-readable story as a first-class output, enabling any consumer\n * (cheaper LLM, follow-up agent, logging system) to understand what happened\n * without parsing technical structures.\n *\n * This is the FLOW narrative — it captures control flow decisions.\n * The DATA narrative comes from scope/recorders/NarrativeRecorder.\n * CombinedNarrativeBuilder merges both into one story.\n */\n\nimport type { IControlFlowNarrative } from './types.js';\n\nexport class ControlFlowNarrativeGenerator implements IControlFlowNarrative {\n  private sentences: string[] = [];\n  private isFirstStage = true;\n\n  onStageExecuted(stageName: string, description?: string): void {\n    if (this.isFirstStage) {\n      if (description) {\n        this.sentences.push(`The process began: ${description}.`);\n      } else {\n        this.sentences.push(`The process began with ${stageName}.`);\n      }\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(fromStage: string, toStage: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Next step: ${description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${toStage}.`);\n    }\n  }\n\n  onDecision(deciderName: string, chosenBranch: string, rationale?: string, deciderDescription?: string): void {\n    const branchName = chosenBranch;\n    if (deciderDescription && rationale) {\n      this.sentences.push(`It ${deciderDescription}: ${rationale}, so it chose ${branchName}.`);\n    } else if (deciderDescription) {\n      this.sentences.push(`It ${deciderDescription} and chose ${branchName}.`);\n    } else if (rationale) {\n      this.sentences.push(`A decision was made: ${rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n  }\n\n  onFork(parentStage: string, childNames: string[]): void {\n    const names = childNames.join(', ');\n    this.sentences.push(`${childNames.length} paths were executed in parallel: ${names}.`);\n  }\n\n  onSelected(parentStage: string, selectedNames: string[], totalCount: number): void {\n    const names = selectedNames.join(', ');\n    this.sentences.push(`${selectedNames.length} of ${totalCount} paths were selected: ${names}.`);\n  }\n\n  onSubflowEntry(subflowName: string, _subflowId?: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Entering the ${subflowName} subflow: ${description}.`);\n    } else {\n      this.sentences.push(`Entering the ${subflowName} subflow.`);\n    }\n  }\n\n  onSubflowExit(subflowName: string, _subflowId?: string): void {\n    this.sentences.push(`Exiting the ${subflowName} subflow.`);\n  }\n\n  onSubflowRegistered(_subflowId: string, _name: string, _description?: string, _specStructure?: unknown): void {\n    // No narrative output for registration events\n  }\n\n  onLoop(targetStage: string, iteration: number, description?: string): void {\n    if (description) {\n      this.sentences.push(`On pass ${iteration}: ${description} again.`);\n    } else {\n      this.sentences.push(`On pass ${iteration} through ${targetStage}.`);\n    }\n  }\n\n  onBreak(stageName: string): void {\n    this.sentences.push(`Execution stopped at ${stageName}.`);\n  }\n\n  onError(stageName: string, errorMessage: string, _error: unknown): void {\n    this.sentences.push(`An error occurred at ${stageName}: ${errorMessage}.`);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n}\n"]}
97
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ControlFlowNarrativeGenerator.js","sourceRoot":"","sources":["../../../../src/lib/engine/narrative/ControlFlowNarrativeGenerator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ControlFlowNarrativeGenerator — Active implementation of IControlFlowNarrative.\n *\n * Converts traversal events into plain-English sentences at traversal time.\n * Produces a human-readable story as a first-class output, enabling any consumer\n * (cheaper LLM, follow-up agent, logging system) to understand what happened\n * without parsing technical structures.\n *\n * This is the FLOW narrative — it captures control flow decisions.\n * The DATA narrative comes from scope/recorders/NarrativeRecorder.\n * CombinedNarrativeBuilder merges both into one story.\n */\n\nimport type { IControlFlowNarrative } from './types.js';\n\nexport class ControlFlowNarrativeGenerator implements IControlFlowNarrative {\n  private sentences: string[] = [];\n  private isFirstStage = true;\n\n  onStageExecuted(stageName: string, description?: string): void {\n    if (this.isFirstStage) {\n      if (description) {\n        this.sentences.push(`The process began: ${description}.`);\n      } else {\n        this.sentences.push(`The process began with ${stageName}.`);\n      }\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(fromStage: string, toStage: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Next step: ${description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${toStage}.`);\n    }\n  }\n\n  onDecision(deciderName: string, chosenBranch: string, rationale?: string, deciderDescription?: string): void {\n    const branchName = chosenBranch;\n    if (deciderDescription && rationale) {\n      this.sentences.push(`It ${deciderDescription}: ${rationale}, so it chose ${branchName}.`);\n    } else if (deciderDescription) {\n      this.sentences.push(`It ${deciderDescription} and chose ${branchName}.`);\n    } else if (rationale) {\n      this.sentences.push(`A decision was made: ${rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n  }\n\n  onFork(parentStage: string, childNames: string[]): void {\n    const names = childNames.join(', ');\n    this.sentences.push(`Forking into ${childNames.length} parallel paths: ${names}.`);\n  }\n\n  onSelected(parentStage: string, selectedNames: string[], totalCount: number): void {\n    const names = selectedNames.join(', ');\n    this.sentences.push(`${selectedNames.length} of ${totalCount} paths were selected: ${names}.`);\n  }\n\n  onSubflowEntry(subflowName: string, _subflowId?: string, description?: string): void {\n    if (description) {\n      this.sentences.push(`Entering the ${subflowName} subflow: ${description}.`);\n    } else {\n      this.sentences.push(`Entering the ${subflowName} subflow.`);\n    }\n  }\n\n  onSubflowExit(subflowName: string, _subflowId?: string): void {\n    this.sentences.push(`Exiting the ${subflowName} subflow.`);\n  }\n\n  onSubflowRegistered(_subflowId: string, _name: string, _description?: string, _specStructure?: unknown): void {\n    // No narrative output for registration events\n  }\n\n  onLoop(targetStage: string, iteration: number, description?: string): void {\n    if (description) {\n      this.sentences.push(`On pass ${iteration}: ${description} again.`);\n    } else {\n      this.sentences.push(`On pass ${iteration} through ${targetStage}.`);\n    }\n  }\n\n  onBreak(stageName: string): void {\n    this.sentences.push(`Execution stopped at ${stageName}.`);\n  }\n\n  onError(stageName: string, errorMessage: string, _error: unknown): void {\n    this.sentences.push(`An error occurred at ${stageName}: ${errorMessage}.`);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n}\n"]}
@@ -58,7 +58,7 @@ class NarrativeFlowRecorder {
58
58
  }
59
59
  onFork(event) {
60
60
  const names = event.children.join(', ');
61
- this.sentences.push(`${event.children.length} paths were executed in parallel: ${names}.`);
61
+ this.sentences.push(`Forking into ${event.children.length} parallel paths: ${names}.`);
62
62
  this.stageNames.push(undefined);
63
63
  }
64
64
  onSelected(event) {
@@ -130,4 +130,4 @@ class NarrativeFlowRecorder {
130
130
  }
131
131
  }
132
132
  exports.NarrativeFlowRecorder = NarrativeFlowRecorder;
133
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NarrativeFlowRecorder.js","sourceRoot":"","sources":["../../../../src/lib/engine/narrative/NarrativeFlowRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * NarrativeFlowRecorder — Default FlowRecorder that generates plain-English narrative.\n *\n * This is the FlowRecorder equivalent of ControlFlowNarrativeGenerator.\n * Produces the same sentences, same format, same behavior — but as a\n * pluggable FlowRecorder that can be swapped, extended, or composed.\n *\n * Consumers who want different narrative behavior (windowed loops, adaptive\n * summarization, etc.) can replace this with a different FlowRecorder.\n */\n\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowNextEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\nexport class NarrativeFlowRecorder implements FlowRecorder {\n  readonly id: string;\n  private sentences: string[] = [];\n  /** Parallel array: the actual stage name that produced each sentence. */\n  private stageNames: (string | undefined)[] = [];\n  private isFirstStage = true;\n\n  constructor(id?: string) {\n    this.id = id ?? 'narrative';\n  }\n\n  onStageExecuted(event: FlowStageEvent): void {\n    if (this.isFirstStage) {\n      if (event.description) {\n        this.sentences.push(`The process began: ${event.description}.`);\n      } else {\n        this.sentences.push(`The process began with ${event.stageName}.`);\n      }\n      this.stageNames.push(event.stageName);\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(event: FlowNextEvent): void {\n    if (event.description) {\n      this.sentences.push(`Next step: ${event.description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${event.to}.`);\n    }\n    this.stageNames.push(event.to);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const branchName = event.chosen;\n    if (event.description && event.rationale) {\n      this.sentences.push(`It ${event.description}: ${event.rationale}, so it chose ${branchName}.`);\n    } else if (event.description) {\n      this.sentences.push(`It ${event.description} and chose ${branchName}.`);\n    } else if (event.rationale) {\n      this.sentences.push(`A decision was made: ${event.rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n    this.stageNames.push(event.decider);\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.sentences.push(`${event.children.length} paths were executed in parallel: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.sentences.push(`${event.selected.length} of ${event.total} paths were selected: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    if (event.description) {\n      this.sentences.push(`Entering the ${event.name} subflow: ${event.description}.`);\n    } else {\n      this.sentences.push(`Entering the ${event.name} subflow.`);\n    }\n    this.stageNames.push(event.name);\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.sentences.push(`Exiting the ${event.name} subflow.`);\n    this.stageNames.push(event.name);\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    if (event.description) {\n      this.sentences.push(`On pass ${event.iteration}: ${event.description} again.`);\n    } else {\n      this.sentences.push(`On pass ${event.iteration} through ${event.target}.`);\n    }\n    this.stageNames.push(event.target);\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.sentences.push(`Execution stopped at ${event.stageName}.`);\n    this.stageNames.push(event.stageName);\n  }\n\n  onError(event: FlowErrorEvent): void {\n    let sentence = `An error occurred at ${event.stageName}: ${event.message}.`;\n\n    // Enrich with field-level issues when available\n    if (event.structuredError.issues && event.structuredError.issues.length > 0) {\n      const issueDetails = event.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      sentence += ` Validation issues: ${issueDetails}.`;\n    }\n\n    this.sentences.push(sentence);\n    this.stageNames.push(event.stageName);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n\n  /**\n   * Returns a parallel array of stage names corresponding to each sentence.\n   * Used by CombinedNarrativeBuilder to match flow sentences to data-level stage operations.\n   *\n   * @deprecated Since 0.9.x — Only needed by the deprecated {@link CombinedNarrativeBuilder}.\n   * The default executor path now uses {@link CombinedNarrativeRecorder} which tracks stage\n   * names inline. Will be removed in v1.0.\n   */\n  getSentenceStageNames(): (string | undefined)[] {\n    return [...this.stageNames];\n  }\n\n  /** Clears accumulated sentences. Useful for reuse across runs. */\n  clear(): void {\n    this.sentences = [];\n    this.stageNames = [];\n    this.isFirstStage = true;\n  }\n}\n"]}
133
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NarrativeFlowRecorder.js","sourceRoot":"","sources":["../../../../src/lib/engine/narrative/NarrativeFlowRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * NarrativeFlowRecorder — Default FlowRecorder that generates plain-English narrative.\n *\n * This is the FlowRecorder equivalent of ControlFlowNarrativeGenerator.\n * Produces the same sentences, same format, same behavior — but as a\n * pluggable FlowRecorder that can be swapped, extended, or composed.\n *\n * Consumers who want different narrative behavior (windowed loops, adaptive\n * summarization, etc.) can replace this with a different FlowRecorder.\n */\n\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowNextEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\nexport class NarrativeFlowRecorder implements FlowRecorder {\n  readonly id: string;\n  private sentences: string[] = [];\n  /** Parallel array: the actual stage name that produced each sentence. */\n  private stageNames: (string | undefined)[] = [];\n  private isFirstStage = true;\n\n  constructor(id?: string) {\n    this.id = id ?? 'narrative';\n  }\n\n  onStageExecuted(event: FlowStageEvent): void {\n    if (this.isFirstStage) {\n      if (event.description) {\n        this.sentences.push(`The process began: ${event.description}.`);\n      } else {\n        this.sentences.push(`The process began with ${event.stageName}.`);\n      }\n      this.stageNames.push(event.stageName);\n      this.isFirstStage = false;\n    }\n  }\n\n  onNext(event: FlowNextEvent): void {\n    if (event.description) {\n      this.sentences.push(`Next step: ${event.description}.`);\n    } else {\n      this.sentences.push(`Next, it moved on to ${event.to}.`);\n    }\n    this.stageNames.push(event.to);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const branchName = event.chosen;\n    if (event.description && event.rationale) {\n      this.sentences.push(`It ${event.description}: ${event.rationale}, so it chose ${branchName}.`);\n    } else if (event.description) {\n      this.sentences.push(`It ${event.description} and chose ${branchName}.`);\n    } else if (event.rationale) {\n      this.sentences.push(`A decision was made: ${event.rationale}, so the path taken was ${branchName}.`);\n    } else {\n      this.sentences.push(`A decision was made, and the path taken was ${branchName}.`);\n    }\n    this.stageNames.push(event.decider);\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.sentences.push(`Forking into ${event.children.length} parallel paths: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.sentences.push(`${event.selected.length} of ${event.total} paths were selected: ${names}.`);\n    this.stageNames.push(undefined);\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    if (event.description) {\n      this.sentences.push(`Entering the ${event.name} subflow: ${event.description}.`);\n    } else {\n      this.sentences.push(`Entering the ${event.name} subflow.`);\n    }\n    this.stageNames.push(event.name);\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.sentences.push(`Exiting the ${event.name} subflow.`);\n    this.stageNames.push(event.name);\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    if (event.description) {\n      this.sentences.push(`On pass ${event.iteration}: ${event.description} again.`);\n    } else {\n      this.sentences.push(`On pass ${event.iteration} through ${event.target}.`);\n    }\n    this.stageNames.push(event.target);\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.sentences.push(`Execution stopped at ${event.stageName}.`);\n    this.stageNames.push(event.stageName);\n  }\n\n  onError(event: FlowErrorEvent): void {\n    let sentence = `An error occurred at ${event.stageName}: ${event.message}.`;\n\n    // Enrich with field-level issues when available\n    if (event.structuredError.issues && event.structuredError.issues.length > 0) {\n      const issueDetails = event.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      sentence += ` Validation issues: ${issueDetails}.`;\n    }\n\n    this.sentences.push(sentence);\n    this.stageNames.push(event.stageName);\n  }\n\n  /** Returns a defensive copy of accumulated sentences. */\n  getSentences(): string[] {\n    return [...this.sentences];\n  }\n\n  /**\n   * Returns a parallel array of stage names corresponding to each sentence.\n   * Used by CombinedNarrativeBuilder to match flow sentences to data-level stage operations.\n   *\n   * @deprecated Since 0.9.x — Only needed by the deprecated {@link CombinedNarrativeBuilder}.\n   * The default executor path now uses {@link CombinedNarrativeRecorder} which tracks stage\n   * names inline. Will be removed in v1.0.\n   */\n  getSentenceStageNames(): (string | undefined)[] {\n    return [...this.stageNames];\n  }\n\n  /** Clears accumulated sentences. Useful for reuse across runs. */\n  clear(): void {\n    this.sentences = [];\n    this.stageNames = [];\n    this.isFirstStage = true;\n  }\n}\n"]}
@@ -24,6 +24,7 @@ class FlowChartExecutor {
24
24
  constructor(flowChart, scopeFactory = defaultScopeFactory, defaultValuesForContext, initialContext, readOnlyContext, throttlingErrorChecker, streamHandlers, scopeProtectionMode, enrichSnapshots) {
25
25
  this.narrativeEnabled = false;
26
26
  this.flowRecorders = [];
27
+ this.scopeRecorders = [];
27
28
  this.sharedRedactedKeys = new Set();
28
29
  this.sharedRedactedFieldsByKey = new Map();
29
30
  this.flowChartArgs = {
@@ -63,6 +64,20 @@ class FlowChartExecutor {
63
64
  else {
64
65
  this.combinedRecorder = undefined;
65
66
  }
67
+ // Attach user-provided scope recorders (from attachRecorder())
68
+ if (this.scopeRecorders.length > 0) {
69
+ const recorders = this.scopeRecorders;
70
+ const prevFactory = scopeFactory;
71
+ scopeFactory = ((ctx, stageName, readOnly, envArg) => {
72
+ const scope = prevFactory(ctx, stageName, readOnly, envArg);
73
+ if (scope && typeof scope.attachRecorder === 'function') {
74
+ for (const r of recorders) {
75
+ scope.attachRecorder(r);
76
+ }
77
+ }
78
+ return scope;
79
+ });
80
+ }
66
81
  // Share redacted keys across all scope instances in this pipeline run.
67
82
  // This ensures that once a key is marked as redacted in one stage,
68
83
  // subsequent stages' recorders also see it as redacted.
@@ -131,6 +146,23 @@ class FlowChartExecutor {
131
146
  patterns: ((_b = (_a = this.redactionPolicy) === null || _a === void 0 ? void 0 : _a.patterns) !== null && _b !== void 0 ? _b : []).map((p) => p.source),
132
147
  };
133
148
  }
149
+ // ─── Recorder Management ───
150
+ /**
151
+ * Attach a scope Recorder to observe data operations (reads, writes, commits).
152
+ * Automatically attached to every ScopeFacade created during traversal.
153
+ * Must be called before run().
154
+ */
155
+ attachRecorder(recorder) {
156
+ this.scopeRecorders.push(recorder);
157
+ }
158
+ /** Detach all scope Recorders with the given ID. */
159
+ detachRecorder(id) {
160
+ this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);
161
+ }
162
+ /** Returns a defensive copy of attached scope Recorders. */
163
+ getRecorders() {
164
+ return [...this.scopeRecorders];
165
+ }
134
166
  // ─── FlowRecorder Management ───
135
167
  /**
136
168
  * Attach a FlowRecorder to observe control flow events.
@@ -292,4 +324,4 @@ class FlowChartExecutor {
292
324
  }
293
325
  }
294
326
  exports.FlowChartExecutor = FlowChartExecutor;
295
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;;AAGH,mGAA6F;AAC7F,2FAAqF;AAErF,mGAA6F;AAE7F,qFAA+E;AAC/E,iDAW4B;AAE5B,4DAAsD;AAEtD,+DAAsG;AACtG,yDAAmD;AAEnD,0EAA0E;AAC1E,MAAM,mBAAmB,GAAiB,CAAC,GAAG,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,EAAE,EAAE,CAC1E,IAAI,4BAAW,CAAC,GAAG,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,CAAC,CAAC;AAEjD,MAAa,iBAAiB;IAqB5B,YACE,SAAkC,EAClC,eAAqC,mBAA2C,EAChF,uBAAiC,EACjC,cAAwB,EACxB,eAAyB,EACzB,sBAAoD,EACpD,cAA+B,EAC/B,mBAAyC,EACzC,eAAyB;QA5BnB,qBAAgB,GAAG,KAAK,CAAC;QAEzB,kBAAa,GAAmB,EAAE,CAAC;QAEnC,uBAAkB,GAAG,IAAI,GAAG,EAAU,CAAC;QACvC,8BAAyB,GAAG,IAAI,GAAG,EAAuB,CAAC;QAyBjE,IAAI,CAAC,aAAa,GAAG;YACnB,SAAS;YACT,YAAY,EAAE,YAAY,aAAZ,YAAY,cAAZ,YAAY,GAAK,mBAA4C;YAC3E,uBAAuB;YACvB,cAAc;YACd,eAAe;YACf,sBAAsB;YACtB,cAAc;YACd,mBAAmB;YACnB,eAAe;SAChB,CAAC;QACF,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;IAC1C,CAAC;IAEO,eAAe,CACrB,MAAoB,EACpB,uBAAiC,EACjC,GAA4C;;QAE5C,MAAM,IAAI,GAAG,IAAI,CAAC,aAAa,CAAC;QAChC,MAAM,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC;QAC1B,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,IAAI,CAAC,MAAA,EAAE,CAAC,eAAe,mCAAI,KAAK,CAAC,CAAC;QAE7E,gFAAgF;QAChF,4EAA4E;QAC5E,gFAAgF;QAChF,IAAI,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QACrC,IAAI,aAAa,EAAE,CAAC;YAClB,IAAI,CAAC,gBAAgB,GAAG,IAAI,wDAAyB,EAAE,CAAC;YACxD,MAAM,QAAQ,GAAG,IAAI,CAAC,gBAAgB,CAAC;YACvC,MAAM,eAAe,GAAG,IAAI,CAAC,YAAY,CAAC;YAC1C,YAAY,GAAG,CAAC,CAAC,GAAQ,EAAE,SAAiB,EAAE,QAAkB,EAAE,MAAY,EAAE,EAAE;gBAChF,MAAM,KAAK,GAAG,eAAe,CAAC,GAAG,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;gBAChE,IAAI,KAAK,IAAI,OAAQ,KAAa,CAAC,cAAc,KAAK,UAAU,EAAE,CAAC;oBAChE,KAAa,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC;gBAC1C,CAAC;gBACD,OAAO,KAAK,CAAC;YACf,CAAC,CAAyB,CAAC;QAC7B,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC;QACpC,CAAC;QAED,uEAAuE;QACvE,mEAAmE;QACnE,wDAAwD;QACxD,wDAAwD;QACxD,CAAC;YACC,IAAI,CAAC,kBAAkB,GAAG,IAAI,GAAG,EAAU,CAAC;YAC5C,IAAI,CAAC,yBAAyB,GAAG,IAAI,GAAG,EAAuB,CAAC;YAChE,MAAM,kBAAkB,GAAG,IAAI,CAAC,kBAAkB,CAAC;YACnD,MAAM,MAAM,GAAG,IAAI,CAAC,eAAe,CAAC;YACpC,MAAM,WAAW,GAAG,YAAY,CAAC;YACjC,YAAY,GAAG,CAAC,CAAC,GAAQ,EAAE,SAAiB,EAAE,QAAkB,EAAE,MAAY,EAAE,EAAE;gBAChF,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;gBAC5D,IAAI,KAAK,IAAI,OAAQ,KAAa,CAAC,qBAAqB,KAAK,UAAU,EAAE,CAAC;oBACvE,KAAa,CAAC,qBAAqB,CAAC,kBAAkB,CAAC,CAAC;gBAC3D,CAAC;gBACD,IAAI,MAAM,IAAI,KAAK,IAAI,OAAQ,KAAa,CAAC,kBAAkB,KAAK,UAAU,EAAE,CAAC;oBAC9E,KAAa,CAAC,kBAAkB,CAAC,MAAM,CAAC,CAAC;gBAC5C,CAAC;gBACD,OAAO,KAAK,CAAC;YACf,CAAC,CAAyB,CAAC;QAC7B,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,sCAAgB,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,uBAAuB,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC;QAElH,OAAO,IAAI,0CAAkB,CAAe;YAC1C,IAAI,EAAE,EAAE,CAAC,IAAI;YACb,QAAQ,EAAE,EAAE,CAAC,QAAQ;YACrB,YAAY;YACZ,gBAAgB,EAAE,OAAO;YACzB,eAAe,EAAE,uBAAuB,aAAvB,uBAAuB,cAAvB,uBAAuB,GAAI,IAAI,CAAC,eAAe;YAChE,sBAAsB,EAAE,IAAI,CAAC,sBAAsB;YACnD,cAAc,EAAE,IAAI,CAAC,cAAc;YACnC,SAAS,EAAE,EAAE,CAAC,SAAS;YACvB,mBAAmB,EAAE,IAAI,CAAC,mBAAmB;YAC7C,QAAQ,EAAE,EAAE,CAAC,QAAQ;YACrB,eAAe,EAAE,MAAA,IAAI,CAAC,eAAe,mCAAI,EAAE,CAAC,eAAe;YAC3D,gBAAgB,EAAE,aAAa;YAC/B,kBAAkB,EAAE,EAAE,CAAC,kBAAkB;YACzC,MAAM,EAAE,MAAA,EAAE,CAAC,MAAM,mCAAI,wBAAa;YAClC,MAAM;YACN,YAAY,EAAE,GAAG;YACjB,aAAa,EAAE,IAAI,CAAC,sBAAsB,EAAE;SAC7C,CAAC,CAAC;IACL,CAAC;IAED,eAAe;QACb,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACH,kBAAkB,CAAC,MAAuB;QACxC,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC;IAChC,CAAC;IAED;;;OAGG;IACH,kBAAkB;;QAChB,MAAM,eAAe,GAA6B,EAAE,CAAC;QACrD,KAAK,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,IAAI,IAAI,CAAC,yBAAyB,EAAE,CAAC;YAC3D,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,CAAC;QACrC,CAAC;QACD,OAAO;YACL,YAAY,EAAE,CAAC,GAAG,IAAI,CAAC,kBAAkB,CAAC;YAC1C,eAAe;YACf,QAAQ,EAAE,CAAC,MAAA,MAAA,IAAI,CAAC,eAAe,0CAAE,QAAQ,mCAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC;SACtE,CAAC;IACJ,CAAC;IAED,kCAAkC;IAElC;;;;OAIG;IACH,kBAAkB,CAAC,QAAsB;QACvC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAClC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;IAC/B,CAAC;IAED,kDAAkD;IAClD,kBAAkB,CAAC,EAAU;QAC3B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IACrE,CAAC;IAED,0DAA0D;IAC1D,gBAAgB;QACd,OAAO,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC;IACjC,CAAC;IAED;;;;;;;OAOG;IACH,YAAY;QACV,gFAAgF;QAChF,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,gBAAgB,CAAC,YAAY,EAAE,CAAC;QAC9C,CAAC;QACD,OAAO,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;IACvC,CAAC;IAED;;;OAGG;IACH,mBAAmB;QACjB,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,gBAAgB,CAAC,UAAU,EAAE,CAAC;QAC5C,CAAC;QACD,MAAM,aAAa,GAAG,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;QACpD,OAAO,aAAa,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,OAAgB,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;IACnF,CAAC;IAED;;;;;OAKG;IACK,sBAAsB;QAC5B,MAAM,SAAS,GAAmB,EAAE,CAAC;QACrC,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YACtC,2EAA2E;YAC3E,SAAS,CAAC,IAAI,CAAC,IAAI,gDAAqB,EAAE,CAAC,CAAC;QAC9C,CAAC;QACD,SAAS,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC;QACtC,OAAO,SAAS,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC;IACtD,CAAC;IAED;;;OAGG;IACH,gBAAgB;QACd,OAAO,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;IACvC,CAAC;IAED,KAAK,CAAC,GAAG,CAAC,OAAoB;;QAC5B,IAAI,MAAM,GAAG,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,MAAM,CAAC;QAC7B,IAAI,SAAoD,CAAC;QAEzD,mDAAmD;QACnD,IAAI,CAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,SAAS,KAAI,CAAC,MAAM,EAAE,CAAC;YAClC,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;YACzC,MAAM,GAAG,UAAU,CAAC,MAAM,CAAC;YAC3B,SAAS,GAAG,UAAU,CACpB,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,6BAA6B,OAAO,CAAC,SAAS,IAAI,CAAC,CAAC,EACrF,OAAO,CAAC,SAAS,CAClB,CAAC;QACJ,CAAC;QAED,yDAAyD;QACzD,IAAI,cAAc,GAAG,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,KAAK,CAAC;QACpC,IAAI,cAAc,IAAI,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;YAC/D,cAAc,GAAG,IAAA,gCAAa,EAAC,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,WAAW,EAAE,cAAc,CAAC,CAAC;QAC3F,CAAC;QAED,2EAA2E;QAC3E,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACnC,MAAA,CAAC,CAAC,KAAK,iDAAI,CAAC;QACd,CAAC;QAED,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,cAAc,EAAE,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,GAAG,CAAC,CAAC;QAC5E,IAAI,CAAC;YACH,OAAO,MAAM,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC;QACxC,CAAC;gBAAS,CAAC;YACT,IAAI,SAAS,KAAK,SAAS;gBAAE,YAAY,CAAC,SAAS,CAAC,CAAC;QACvD,CAAC;IACH,CAAC;IAED,wBAAwB;IAExB,WAAW;QACT,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAqB,CAAC;QACjE,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,iBAAiB,EAAE,CAAC;QACrD,IAAI,SAAS,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;YACvB,QAAQ,CAAC,cAAc,GAAG,MAAM,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;QAC1D,CAAC;QAED,mEAAmE;QACnE,MAAM,iBAAiB,GAAuB,EAAE,CAAC;QACjD,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACnC,IAAI,CAAC,CAAC,UAAU,EAAE,CAAC;gBACjB,MAAM,EAAE,IAAI,EAAE,IAAI,EAAE,GAAG,CAAC,CAAC,UAAU,EAAE,CAAC;gBACtC,iBAAiB,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;YACnD,CAAC;QACH,CAAC;QACD,IAAI,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACjC,QAAQ,CAAC,SAAS,GAAG,iBAAiB,CAAC;QACzC,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,UAAU;QACR,OAAO,IAAI,CAAC,SAAS,CAAC,UAAU,EAAE,CAAC;IACrC,CAAC;IAED,aAAa,CAAC,IAAc,EAAE,GAAW,EAAE,KAAc;QACvD,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,IAAI,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC;IACjD,CAAC;IAED,YAAY;QACV,OAAO,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;IACvC,CAAC;IAED,cAAc;QACZ,OAAO,IAAI,CAAC,SAAS,CAAC,cAAc,EAAE,CAAC;IACzC,CAAC;IAED,mBAAmB;QACjB,OAAO,IAAI,CAAC,SAAS,CAAC,mBAAmB,EAAE,CAAC;IAC9C,CAAC;IAED,iBAAiB;QACf,OAAO,IAAI,CAAC,SAAS,CAAC,iBAAiB,EAAE,CAAC;IAC5C,CAAC;IAED,mBAAmB;QACjB,OAAO,IAAI,CAAC,SAAS,CAAC,mBAAmB,EAAW,CAAC;IACvD,CAAC;IAED,kBAAkB;QAChB,OAAO,IAAI,CAAC,SAAS,CAAC,mBAAmB,EAAW,CAAC;IACvD,CAAC;IAED,kBAAkB;QAChB,OAAO,IAAI,CAAC,SAAS,CAAC,kBAAkB,EAAE,CAAC;IAC7C,CAAC;IAED;;;OAGG;IACH,kBAAkB;;QAChB,MAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,YAAY,8CAAoB,CAEpE,CAAC;QACd,OAAO,MAAA,QAAQ,aAAR,QAAQ,uBAAR,QAAQ,CAAE,WAAW,EAAE,mCAAI,EAAE,CAAC;IACvC,CAAC;IAED;;;OAGG;IACH,cAAc,CAAC,SAAiB;QAC9B,MAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,YAAY,8CAAoB,CAEpE,CAAC;QACd,OAAO,QAAQ,aAAR,QAAQ,uBAAR,QAAQ,CAAE,OAAO,CAAC,SAAS,CAAC,CAAC;IACtC,CAAC;CACF;AAhVD,8CAgVC","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Pairs with FlowChartBuilder:\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *   const executor = new FlowChartExecutor(chart);          // uses default ScopeFacade\n *   const executor = new FlowChartExecutor(chart, myFactory); // custom scope factory\n *   const result = await executor.run();\n */\n\nimport type { CombinedNarrativeEntry } from '../engine/narrative/CombinedNarrativeBuilder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExtractorError,\n  type FlowChart,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    scopeFactory: ScopeFactory<TScope> = defaultScopeFactory as ScopeFactory<TScope>,\n    defaultValuesForContext?: unknown,\n    initialContext?: unknown,\n    readOnlyContext?: unknown,\n    throttlingErrorChecker?: (error: unknown) => boolean,\n    streamHandlers?: StreamHandlers,\n    scopeProtectionMode?: ScopeProtectionMode,\n    enrichSnapshots?: boolean,\n  ) {\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // When narrative is enabled, create a CombinedNarrativeRecorder that implements\n    // BOTH FlowRecorder (control flow) and Recorder (scope data). It builds the\n    // combined narrative inline during traversal — no post-processing merge needed.\n    let scopeFactory = args.scopeFactory;\n    if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder();\n      const recorder = this.combinedRecorder;\n      const originalFactory = args.scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = originalFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).attachRecorder === 'function') {\n          (scope as any).attachRecorder(recorder);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    // Share redacted keys across all scope instances in this pipeline run.\n    // This ensures that once a key is marked as redacted in one stage,\n    // subsequent stages' recorders also see it as redacted.\n    // Also injects the RedactionPolicy if one has been set.\n    {\n      this.sharedRedactedKeys = new Set<string>();\n      this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n      const sharedRedactedKeys = this.sharedRedactedKeys;\n      const policy = this.redactionPolicy;\n      const prevFactory = scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = prevFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).useSharedRedactedKeys === 'function') {\n          (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n        }\n        if (policy && scope && typeof (scope as any).useRedactionPolicy === 'function') {\n          (scope as any).useRedactionPolicy(policy);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    }\n\n    const runtime = new ExecutionRuntime(fc.root.name, fc.root.id, args.defaultValuesForContext, args.initialContext);\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: fc.root,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n    });\n  }\n\n  enableNarrative(): void {\n    this.narrativeEnabled = true;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<TraversalResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // Clear stateful recorders before re-run to prevent cross-run accumulation\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env);\n    try {\n      return await this.traverser.execute();\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getEnrichedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}
327
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Pairs with FlowChartBuilder:\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *   const executor = new FlowChartExecutor(chart);          // uses default ScopeFacade\n *   const executor = new FlowChartExecutor(chart, myFactory); // custom scope factory\n *   const result = await executor.run();\n */\n\nimport type { CombinedNarrativeEntry } from '../engine/narrative/CombinedNarrativeBuilder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExtractorError,\n  type FlowChart,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { Recorder, RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private scopeRecorders: Recorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    scopeFactory: ScopeFactory<TScope> = defaultScopeFactory as ScopeFactory<TScope>,\n    defaultValuesForContext?: unknown,\n    initialContext?: unknown,\n    readOnlyContext?: unknown,\n    throttlingErrorChecker?: (error: unknown) => boolean,\n    streamHandlers?: StreamHandlers,\n    scopeProtectionMode?: ScopeProtectionMode,\n    enrichSnapshots?: boolean,\n  ) {\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // When narrative is enabled, create a CombinedNarrativeRecorder that implements\n    // BOTH FlowRecorder (control flow) and Recorder (scope data). It builds the\n    // combined narrative inline during traversal — no post-processing merge needed.\n    let scopeFactory = args.scopeFactory;\n    if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder();\n      const recorder = this.combinedRecorder;\n      const originalFactory = args.scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = originalFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).attachRecorder === 'function') {\n          (scope as any).attachRecorder(recorder);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    // Attach user-provided scope recorders (from attachRecorder())\n    if (this.scopeRecorders.length > 0) {\n      const recorders = this.scopeRecorders;\n      const prevFactory = scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = prevFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).attachRecorder === 'function') {\n          for (const r of recorders) {\n            (scope as any).attachRecorder(r);\n          }\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    }\n\n    // Share redacted keys across all scope instances in this pipeline run.\n    // This ensures that once a key is marked as redacted in one stage,\n    // subsequent stages' recorders also see it as redacted.\n    // Also injects the RedactionPolicy if one has been set.\n    {\n      this.sharedRedactedKeys = new Set<string>();\n      this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n      const sharedRedactedKeys = this.sharedRedactedKeys;\n      const policy = this.redactionPolicy;\n      const prevFactory = scopeFactory;\n      scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n        const scope = prevFactory(ctx, stageName, readOnly, envArg);\n        if (scope && typeof (scope as any).useSharedRedactedKeys === 'function') {\n          (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n        }\n        if (policy && scope && typeof (scope as any).useRedactionPolicy === 'function') {\n          (scope as any).useRedactionPolicy(policy);\n        }\n        return scope;\n      }) as ScopeFactory<TScope>;\n    }\n\n    const runtime = new ExecutionRuntime(fc.root.name, fc.root.id, args.defaultValuesForContext, args.initialContext);\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: fc.root,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n    });\n  }\n\n  enableNarrative(): void {\n    this.narrativeEnabled = true;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── Recorder Management ───\n\n  /**\n   * Attach a scope Recorder to observe data operations (reads, writes, commits).\n   * Automatically attached to every ScopeFacade created during traversal.\n   * Must be called before run().\n   */\n  attachRecorder(recorder: Recorder): void {\n    this.scopeRecorders.push(recorder);\n  }\n\n  /** Detach all scope Recorders with the given ID. */\n  detachRecorder(id: string): void {\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached scope Recorders. */\n  getRecorders(): Recorder[] {\n    return [...this.scopeRecorders];\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<TraversalResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // Clear stateful recorders before re-run to prevent cross-run accumulation\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env);\n    try {\n      return await this.traverser.execute();\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getEnrichedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}
@@ -12,13 +12,14 @@ import type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRe
12
12
  import type { FlowRecorder } from '../engine/narrative/types.js';
13
13
  import { type ExtractorError, type FlowChart, type RunOptions, type ScopeFactory, type SerializedPipelineStructure, type StageNode, type StreamHandlers, type SubflowResult, type TraversalResult } from '../engine/types.js';
14
14
  import type { ScopeProtectionMode } from '../scope/protection/types.js';
15
- import type { RedactionPolicy, RedactionReport } from '../scope/types.js';
15
+ import type { Recorder, RedactionPolicy, RedactionReport } from '../scope/types.js';
16
16
  import { type RuntimeSnapshot } from './ExecutionRuntime.js';
17
17
  export declare class FlowChartExecutor<TOut = any, TScope = any> {
18
18
  private traverser;
19
19
  private narrativeEnabled;
20
20
  private combinedRecorder;
21
21
  private flowRecorders;
22
+ private scopeRecorders;
22
23
  private redactionPolicy;
23
24
  private sharedRedactedKeys;
24
25
  private sharedRedactedFieldsByKey;
@@ -36,6 +37,16 @@ export declare class FlowChartExecutor<TOut = any, TScope = any> {
36
37
  * most recent run. Never includes actual values.
37
38
  */
38
39
  getRedactionReport(): RedactionReport;
40
+ /**
41
+ * Attach a scope Recorder to observe data operations (reads, writes, commits).
42
+ * Automatically attached to every ScopeFacade created during traversal.
43
+ * Must be called before run().
44
+ */
45
+ attachRecorder(recorder: Recorder): void;
46
+ /** Detach all scope Recorders with the given ID. */
47
+ detachRecorder(id: string): void;
48
+ /** Returns a defensive copy of attached scope Recorders. */
49
+ getRecorders(): Recorder[];
39
50
  /**
40
51
  * Attach a FlowRecorder to observe control flow events.
41
52
  * Automatically enables narrative if not already enabled.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "footprintjs",
3
- "version": "0.17.2",
3
+ "version": "0.18.0",
4
4
  "description": "Turn your whiteboard flowchart into running code — with automatic causal traces for LLM reasoning",
5
5
  "license": "MIT",
6
6
  "author": "Sanjay Krishna Anbalagan",