footprintjs 0.16.0 → 0.17.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/dist/esm/index.js +1 -1
- package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +45 -21
- package/dist/esm/lib/engine/narrative/recorders/ManifestFlowRecorder.js +4 -1
- package/dist/esm/lib/engine/narrative/types.js +1 -1
- package/dist/esm/lib/engine/types.js +1 -1
- package/dist/esm/lib/runner/ExecutionRuntime.js +1 -1
- package/dist/esm/lib/runner/FlowChartExecutor.js +12 -1
- package/dist/esm/lib/runner/index.js +1 -1
- package/dist/esm/lib/scope/recorders/MetricRecorder.js +14 -1
- package/dist/index.js +1 -1
- package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +45 -21
- package/dist/lib/engine/narrative/recorders/ManifestFlowRecorder.js +4 -1
- package/dist/lib/engine/narrative/types.js +1 -1
- package/dist/lib/engine/types.js +1 -1
- package/dist/lib/runner/ExecutionRuntime.js +1 -1
- package/dist/lib/runner/FlowChartExecutor.js +12 -1
- package/dist/lib/runner/index.js +1 -1
- package/dist/lib/scope/recorders/MetricRecorder.js +14 -1
- package/dist/types/index.d.ts +2 -2
- package/dist/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +8 -2
- package/dist/types/lib/engine/narrative/recorders/ManifestFlowRecorder.d.ts +4 -0
- package/dist/types/lib/engine/narrative/types.d.ts +5 -0
- package/dist/types/lib/engine/traversal/FlowchartTraverser.d.ts +5 -0
- package/dist/types/lib/engine/types.d.ts +5 -0
- package/dist/types/lib/runner/ExecutionRuntime.d.ts +8 -0
- package/dist/types/lib/runner/FlowChartExecutor.d.ts +2 -6
- package/dist/types/lib/runner/index.d.ts +1 -1
- package/dist/types/lib/scope/recorders/MetricRecorder.d.ts +4 -0
- package/package.json +1 -1
package/dist/esm/index.js
CHANGED
|
@@ -40,4 +40,4 @@ export { normalizeSchema, zodToJsonSchema } from './lib/contract/index.js';
|
|
|
40
40
|
export { generateOpenAPI } from './lib/contract/index.js';
|
|
41
41
|
export { detectSchema, isValidatable, isZod } from './lib/schema/index.js';
|
|
42
42
|
export { InputValidationError, validateAgainstSchema, validateOrThrow } from './lib/schema/index.js';
|
|
43
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
43
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAOH,OAAO,EAAE,SAAS,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAErE,+EAA+E;AAC/E,uCAAuC;AACvC,+EAA+E;AAE/E,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAQ1D,OAAO,EAAE,kBAAkB,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AAE7E,+EAA+E;AAC/E,0CAA0C;AAC1C,+EAA+E;AAE/E,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAEnD,YAAY;AACZ,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AAazD,8BAA8B;AAC9B,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAO1D,OAAO,EAAE,wBAAwB,EAAE,MAAM,uBAAuB,CAAC;AAiBjE,OAAO,EAAE,qBAAqB,EAAE,MAAM,uBAAuB,CAAC;AAI9D,OAAO,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAI1E,OAAO,EAAE,oBAAoB,EAAE,MAAM,uBAAuB,CAAC;AAC7D,OAAO,EAAE,6BAA6B,EAAE,MAAM,uBAAuB,CAAC;AACtE,OAAO,EAAE,8BAA8B,EAAE,MAAM,uBAAuB,CAAC;AACvE,OAAO,EAAE,gCAAgC,EAAE,MAAM,uBAAuB,CAAC;AACzE,OAAO,EAAE,wBAAwB,EAAE,MAAM,uBAAuB,CAAC;AACjE,OAAO,EAAE,6BAA6B,EAAE,MAAM,uBAAuB,CAAC;AACtE,OAAO,EAAE,2BAA2B,EAAE,MAAM,uBAAuB,CAAC;AACpE,OAAO,EAAE,6BAA6B,EAAE,MAAM,uBAAuB,CAAC;AAoBtE,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AAC3E,OAAO,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AAO1D,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,uBAAuB,CAAC;AAC3E,OAAO,EAAE,oBAAoB,EAAE,qBAAqB,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC","sourcesContent":["/**\n * FootPrint — Public API\n *\n * Connected causal trace library for LLM pipelines.\n * Builds flowcharts, executes them via DFS traversal, and captures\n * every stage's context (state, decisions, errors) in an auditable trace.\n *\n * For advanced/internal APIs (memory primitives, engine handlers, providers),\n * import from 'footprint/advanced'.\n */\n\n// ============================================================================\n// Builder — Flowchart construction\n// ============================================================================\n\nexport type { FlowChart, PipelineStageFunction as StageHandler, StreamHandlers } from './lib/builder/index.js';\nexport { flowChart, FlowChartBuilder } from './lib/builder/index.js';\n\n// ============================================================================\n// Runner — Execution convenience layer\n// ============================================================================\n\nexport { FlowChartExecutor } from './lib/runner/index.js';\n\n// ComposableRunner — interface for runners that expose their internal flowChart\n// for subflow composition (enables UI drill-down into nested runners)\nexport type { ComposableRunner } from './lib/runner/index.js';\n\n// Snapshot navigation — drill into subflow subtrees by path\nexport type { RecorderSnapshot, RuntimeSnapshot, SubtreeSnapshot } from './lib/runner/index.js';\nexport { getSubtreeSnapshot, listSubflowPaths } from './lib/runner/index.js';\n\n// ============================================================================\n// Scope — Per-stage facades and recorders\n// ============================================================================\n\nexport { ScopeFacade } from './lib/scope/index.js';\n\n// Recorders\nexport { MetricRecorder } from './lib/scope/index.js';\nexport { DebugRecorder } from './lib/scope/index.js';\nexport { NarrativeRecorder } from './lib/scope/index.js';\n\n// Recorder interface and core event types (needed to implement custom Recorder)\nexport type {\n  CommitEvent,\n  ErrorEvent,\n  ReadEvent,\n  Recorder,\n  RedactionPolicy,\n  RedactionReport,\n  WriteEvent,\n} from './lib/scope/index.js';\n\n// Zod-based scope definitions\nexport { defineScopeFromZod } from './lib/scope/index.js';\n\n// ============================================================================\n// Engine — Narrative (commonly used)\n// ============================================================================\n\nexport type { CombinedNarrativeEntry } from './lib/engine/index.js';\nexport { CombinedNarrativeBuilder } from './lib/engine/index.js';\n\n// FlowRecorder — Pluggable observer for control flow events (mirrors scope Recorder)\nexport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowNextEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n  FlowSubflowRegisteredEvent,\n  TraversalContext,\n} from './lib/engine/index.js';\nexport { NarrativeFlowRecorder } from './lib/engine/index.js';\n\n// Structured error extraction — preserves field-level details through the pipeline\nexport type { StructuredErrorInfo } from './lib/engine/index.js';\nexport { extractErrorInfo, formatErrorInfo } from './lib/engine/index.js';\n\n// Built-in FlowRecorder strategies (tree-shakeable — import only what you use)\nexport type { ManifestEntry } from './lib/engine/index.js';\nexport { ManifestFlowRecorder } from './lib/engine/index.js';\nexport { AdaptiveNarrativeFlowRecorder } from './lib/engine/index.js';\nexport { MilestoneNarrativeFlowRecorder } from './lib/engine/index.js';\nexport { ProgressiveNarrativeFlowRecorder } from './lib/engine/index.js';\nexport { RLENarrativeFlowRecorder } from './lib/engine/index.js';\nexport { SeparateNarrativeFlowRecorder } from './lib/engine/index.js';\nexport { SilentNarrativeFlowRecorder } from './lib/engine/index.js';\nexport { WindowedNarrativeFlowRecorder } from './lib/engine/index.js';\n\n// ============================================================================\n// Memory — ScopeFactory type (needed for FlowChartExecutor constructor)\n// ============================================================================\n\nexport type { ExecutionEnv, RunOptions } from './lib/engine/index.js';\nexport type { ScopeFactory } from './lib/memory/index.js';\n\n// ============================================================================\n// Contract — I/O boundary, schemas, and OpenAPI generation\n// ============================================================================\n\nexport type {\n  FlowChartContract,\n  FlowChartContractOptions,\n  JsonSchema,\n  OpenAPIOptions,\n  OpenAPISpec,\n} from './lib/contract/index.js';\nexport { defineContract } from './lib/contract/index.js';\nexport { normalizeSchema, zodToJsonSchema } from './lib/contract/index.js';\nexport { generateOpenAPI } from './lib/contract/index.js';\n\n// ============================================================================\n// Schema — Unified detection, validation, and structured errors\n// ============================================================================\n\nexport type { SchemaKind, ValidationIssue, ValidationResult } from './lib/schema/index.js';\nexport { detectSchema, isValidatable, isZod } from './lib/schema/index.js';\nexport { InputValidationError, validateAgainstSchema, validateOrThrow } from './lib/schema/index.js';\n"]}
|
|
@@ -18,8 +18,10 @@ export class CombinedNarrativeRecorder {
|
|
|
18
18
|
var _a, _b, _c, _d;
|
|
19
19
|
this.entries = [];
|
|
20
20
|
this.pendingOps = new Map();
|
|
21
|
-
|
|
22
|
-
this.
|
|
21
|
+
/** Per-subflow stage counters. Key '' = root flow. */
|
|
22
|
+
this.stageCounters = new Map();
|
|
23
|
+
/** Per-subflow first-stage flags. Key '' = root flow. */
|
|
24
|
+
this.firstStageFlags = new Map();
|
|
23
25
|
this.id = (_a = options === null || options === void 0 ? void 0 : options.id) !== null && _a !== void 0 ? _a : 'combined-narrative';
|
|
24
26
|
this.includeStepNumbers = (_b = options === null || options === void 0 ? void 0 : options.includeStepNumbers) !== null && _b !== void 0 ? _b : true;
|
|
25
27
|
this.includeValues = (_c = options === null || options === void 0 ? void 0 : options.includeValues) !== null && _c !== void 0 ? _c : true;
|
|
@@ -45,20 +47,21 @@ export class CombinedNarrativeRecorder {
|
|
|
45
47
|
}
|
|
46
48
|
// ── Flow channel (fires after stage execution) ────────────────────────
|
|
47
49
|
onStageExecuted(event) {
|
|
48
|
-
var _a;
|
|
49
|
-
|
|
50
|
-
const
|
|
50
|
+
var _a, _b, _c;
|
|
51
|
+
const sfKey = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.subflowId) !== null && _b !== void 0 ? _b : '';
|
|
52
|
+
const stageNum = this.incrementStageCounter(sfKey);
|
|
53
|
+
const isFirst = this.consumeFirstStageFlag(sfKey);
|
|
54
|
+
const text = isFirst
|
|
51
55
|
? event.description
|
|
52
56
|
? `The process began: ${event.description}.`
|
|
53
57
|
: `The process began with ${event.stageName}.`
|
|
54
58
|
: event.description
|
|
55
59
|
? `Next step: ${event.description}.`
|
|
56
60
|
: `Next, it moved on to ${event.stageName}.`;
|
|
57
|
-
|
|
58
|
-
const sfId = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.subflowId;
|
|
61
|
+
const sfId = (_c = event.traversalContext) === null || _c === void 0 ? void 0 : _c.subflowId;
|
|
59
62
|
this.entries.push({
|
|
60
63
|
type: 'stage',
|
|
61
|
-
text: `Stage ${
|
|
64
|
+
text: `Stage ${stageNum}: ${text}`,
|
|
62
65
|
depth: 0,
|
|
63
66
|
stageName: event.stageName,
|
|
64
67
|
subflowId: sfId,
|
|
@@ -66,25 +69,26 @@ export class CombinedNarrativeRecorder {
|
|
|
66
69
|
this.flushOps(event.stageName, sfId);
|
|
67
70
|
}
|
|
68
71
|
onDecision(event) {
|
|
69
|
-
var _a, _b;
|
|
72
|
+
var _a, _b, _c, _d, _e;
|
|
70
73
|
// Emit the decider stage entry (deciders don't fire onStageExecuted)
|
|
71
|
-
|
|
72
|
-
const
|
|
74
|
+
const sfKey = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.subflowId) !== null && _b !== void 0 ? _b : '';
|
|
75
|
+
const stageNum = this.incrementStageCounter(sfKey);
|
|
76
|
+
const isFirst = this.consumeFirstStageFlag(sfKey);
|
|
77
|
+
const stageText = isFirst
|
|
73
78
|
? event.description
|
|
74
79
|
? `The process began: ${event.description}.`
|
|
75
80
|
: `The process began with ${event.decider}.`
|
|
76
81
|
: event.description
|
|
77
82
|
? `Next step: ${event.description}.`
|
|
78
83
|
: `Next, it moved on to ${event.decider}.`;
|
|
79
|
-
this.isFirstStage = false;
|
|
80
84
|
this.entries.push({
|
|
81
85
|
type: 'stage',
|
|
82
|
-
text: `Stage ${
|
|
86
|
+
text: `Stage ${stageNum}: ${stageText}`,
|
|
83
87
|
depth: 0,
|
|
84
88
|
stageName: event.decider,
|
|
85
|
-
subflowId: (
|
|
89
|
+
subflowId: (_c = event.traversalContext) === null || _c === void 0 ? void 0 : _c.subflowId,
|
|
86
90
|
});
|
|
87
|
-
this.flushOps(event.decider);
|
|
91
|
+
this.flushOps(event.decider, (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.subflowId);
|
|
88
92
|
// Emit the condition entry
|
|
89
93
|
const branchName = event.chosen;
|
|
90
94
|
let conditionText;
|
|
@@ -105,7 +109,7 @@ export class CombinedNarrativeRecorder {
|
|
|
105
109
|
text: `[Condition]: ${conditionText}`,
|
|
106
110
|
depth: 0,
|
|
107
111
|
stageName: event.decider,
|
|
108
|
-
subflowId: (
|
|
112
|
+
subflowId: (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.subflowId,
|
|
109
113
|
});
|
|
110
114
|
}
|
|
111
115
|
onNext() {
|
|
@@ -133,11 +137,15 @@ export class CombinedNarrativeRecorder {
|
|
|
133
137
|
});
|
|
134
138
|
}
|
|
135
139
|
onSubflowEntry(event) {
|
|
136
|
-
var _a;
|
|
140
|
+
var _a, _b;
|
|
141
|
+
// Reset stage counter for this subflow so stages start at "Stage 1" on re-entry
|
|
142
|
+
const sfKey = (_a = event.subflowId) !== null && _a !== void 0 ? _a : '';
|
|
143
|
+
this.stageCounters.delete(sfKey);
|
|
144
|
+
this.firstStageFlags.delete(sfKey);
|
|
137
145
|
const text = event.description
|
|
138
146
|
? `Entering the ${event.name} subflow: ${event.description}.`
|
|
139
147
|
: `Entering the ${event.name} subflow.`;
|
|
140
|
-
this.entries.push({ type: 'subflow', text, depth: 0, subflowId: (
|
|
148
|
+
this.entries.push({ type: 'subflow', text, depth: 0, subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId });
|
|
141
149
|
}
|
|
142
150
|
onSubflowExit(event) {
|
|
143
151
|
var _a;
|
|
@@ -222,10 +230,26 @@ export class CombinedNarrativeRecorder {
|
|
|
222
230
|
clear() {
|
|
223
231
|
this.entries = [];
|
|
224
232
|
this.pendingOps.clear();
|
|
225
|
-
this.
|
|
226
|
-
this.
|
|
233
|
+
this.stageCounters.clear();
|
|
234
|
+
this.firstStageFlags.clear();
|
|
227
235
|
}
|
|
228
236
|
// ── Private helpers ───────────────────────────────────────────────────
|
|
237
|
+
/** Increment and return the stage counter for a given subflow ('' = root). */
|
|
238
|
+
incrementStageCounter(subflowKey) {
|
|
239
|
+
var _a;
|
|
240
|
+
const current = (_a = this.stageCounters.get(subflowKey)) !== null && _a !== void 0 ? _a : 0;
|
|
241
|
+
const next = current + 1;
|
|
242
|
+
this.stageCounters.set(subflowKey, next);
|
|
243
|
+
return next;
|
|
244
|
+
}
|
|
245
|
+
/** Returns true if this is the first stage for the given subflow, consuming the flag. */
|
|
246
|
+
consumeFirstStageFlag(subflowKey) {
|
|
247
|
+
if (!this.firstStageFlags.has(subflowKey)) {
|
|
248
|
+
this.firstStageFlags.set(subflowKey, false);
|
|
249
|
+
return true;
|
|
250
|
+
}
|
|
251
|
+
return false;
|
|
252
|
+
}
|
|
229
253
|
bufferOp(stageName, op) {
|
|
230
254
|
let ops = this.pendingOps.get(stageName);
|
|
231
255
|
if (!ops) {
|
|
@@ -297,4 +321,4 @@ function summarizeValue(value, maxLen) {
|
|
|
297
321
|
}
|
|
298
322
|
return String(value);
|
|
299
323
|
}
|
|
300
|
-
//# 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  private stageCounter = 0;\n  private isFirstStage = true;\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    this.stageCounter++;\n    const text = this.isFirstStage\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    this.isFirstStage = false;\n\n    const sfId = event.traversalContext?.subflowId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${this.stageCounter}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    this.stageCounter++;\n    const stageText = this.isFirstStage\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    this.isFirstStage = false;\n\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${this.stageCounter}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider);\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      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      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      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    const text = event.description\n      ? `Entering the ${event.name} subflow: ${event.description}.`\n      : `Entering the ${event.name} subflow.`;\n    this.entries.push({ type: 'subflow', text, depth: 0, subflowId: event.traversalContext?.subflowId });\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      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({ type: 'loop', text, depth: 0, subflowId: event.traversalContext?.subflowId });\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      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      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.stageCounter = 0;\n    this.isFirstStage = true;\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\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): 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        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"]}
|
|
324
|
+
//# 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    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId);\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    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider, event.traversalContext?.subflowId);\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      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      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      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({ type: 'subflow', text, depth: 0, subflowId: event.traversalContext?.subflowId });\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      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({ type: 'loop', text, depth: 0, subflowId: event.traversalContext?.subflowId });\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      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      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): 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        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"]}
|
|
@@ -77,6 +77,9 @@ export class ManifestFlowRecorder {
|
|
|
77
77
|
getSpecIds() {
|
|
78
78
|
return Array.from(this.specs.keys());
|
|
79
79
|
}
|
|
80
|
+
toSnapshot() {
|
|
81
|
+
return { name: 'Manifest', data: this.getManifest() };
|
|
82
|
+
}
|
|
80
83
|
/** Clears state for reuse. */
|
|
81
84
|
clear() {
|
|
82
85
|
this.stack = [];
|
|
@@ -84,4 +87,4 @@ export class ManifestFlowRecorder {
|
|
|
84
87
|
this.specs.clear();
|
|
85
88
|
}
|
|
86
89
|
}
|
|
87
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTWFuaWZlc3RGbG93UmVjb3JkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2VuZ2luZS9uYXJyYXRpdmUvcmVjb3JkZXJzL01hbmlmZXN0Rmxvd1JlY29yZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVCRztBQWdCSCxNQUFNLE9BQU8sb0JBQW9CO0lBVS9CLFlBQVksRUFBVztRQVB2QixvRUFBb0U7UUFDNUQsVUFBSyxHQUFvQixFQUFFLENBQUM7UUFDcEMsK0RBQStEO1FBQ3ZELFVBQUssR0FBb0IsRUFBRSxDQUFDO1FBQ3BDLDBEQUEwRDtRQUNsRCxVQUFLLEdBQUcsSUFBSSxHQUFHLEVBQW1CLENBQUM7UUFHekMsSUFBSSxDQUFDLEVBQUUsR0FBRyxFQUFFLGFBQUYsRUFBRSxjQUFGLEVBQUUsR0FBSSxVQUFVLENBQUM7SUFDN0IsQ0FBQztJQUVELGNBQWMsQ0FBQyxLQUF1Qjs7UUFDcEMsTUFBTSxLQUFLLEdBQWtCO1lBQzNCLFNBQVMsRUFBRSxNQUFBLEtBQUssQ0FBQyxTQUFTLG1DQUFJLEtBQUssQ0FBQyxJQUFJO1lBQ3hDLElBQUksRUFBRSxLQUFLLENBQUMsSUFBSTtZQUNoQixXQUFXLEVBQUUsS0FBSyxDQUFDLFdBQVc7WUFDOUIsUUFBUSxFQUFFLEVBQUU7U0FDYixDQUFDO1FBQ0YsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDekIsQ0FBQztJQUVELGFBQWEsQ0FBQyxNQUF3QjtRQUNwQyxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsRUFBRSxDQUFDO1FBQ25DLElBQUksQ0FBQyxTQUFTO1lBQUUsT0FBTztRQUV2QixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQyxDQUFDO1FBQ2pELElBQUksTUFBTSxFQUFFLENBQUM7WUFDWCxNQUFNLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUNsQyxDQUFDO2FBQU0sQ0FBQztZQUNOLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQzdCLENBQUM7SUFDSCxDQUFDO0lBRUQsbUJBQW1CLENBQUMsS0FBaUM7UUFDbkQsSUFBSSxLQUFLLENBQUMsYUFBYSxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUM7WUFDNUQsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLFNBQVMsRUFBRSxLQUFLLENBQUMsYUFBYSxDQUFDLENBQUM7UUFDdkQsQ0FBQztJQUNILENBQUM7SUFFRCxnRkFBZ0Y7SUFDaEYsV0FBVztRQUNULE9BQU8sQ0FBQyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUN6QixDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0gsT0FBTyxDQUFDLFNBQWlCO1FBQ3ZCLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDbkMsQ0FBQztJQUVELG1DQUFtQztJQUNuQyxVQUFVO1FBQ1IsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUN2QyxDQUFDO0lBRUQsVUFBVTtRQUNSLE9BQU8sRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFLElBQUksRUFBRSxJQUFJLENBQUMsV0FBVyxFQUFFLEVBQUUsQ0FBQztJQUN4RCxDQUFDO0lBRUQsOEJBQThCO0lBQzlCLEtBQUs7UUFDSCxJQUFJLENBQUMsS0FBSyxHQUFHLEVBQUUsQ0FBQztRQUNoQixJQUFJLENBQUMsS0FBSyxHQUFHLEVBQUUsQ0FBQztRQUNoQixJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ3JCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogTWFuaWZlc3RGbG93UmVjb3JkZXIg4oCUIEJ1aWxkcyBhIGxpZ2h0d2VpZ2h0IHN1YmZsb3cgbWFuaWZlc3QgZHVyaW5nIHRyYXZlcnNhbC5cbiAqXG4gKiBDb2xsZWN0cyBzdWJmbG93IG1ldGFkYXRhIChJRCwgbmFtZSwgZGVzY3JpcHRpb24pIGFzIGEgc2lkZSBlZmZlY3Qgb2ZcbiAqIG9ic2VydmluZyB0cmF2ZXJzYWwgZXZlbnRzLiBQcm9kdWNlcyBhIHRyZWUgc3RydWN0dXJlIHN1aXRhYmxlIGZvciBMTE1cbiAqIG5hdmlnYXRpb246IGxpZ2h0d2VpZ2h0IGVub3VnaCB0byBpbmNsdWRlIGluIHNuYXBzaG90cywgd2l0aCBvbi1kZW1hbmRcbiAqIGFjY2VzcyB0byBmdWxsIHNwZWNzIHZpYSBnZXRTcGVjKCkuXG4gKlxuICogVGhlIG1hbmlmZXN0IHJlZmxlY3RzIG9ubHkgc3ViZmxvd3MgdGhhdCB3ZXJlIGFjdHVhbGx5IGVudGVyZWQgZHVyaW5nXG4gKiBleGVjdXRpb24g4oCUIHVudmlzaXRlZCBicmFuY2hlcyBhcmUgbm90IGluY2x1ZGVkLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjb25zdCBtYW5pZmVzdCA9IG5ldyBNYW5pZmVzdEZsb3dSZWNvcmRlcigpO1xuICogZXhlY3V0b3IuYXR0YWNoRmxvd1JlY29yZGVyKG1hbmlmZXN0KTtcbiAqIGF3YWl0IGV4ZWN1dG9yLnJ1bih7IGlucHV0OiBkYXRhIH0pO1xuICpcbiAqIC8vIExpZ2h0d2VpZ2h0IHRyZWUgb2Ygc3ViZmxvdyBJRHMgKyBkZXNjcmlwdGlvbnNcbiAqIGNvbnN0IHRyZWUgPSBtYW5pZmVzdC5nZXRNYW5pZmVzdCgpO1xuICpcbiAqIC8vIEZ1bGwgc3BlYyBmb3IgYSBzcGVjaWZpYyBzdWJmbG93IChpZiBhdmFpbGFibGUpXG4gKiBjb25zdCBzcGVjID0gbWFuaWZlc3QuZ2V0U3BlYygnc2YtY3JlZGl0LWNoZWNrJyk7XG4gKiBgYGBcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IEZsb3dSZWNvcmRlciwgRmxvd1N1YmZsb3dFdmVudCwgRmxvd1N1YmZsb3dSZWdpc3RlcmVkRXZlbnQgfSBmcm9tICcuLi90eXBlcy5qcyc7XG5cbi8qKiBBIHNpbmdsZSBlbnRyeSBpbiB0aGUgc3ViZmxvdyBtYW5pZmVzdCB0cmVlLiAqL1xuZXhwb3J0IGludGVyZmFjZSBNYW5pZmVzdEVudHJ5IHtcbiAgLyoqIFN1YmZsb3cgaWRlbnRpZmllciDigJQgdXNlIGZvciBvbi1kZW1hbmQgc3BlYyBsb29rdXAuICovXG4gIHN1YmZsb3dJZDogc3RyaW5nO1xuICAvKiogSHVtYW4tcmVhZGFibGUgbmFtZS4gKi9cbiAgbmFtZTogc3RyaW5nO1xuICAvKiogQnVpbGQtdGltZSBkZXNjcmlwdGlvbiBvZiB3aGF0IHRoaXMgc3ViZmxvdyBkb2VzLiAqL1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgLyoqIE5lc3RlZCBzdWJmbG93cyBlbnRlcmVkIHdpdGhpbiB0aGlzIHN1YmZsb3cuICovXG4gIGNoaWxkcmVuOiBNYW5pZmVzdEVudHJ5W107XG59XG5cbmV4cG9ydCBjbGFzcyBNYW5pZmVzdEZsb3dSZWNvcmRlciBpbXBsZW1lbnRzIEZsb3dSZWNvcmRlciB7XG4gIHJlYWRvbmx5IGlkOiBzdHJpbmc7XG5cbiAgLyoqIFN0YWNrIHRyYWNrcyBuZXN0aW5nIGRlcHRoIOKAlCBjdXJyZW50IHN1YmZsb3cgaXMgdG9wIG9mIHN0YWNrLiAqL1xuICBwcml2YXRlIHN0YWNrOiBNYW5pZmVzdEVudHJ5W10gPSBbXTtcbiAgLyoqIFJvb3QtbGV2ZWwgc3ViZmxvd3MgKG5vdCBuZXN0ZWQgaW5zaWRlIGFub3RoZXIgc3ViZmxvdykuICovXG4gIHByaXZhdGUgcm9vdHM6IE1hbmlmZXN0RW50cnlbXSA9IFtdO1xuICAvKiogRnVsbCBzcGVjcyBzdG9yZWQgZnJvbSBkeW5hbWljIHJlZ2lzdHJhdGlvbiBldmVudHMuICovXG4gIHByaXZhdGUgc3BlY3MgPSBuZXcgTWFwPHN0cmluZywgdW5rbm93bj4oKTtcblxuICBjb25zdHJ1Y3RvcihpZD86IHN0cmluZykge1xuICAgIHRoaXMuaWQgPSBpZCA/PyAnbWFuaWZlc3QnO1xuICB9XG5cbiAgb25TdWJmbG93RW50cnkoZXZlbnQ6IEZsb3dTdWJmbG93RXZlbnQpOiB2b2lkIHtcbiAgICBjb25zdCBlbnRyeTogTWFuaWZlc3RFbnRyeSA9IHtcbiAgICAgIHN1YmZsb3dJZDogZXZlbnQuc3ViZmxvd0lkID8/IGV2ZW50Lm5hbWUsXG4gICAgICBuYW1lOiBldmVudC5uYW1lLFxuICAgICAgZGVzY3JpcHRpb246IGV2ZW50LmRlc2NyaXB0aW9uLFxuICAgICAgY2hpbGRyZW46IFtdLFxuICAgIH07XG4gICAgdGhpcy5zdGFjay5wdXNoKGVudHJ5KTtcbiAgfVxuXG4gIG9uU3ViZmxvd0V4aXQoX2V2ZW50OiBGbG93U3ViZmxvd0V2ZW50KTogdm9pZCB7XG4gICAgY29uc3QgY29tcGxldGVkID0gdGhpcy5zdGFjay5wb3AoKTtcbiAgICBpZiAoIWNvbXBsZXRlZCkgcmV0dXJuO1xuXG4gICAgY29uc3QgcGFyZW50ID0gdGhpcy5zdGFja1t0aGlzLnN0YWNrLmxlbmd0aCAtIDFdO1xuICAgIGlmIChwYXJlbnQpIHtcbiAgICAgIHBhcmVudC5jaGlsZHJlbi5wdXNoKGNvbXBsZXRlZCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHRoaXMucm9vdHMucHVzaChjb21wbGV0ZWQpO1xuICAgIH1cbiAgfVxuXG4gIG9uU3ViZmxvd1JlZ2lzdGVyZWQoZXZlbnQ6IEZsb3dTdWJmbG93UmVnaXN0ZXJlZEV2ZW50KTogdm9pZCB7XG4gICAgaWYgKGV2ZW50LnNwZWNTdHJ1Y3R1cmUgJiYgIXRoaXMuc3BlY3MuaGFzKGV2ZW50LnN1YmZsb3dJZCkpIHtcbiAgICAgIHRoaXMuc3BlY3Muc2V0KGV2ZW50LnN1YmZsb3dJZCwgZXZlbnQuc3BlY1N0cnVjdHVyZSk7XG4gICAgfVxuICB9XG5cbiAgLyoqIFJldHVybnMgdGhlIG1hbmlmZXN0IHRyZWUg4oCUIGxpZ2h0d2VpZ2h0LCBzdWl0YWJsZSBmb3Igc25hcHNob3QgaW5jbHVzaW9uLiAqL1xuICBnZXRNYW5pZmVzdCgpOiBNYW5pZmVzdEVudHJ5W10ge1xuICAgIHJldHVybiBbLi4udGhpcy5yb290c107XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyB0aGUgZnVsbCBzcGVjIGZvciBhIGR5bmFtaWNhbGx5LXJlZ2lzdGVyZWQgc3ViZmxvdy5cbiAgICogT25seSBwb3B1bGF0ZWQgZm9yIHN1YmZsb3dzIGF1dG8tcmVnaXN0ZXJlZCBhdCBydW50aW1lICh2aWEgU3RhZ2VOb2RlXG4gICAqIHJldHVybiB3aXRoIHN1YmZsb3dEZWYpLiBTdGF0aWNhbGx5LWNvbmZpZ3VyZWQgc3ViZmxvd3MgYXJlIG5vdCBpbmNsdWRlZFxuICAgKiBldmVuIGlmIHRoZXkgYXBwZWFyIGluIGdldE1hbmlmZXN0KCkuIFVzZSBGbG93Q2hhcnQuYnVpbGRUaW1lU3RydWN0dXJlXG4gICAqIHRvIGFjY2VzcyBzdGF0aWNhbGx5LWRlZmluZWQgc3ViZmxvdyBzcGVjcy5cbiAgICovXG4gIGdldFNwZWMoc3ViZmxvd0lkOiBzdHJpbmcpOiB1bmtub3duIHwgdW5kZWZpbmVkIHtcbiAgICByZXR1cm4gdGhpcy5zcGVjcy5nZXQoc3ViZmxvd0lkKTtcbiAgfVxuXG4gIC8qKiBSZXR1cm5zIGFsbCBzdG9yZWQgc3BlYyBJRHMuICovXG4gIGdldFNwZWNJZHMoKTogc3RyaW5nW10ge1xuICAgIHJldHVybiBBcnJheS5mcm9tKHRoaXMuc3BlY3Mua2V5cygpKTtcbiAgfVxuXG4gIHRvU25hcHNob3QoKTogeyBuYW1lOiBzdHJpbmc7IGRhdGE6IHVua25vd24gfSB7XG4gICAgcmV0dXJuIHsgbmFtZTogJ01hbmlmZXN0JywgZGF0YTogdGhpcy5nZXRNYW5pZmVzdCgpIH07XG4gIH1cblxuICAvKiogQ2xlYXJzIHN0YXRlIGZvciByZXVzZS4gKi9cbiAgY2xlYXIoKTogdm9pZCB7XG4gICAgdGhpcy5zdGFjayA9IFtdO1xuICAgIHRoaXMucm9vdHMgPSBbXTtcbiAgICB0aGlzLnNwZWNzLmNsZWFyKCk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -8,4 +8,4 @@
|
|
|
8
8
|
* interface with empty methods for zero-cost disabled path.
|
|
9
9
|
*/
|
|
10
10
|
export {};
|
|
11
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG","sourcesContent":["/**\n * IControlFlowNarrative — Interface for control flow narrative generation.\n *\n * Captures FLOW events during traversal: decisions, forks, loops, subflows.\n * Complementary to scope/recorders/NarrativeRecorder which captures DATA events.\n *\n * Uses Null Object pattern: NullControlFlowNarrativeGenerator satisfies this\n * interface with empty methods for zero-cost disabled path.\n */\n\nexport interface IControlFlowNarrative {\n  /** Called when a stage executes. First stage uses distinct opening pattern. */\n  onStageExecuted(stageName: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called on linear continuation from one stage to the next. */\n  onNext(fromStage: string, toStage: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a decider selects a branch. Most valuable for LLM context. */\n  onDecision(\n    deciderName: string,\n    chosenBranch: string,\n    rationale?: string,\n    deciderDescription?: string,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when a fork executes all children in parallel. */\n  onFork(parentStage: string, childNames: string[], traversalContext?: TraversalContext): void;\n\n  /** Called when a selector picks a subset of children. */\n  onSelected(\n    parentStage: string,\n    selectedNames: string[],\n    totalCount: number,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when entering a subflow (nested context boundary). */\n  onSubflowEntry(\n    subflowName: string,\n    subflowId?: string,\n    description?: string,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when exiting a subflow. */\n  onSubflowExit(subflowName: string, subflowId?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered(subflowId: string, name: string, description?: string, specStructure?: unknown): void;\n\n  /** Called on loop iteration (back-edge traversal). */\n  onLoop(targetStage: string, iteration: number, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage triggers break (early termination). */\n  onBreak(stageName: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage throws an error. Raw error is extracted into structured details. */\n  onError(stageName: string, errorMessage: string, error: unknown, traversalContext?: TraversalContext): void;\n\n  /** Returns accumulated narrative sentences in execution order. */\n  getSentences(): string[];\n}\n\n// ============================================================================\n// TraversalContext — read-only execution context from the traverser.\n// ============================================================================\n\n/**\n * Traversal context attached to every FlowRecorder event.\n * Created by the traverser during DFS, passed to recorders as read-only data.\n * Enables recorders to build trees, group by subflow, and correlate events\n * without maintaining their own stacks or post-processing.\n *\n * Like OpenTelemetry's span context: stageId + parentStageId form a tree.\n */\nexport interface TraversalContext {\n  /** Stable stage identifier from the builder (matches spec node id). */\n  readonly stageId: string;\n  /** Human-readable stage name. */\n  readonly stageName: string;\n  /** Parent stage ID — walk up to reconstruct the tree. Undefined at root. */\n  readonly parentStageId?: string;\n  /** Subflow ID when inside a subflow. Undefined at root level. */\n  readonly subflowId?: string;\n  /** Full subflow path for nested subflows (e.g., \"sf-outer/sf-inner\"). */\n  readonly subflowPath?: string;\n  /** Nesting depth (0 = root, 1 = inside first subflow, etc.). */\n  readonly depth: number;\n  /** Loop iteration number when revisiting a node via loopTo. */\n  readonly loopIteration?: number;\n  /** Fork branch ID when inside a parallel or decider branch. */\n  readonly forkBranch?: string;\n}\n\n// ============================================================================\n// FlowRecorder — Pluggable observer for control flow events.\n// ============================================================================\n\n/** Event passed to FlowRecorder.onStageExecuted. */\nexport interface FlowStageEvent {\n  stageName: string;\n  description?: string;\n  /** Traversal context from the engine — read-only, set by traverser. */\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onNext. */\nexport interface FlowNextEvent {\n  from: string;\n  to: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onDecision. */\nexport interface FlowDecisionEvent {\n  decider: string;\n  chosen: string;\n  rationale?: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onFork. */\nexport interface FlowForkEvent {\n  parent: string;\n  children: string[];\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSelected. */\nexport interface FlowSelectedEvent {\n  parent: string;\n  selected: string[];\n  total: number;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSubflow. */\nexport interface FlowSubflowEvent {\n  name: string;\n  /** Subflow identifier — use this to look up the full spec via the manifest. */\n  subflowId?: string;\n  /** Build-time description of what this subflow does. */\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSubflowRegistered (dynamic subflow attachment). */\nexport interface FlowSubflowRegisteredEvent {\n  /** Subflow identifier. */\n  subflowId: string;\n  /** Human-readable name. */\n  name: string;\n  /** Build-time description. */\n  description?: string;\n  /** Full spec structure (when available from buildTimeStructure). */\n  specStructure?: unknown;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onLoop. */\nexport interface FlowLoopEvent {\n  target: string;\n  iteration: number;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onBreak. */\nexport interface FlowBreakEvent {\n  stageName: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onError. */\nexport interface FlowErrorEvent {\n  stageName: string;\n  message: string;\n  /** Structured error details — preserves field-level issues, error codes, etc. */\n  structuredError: import('../errors/errorInfo').StructuredErrorInfo;\n  traversalContext?: TraversalContext;\n}\n\n/**\n * FlowRecorder — Pluggable observer for control flow events.\n *\n * Mirrors the scope-level Recorder pattern for the engine layer.\n * All methods are optional — implement only the hooks you need.\n * Recorders are invoked synchronously in attachment order.\n * If a recorder throws, the error is caught and swallowed; execution continues.\n *\n * @example\n * ```typescript\n * const metricsRecorder: FlowRecorder = {\n *   id: 'metrics',\n *   onLoop: (event) => recordMetric('loop.iteration', event.iteration),\n *   onDecision: (event) => recordMetric('decision', event.chosen),\n * };\n * executor.attachFlowRecorder(metricsRecorder);\n * ```\n */\nexport interface FlowRecorder {\n  readonly id: string;\n  onStageExecuted?(event: FlowStageEvent): void;\n  onNext?(event: FlowNextEvent): void;\n  onDecision?(event: FlowDecisionEvent): void;\n  onFork?(event: FlowForkEvent): void;\n  onSelected?(event: FlowSelectedEvent): void;\n  onSubflowEntry?(event: FlowSubflowEvent): void;\n  onSubflowExit?(event: FlowSubflowEvent): void;\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered?(event: FlowSubflowRegisteredEvent): void;\n  onLoop?(event: FlowLoopEvent): void;\n  onBreak?(event: FlowBreakEvent): void;\n  onError?(event: FlowErrorEvent): void;\n  /** Called before each run to reset per-run state. Implement for stateful recorders. */\n  clear?(): void;\n}\n"]}
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/types.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG","sourcesContent":["/**\n * IControlFlowNarrative — Interface for control flow narrative generation.\n *\n * Captures FLOW events during traversal: decisions, forks, loops, subflows.\n * Complementary to scope/recorders/NarrativeRecorder which captures DATA events.\n *\n * Uses Null Object pattern: NullControlFlowNarrativeGenerator satisfies this\n * interface with empty methods for zero-cost disabled path.\n */\n\nexport interface IControlFlowNarrative {\n  /** Called when a stage executes. First stage uses distinct opening pattern. */\n  onStageExecuted(stageName: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called on linear continuation from one stage to the next. */\n  onNext(fromStage: string, toStage: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a decider selects a branch. Most valuable for LLM context. */\n  onDecision(\n    deciderName: string,\n    chosenBranch: string,\n    rationale?: string,\n    deciderDescription?: string,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when a fork executes all children in parallel. */\n  onFork(parentStage: string, childNames: string[], traversalContext?: TraversalContext): void;\n\n  /** Called when a selector picks a subset of children. */\n  onSelected(\n    parentStage: string,\n    selectedNames: string[],\n    totalCount: number,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when entering a subflow (nested context boundary). */\n  onSubflowEntry(\n    subflowName: string,\n    subflowId?: string,\n    description?: string,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when exiting a subflow. */\n  onSubflowExit(subflowName: string, subflowId?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered(subflowId: string, name: string, description?: string, specStructure?: unknown): void;\n\n  /** Called on loop iteration (back-edge traversal). */\n  onLoop(targetStage: string, iteration: number, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage triggers break (early termination). */\n  onBreak(stageName: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage throws an error. Raw error is extracted into structured details. */\n  onError(stageName: string, errorMessage: string, error: unknown, traversalContext?: TraversalContext): void;\n\n  /** Returns accumulated narrative sentences in execution order. */\n  getSentences(): string[];\n}\n\n// ============================================================================\n// TraversalContext — read-only execution context from the traverser.\n// ============================================================================\n\n/**\n * Traversal context attached to every FlowRecorder event.\n * Created by the traverser during DFS, passed to recorders as read-only data.\n * Enables recorders to build trees, group by subflow, and correlate events\n * without maintaining their own stacks or post-processing.\n *\n * Like OpenTelemetry's span context: stageId + parentStageId form a tree.\n */\nexport interface TraversalContext {\n  /** Stable stage identifier from the builder (matches spec node id). */\n  readonly stageId: string;\n  /** Human-readable stage name. */\n  readonly stageName: string;\n  /** Parent stage ID — walk up to reconstruct the tree. Undefined at root. */\n  readonly parentStageId?: string;\n  /** Subflow ID when inside a subflow. Undefined at root level. */\n  readonly subflowId?: string;\n  /** Full subflow path for nested subflows (e.g., \"sf-outer/sf-inner\"). */\n  readonly subflowPath?: string;\n  /** Nesting depth (0 = root, 1 = inside first subflow, etc.). */\n  readonly depth: number;\n  /** Loop iteration number when revisiting a node via loopTo. */\n  readonly loopIteration?: number;\n  /** Fork branch ID when inside a parallel or decider branch. */\n  readonly forkBranch?: string;\n}\n\n// ============================================================================\n// FlowRecorder — Pluggable observer for control flow events.\n// ============================================================================\n\n/** Event passed to FlowRecorder.onStageExecuted. */\nexport interface FlowStageEvent {\n  stageName: string;\n  description?: string;\n  /** Traversal context from the engine — read-only, set by traverser. */\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onNext. */\nexport interface FlowNextEvent {\n  from: string;\n  to: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onDecision. */\nexport interface FlowDecisionEvent {\n  decider: string;\n  chosen: string;\n  rationale?: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onFork. */\nexport interface FlowForkEvent {\n  parent: string;\n  children: string[];\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSelected. */\nexport interface FlowSelectedEvent {\n  parent: string;\n  selected: string[];\n  total: number;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSubflow. */\nexport interface FlowSubflowEvent {\n  name: string;\n  /** Subflow identifier — use this to look up the full spec via the manifest. */\n  subflowId?: string;\n  /** Build-time description of what this subflow does. */\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSubflowRegistered (dynamic subflow attachment). */\nexport interface FlowSubflowRegisteredEvent {\n  /** Subflow identifier. */\n  subflowId: string;\n  /** Human-readable name. */\n  name: string;\n  /** Build-time description. */\n  description?: string;\n  /** Full spec structure (when available from buildTimeStructure). */\n  specStructure?: unknown;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onLoop. */\nexport interface FlowLoopEvent {\n  target: string;\n  iteration: number;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onBreak. */\nexport interface FlowBreakEvent {\n  stageName: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onError. */\nexport interface FlowErrorEvent {\n  stageName: string;\n  message: string;\n  /** Structured error details — preserves field-level issues, error codes, etc. */\n  structuredError: import('../errors/errorInfo').StructuredErrorInfo;\n  traversalContext?: TraversalContext;\n}\n\n/**\n * FlowRecorder — Pluggable observer for control flow events.\n *\n * Mirrors the scope-level Recorder pattern for the engine layer.\n * All methods are optional — implement only the hooks you need.\n * Recorders are invoked synchronously in attachment order.\n * If a recorder throws, the error is caught and swallowed; execution continues.\n *\n * @example\n * ```typescript\n * const metricsRecorder: FlowRecorder = {\n *   id: 'metrics',\n *   onLoop: (event) => recordMetric('loop.iteration', event.iteration),\n *   onDecision: (event) => recordMetric('decision', event.chosen),\n * };\n * executor.attachFlowRecorder(metricsRecorder);\n * ```\n */\nexport interface FlowRecorder {\n  readonly id: string;\n  onStageExecuted?(event: FlowStageEvent): void;\n  onNext?(event: FlowNextEvent): void;\n  onDecision?(event: FlowDecisionEvent): void;\n  onFork?(event: FlowForkEvent): void;\n  onSelected?(event: FlowSelectedEvent): void;\n  onSubflowEntry?(event: FlowSubflowEvent): void;\n  onSubflowExit?(event: FlowSubflowEvent): void;\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered?(event: FlowSubflowRegisteredEvent): void;\n  onLoop?(event: FlowLoopEvent): void;\n  onBreak?(event: FlowBreakEvent): void;\n  onError?(event: FlowErrorEvent): void;\n  /** Called before each run to reset per-run state. Implement for stateful recorders. */\n  clear?(): void;\n  /** Optional: expose collected data for inclusion in snapshots. */\n  toSnapshot?(): { name: string; data: unknown };\n}\n"]}
|
|
@@ -13,4 +13,4 @@ export const defaultLogger = {
|
|
|
13
13
|
error: (message, ...args) => console.error(message, ...args),
|
|
14
14
|
warn: (message, ...args) => console.warn(message, ...args),
|
|
15
15
|
};
|
|
16
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../src/lib/engine/types.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAuBH,oCAAoC;AACpC,wDAAwD;AACxD,MAAM,CAAC,MAAM,aAAa,GAAY;IACpC,IAAI,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACvE,GAAG,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACrE,KAAK,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACzE,KAAK,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACzE,IAAI,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;CACxE,CAAC","sourcesContent":["/**\n * types.ts — All type definitions for the engine library.\n *\n * Centralizes type definitions to avoid circular dependencies.\n * Every handler receives HandlerDeps (the DI bag) instead of importing the traverser.\n */\n\nimport type { StageContext } from '../memory/StageContext.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport type { Decider, Selector, StageNode } from './graph/StageNode.js';\nimport type { IControlFlowNarrative } from './narrative/types.js';\n\n// Re-export StageNode types for convenience\nexport type { Decider, Selector, StageNode } from './graph/StageNode.js';\n\n// ---------------------------------------------------------------------------\n// Logger\n// ---------------------------------------------------------------------------\n\n/** Minimal logging contract. Mirrors Console API subset. */\nexport interface ILogger {\n  info(message?: any, ...optionalParams: any[]): void;\n  log(message?: any, ...optionalParams: any[]): void;\n  debug(message?: any, ...optionalParams: any[]): void;\n  error(message?: any, ...optionalParams: any[]): void;\n  warn(message?: any, ...optionalParams: any[]): void;\n}\n\n/** Default console-based logger. */\n/* istanbul ignore next -- trivial console delegation */\nexport const defaultLogger: ILogger = {\n  info: (message?: any, ...args: any[]) => console.info(message, ...args),\n  log: (message?: any, ...args: any[]) => console.log(message, ...args),\n  debug: (message?: any, ...args: any[]) => console.debug(message, ...args),\n  error: (message?: any, ...args: any[]) => console.error(message, ...args),\n  warn: (message?: any, ...args: any[]) => console.warn(message, ...args),\n};\n\n// ---------------------------------------------------------------------------\n// Stage Function\n// ---------------------------------------------------------------------------\n\n/** Callback that receives tokens during streaming. */\nexport type StreamCallback = (token: string) => void;\n\n/**\n * The function signature for stage handlers.\n * - TOut: return type produced by the stage\n * - TScope: the scope object passed to the stage\n * - Optional 3rd parameter `streamCallback` injected for streaming stages.\n */\nexport type StageFunction<TOut = any, TScope = any> = (\n  scope: TScope,\n  breakPipeline: () => void,\n  streamCallback?: StreamCallback,\n) => Promise<TOut> | TOut;\n\n/** Factory that creates a scope instance for each stage. */\nexport type ScopeFactory<TScope = any> = (\n  context: StageContext,\n  stageName: string,\n  readOnlyContext?: unknown,\n  executionEnv?: ExecutionEnv,\n) => TScope;\n\n// ---------------------------------------------------------------------------\n// Streaming\n// ---------------------------------------------------------------------------\n\nexport type StreamTokenHandler = (streamId: string, token: string) => void;\nexport type StreamLifecycleHandler = (streamId: string, fullText?: string) => void;\n\nexport interface StreamHandlers {\n  onToken?: StreamTokenHandler;\n  onStart?: StreamLifecycleHandler;\n  onEnd?: StreamLifecycleHandler;\n}\n\n// ---------------------------------------------------------------------------\n// Subflow\n// ---------------------------------------------------------------------------\n\nexport interface SubflowMountOptions<TParentScope = any, TSubflowInput = any, TSubflowOutput = any> {\n  inputMapper?: (parentScope: TParentScope) => TSubflowInput;\n  outputMapper?: (subflowOutput: TSubflowOutput, parentScope: TParentScope) => Record<string, unknown>;\n}\n\nexport interface SubflowResult {\n  subflowId: string;\n  subflowName: string;\n  treeContext: {\n    globalContext: Record<string, unknown>;\n    stageContexts: Record<string, unknown>;\n    history: unknown[];\n  };\n  parentStageId: string;\n  pipelineStructure?: unknown;\n}\n\n// ---------------------------------------------------------------------------\n// Execution Runtime Interface\n// ---------------------------------------------------------------------------\n\n/**\n * IExecutionRuntime — Interface for the runtime environment.\n *\n * Defines the contract that engine handlers need from the runner layer,\n * avoiding circular imports between engine/ and runner/.\n */\nexport interface IExecutionRuntime {\n  globalStore: { getState(): Record<string, unknown> };\n  rootStageContext: StageContext;\n  executionHistory: { list(): unknown[] };\n  getSnapshot(): {\n    sharedState: Record<string, unknown>;\n    executionTree: unknown;\n    commitLog: unknown[];\n    subflowResults?: Record<string, unknown>;\n  };\n  setRootObject(path: string[], key: string, value: unknown): void;\n  getPipelines(): string[];\n}\n\n// ---------------------------------------------------------------------------\n// Execution Environment — read-only, propagates through nested executors\n// ---------------------------------------------------------------------------\n\n/**\n * ExecutionEnv — infrastructure values that propagate through nested executors.\n *\n * Like `process.env` for flowcharts: read-only, inherited by child executors,\n * infrastructure-only (not business logic).\n *\n * Litmus test: Created external to the flowchart + passed in for execution = env.\n * Business config for a specific flowchart = args (getArgs()).\n *\n * Intentionally a closed type — not extensible to prevent coupling between\n * parent and child flowcharts.\n */\nexport interface ExecutionEnv {\n  /** AbortSignal for cooperative cancellation across nested executors. */\n  readonly signal?: AbortSignal;\n  /** Timeout budget in milliseconds. */\n  readonly timeoutMs?: number;\n  /** Trace identifier for distributed tracing / observability. */\n  readonly traceId?: string;\n}\n\n// ---------------------------------------------------------------------------\n// Handler Dependencies (DI bag) — was PipelineContext\n// ---------------------------------------------------------------------------\n\n/**\n * HandlerDeps — Dependency injection bag passed to all handler modules.\n *\n * Provides shared state (stageMap, runtime, scopeFactory, etc.) without\n * handlers needing to import the traverser directly. Avoids circular deps.\n */\nexport interface HandlerDeps<TOut = any, TScope = any> {\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  root: StageNode<TOut, TScope>;\n  executionRuntime: IExecutionRuntime;\n  ScopeFactory: ScopeFactory<TScope>;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  streamHandlers?: StreamHandlers;\n  scopeProtectionMode: ScopeProtectionMode;\n  readOnlyContext?: unknown;\n  /** Execution environment — propagates to subflows automatically. */\n  executionEnv?: ExecutionEnv;\n  narrativeGenerator: IControlFlowNarrative;\n  logger: ILogger;\n  signal?: AbortSignal;\n}\n\n/** Options for FlowChartExecutor.run(). */\nexport interface RunOptions {\n  /** AbortSignal for cooperative cancellation. */\n  signal?: AbortSignal;\n  /** Timeout in milliseconds. Creates an internal AbortController. */\n  timeoutMs?: number;\n  /**\n   * Runtime input data for the pipeline.\n   * Becomes the readOnlyContext accessible via `scope.getArgs()`.\n   * Stages cannot overwrite these keys with `setValue()`.\n   */\n  input?: Record<string, unknown>;\n  /**\n   * Execution environment — read-only infrastructure values that propagate\n   * through nested executors (like `process.env` for flowcharts).\n   * Accessible via `scope.getEnv()`. Inherited by subflows automatically.\n   */\n  env?: ExecutionEnv;\n}\n\n// ---------------------------------------------------------------------------\n// Flow Control Narrative\n// ---------------------------------------------------------------------------\n\nexport type FlowControlType = 'next' | 'branch' | 'children' | 'selected' | 'subflow' | 'loop';\n\nexport interface FlowMessage {\n  type: FlowControlType;\n  description: string;\n  targetStage?: string | string[];\n  rationale?: string;\n  count?: number;\n  iteration?: number;\n  timestamp?: number;\n}\n\n// ---------------------------------------------------------------------------\n// Traversal Extractor\n// ---------------------------------------------------------------------------\n\nexport interface RuntimeStructureMetadata {\n  type: 'stage' | 'decider' | 'fork' | 'streaming';\n  subflowId?: string;\n  isSubflowRoot?: boolean;\n  subflowName?: string;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  loopTarget?: string;\n  isDynamic?: boolean;\n  isLoopReference?: boolean;\n  streamId?: string;\n}\n\nexport interface StageSnapshot<TOut = any, TScope = any> {\n  node: StageNode<TOut, TScope>;\n  context: StageContext;\n  stepNumber: number;\n  structureMetadata: RuntimeStructureMetadata;\n  scopeState?: Record<string, unknown>;\n  debugInfo?: {\n    logs: Record<string, unknown>;\n    errors: Record<string, unknown>;\n    metrics: Record<string, unknown>;\n    evals: Record<string, unknown>;\n    flowMessages?: FlowMessage[];\n  };\n  stageOutput?: unknown;\n  errorInfo?: { type: string; message: string };\n  historyIndex?: number;\n}\n\nexport type TraversalExtractor<TResult = unknown> = (snapshot: StageSnapshot) => TResult | undefined | null;\n\nexport interface ExtractorError {\n  stagePath: string;\n  message: string;\n  error: unknown;\n}\n\n// ---------------------------------------------------------------------------\n// Node Result\n// ---------------------------------------------------------------------------\n\nexport type NodeResultType = {\n  id: string;\n  result: unknown;\n  isError?: boolean;\n};\n\n// ---------------------------------------------------------------------------\n// Execution Response\n// ---------------------------------------------------------------------------\n\nexport type BranchResult = {\n  result: string | Error;\n  isError: boolean;\n};\n\nexport type BranchResults = { [branchId: string]: BranchResult };\nexport type TraversalResult = BranchResults | string | Error;\n\n// ---------------------------------------------------------------------------\n// Serialized Pipeline Structure (for visualization)\n// ---------------------------------------------------------------------------\n\nexport interface SerializedPipelineNode {\n  name: string;\n  id?: string;\n  type?: 'stage' | 'decider' | 'fork' | 'streaming' | 'loop' | 'user' | 'tool' | 'function' | 'sequence';\n  description?: string;\n  children?: SerializedPipelineNode[];\n  next?: SerializedPipelineNode;\n  branches?: Record<string, SerializedPipelineNode>;\n  hasDecider?: boolean;\n  hasSelector?: boolean;\n  hasSubtree?: boolean;\n  isStreaming?: boolean;\n  streamId?: string;\n  isSubflowRoot?: boolean;\n  subflowId?: string;\n  subflowName?: string;\n  loopTarget?: string;\n  isLoopReference?: boolean;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  isDynamic?: boolean;\n}\n\n// ---------------------------------------------------------------------------\n// FlowChart (compiled output of FlowChartBuilder)\n// ---------------------------------------------------------------------------\n\nexport type FlowChart<TOut = any, TScope = any> = {\n  root: StageNode<TOut, TScope>;\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  extractor?: TraversalExtractor;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  enrichSnapshots?: boolean;\n  enableNarrative?: boolean;\n  logger?: ILogger;\n  buildTimeStructure?: SerializedPipelineStructure;\n  /** Input schema (Zod or JSON Schema) — used for runtime input validation. */\n  inputSchema?: unknown;\n};\n\n/** Alias for SerializedPipelineNode used as full structure */\nexport type SerializedPipelineStructure = SerializedPipelineNode & {\n  branchIds?: string[];\n  subflowStructure?: SerializedPipelineStructure;\n  iterationCount?: number;\n};\n"]}
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../src/lib/engine/types.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAuBH,oCAAoC;AACpC,wDAAwD;AACxD,MAAM,CAAC,MAAM,aAAa,GAAY;IACpC,IAAI,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACvE,GAAG,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACrE,KAAK,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACzE,KAAK,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;IACzE,IAAI,EAAE,CAAC,OAAa,EAAE,GAAG,IAAW,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;CACxE,CAAC","sourcesContent":["/**\n * types.ts — All type definitions for the engine library.\n *\n * Centralizes type definitions to avoid circular dependencies.\n * Every handler receives HandlerDeps (the DI bag) instead of importing the traverser.\n */\n\nimport type { StageContext } from '../memory/StageContext.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport type { Decider, Selector, StageNode } from './graph/StageNode.js';\nimport type { IControlFlowNarrative } from './narrative/types.js';\n\n// Re-export StageNode types for convenience\nexport type { Decider, Selector, StageNode } from './graph/StageNode.js';\n\n// ---------------------------------------------------------------------------\n// Logger\n// ---------------------------------------------------------------------------\n\n/** Minimal logging contract. Mirrors Console API subset. */\nexport interface ILogger {\n  info(message?: any, ...optionalParams: any[]): void;\n  log(message?: any, ...optionalParams: any[]): void;\n  debug(message?: any, ...optionalParams: any[]): void;\n  error(message?: any, ...optionalParams: any[]): void;\n  warn(message?: any, ...optionalParams: any[]): void;\n}\n\n/** Default console-based logger. */\n/* istanbul ignore next -- trivial console delegation */\nexport const defaultLogger: ILogger = {\n  info: (message?: any, ...args: any[]) => console.info(message, ...args),\n  log: (message?: any, ...args: any[]) => console.log(message, ...args),\n  debug: (message?: any, ...args: any[]) => console.debug(message, ...args),\n  error: (message?: any, ...args: any[]) => console.error(message, ...args),\n  warn: (message?: any, ...args: any[]) => console.warn(message, ...args),\n};\n\n// ---------------------------------------------------------------------------\n// Stage Function\n// ---------------------------------------------------------------------------\n\n/** Callback that receives tokens during streaming. */\nexport type StreamCallback = (token: string) => void;\n\n/**\n * The function signature for stage handlers.\n * - TOut: return type produced by the stage\n * - TScope: the scope object passed to the stage\n * - Optional 3rd parameter `streamCallback` injected for streaming stages.\n */\nexport type StageFunction<TOut = any, TScope = any> = (\n  scope: TScope,\n  breakPipeline: () => void,\n  streamCallback?: StreamCallback,\n) => Promise<TOut> | TOut;\n\n/** Factory that creates a scope instance for each stage. */\nexport type ScopeFactory<TScope = any> = (\n  context: StageContext,\n  stageName: string,\n  readOnlyContext?: unknown,\n  executionEnv?: ExecutionEnv,\n) => TScope;\n\n// ---------------------------------------------------------------------------\n// Streaming\n// ---------------------------------------------------------------------------\n\nexport type StreamTokenHandler = (streamId: string, token: string) => void;\nexport type StreamLifecycleHandler = (streamId: string, fullText?: string) => void;\n\nexport interface StreamHandlers {\n  onToken?: StreamTokenHandler;\n  onStart?: StreamLifecycleHandler;\n  onEnd?: StreamLifecycleHandler;\n}\n\n// ---------------------------------------------------------------------------\n// Subflow\n// ---------------------------------------------------------------------------\n\nexport interface SubflowMountOptions<TParentScope = any, TSubflowInput = any, TSubflowOutput = any> {\n  inputMapper?: (parentScope: TParentScope) => TSubflowInput;\n  outputMapper?: (subflowOutput: TSubflowOutput, parentScope: TParentScope) => Record<string, unknown>;\n}\n\nexport interface SubflowResult {\n  subflowId: string;\n  subflowName: string;\n  treeContext: {\n    globalContext: Record<string, unknown>;\n    stageContexts: Record<string, unknown>;\n    history: unknown[];\n  };\n  parentStageId: string;\n  pipelineStructure?: unknown;\n}\n\n// ---------------------------------------------------------------------------\n// Execution Runtime Interface\n// ---------------------------------------------------------------------------\n\n/**\n * IExecutionRuntime — Interface for the runtime environment.\n *\n * Defines the contract that engine handlers need from the runner layer,\n * avoiding circular imports between engine/ and runner/.\n */\nexport interface IExecutionRuntime {\n  globalStore: { getState(): Record<string, unknown> };\n  rootStageContext: StageContext;\n  executionHistory: { list(): unknown[] };\n  getSnapshot(): {\n    sharedState: Record<string, unknown>;\n    executionTree: unknown;\n    commitLog: unknown[];\n    subflowResults?: Record<string, unknown>;\n    recorders?: Array<{ id: string; name: string; data: unknown }>;\n  };\n  setRootObject(path: string[], key: string, value: unknown): void;\n  getPipelines(): string[];\n}\n\n// ---------------------------------------------------------------------------\n// Execution Environment — read-only, propagates through nested executors\n// ---------------------------------------------------------------------------\n\n/**\n * ExecutionEnv — infrastructure values that propagate through nested executors.\n *\n * Like `process.env` for flowcharts: read-only, inherited by child executors,\n * infrastructure-only (not business logic).\n *\n * Litmus test: Created external to the flowchart + passed in for execution = env.\n * Business config for a specific flowchart = args (getArgs()).\n *\n * Intentionally a closed type — not extensible to prevent coupling between\n * parent and child flowcharts.\n */\nexport interface ExecutionEnv {\n  /** AbortSignal for cooperative cancellation across nested executors. */\n  readonly signal?: AbortSignal;\n  /** Timeout budget in milliseconds. */\n  readonly timeoutMs?: number;\n  /** Trace identifier for distributed tracing / observability. */\n  readonly traceId?: string;\n}\n\n// ---------------------------------------------------------------------------\n// Handler Dependencies (DI bag) — was PipelineContext\n// ---------------------------------------------------------------------------\n\n/**\n * HandlerDeps — Dependency injection bag passed to all handler modules.\n *\n * Provides shared state (stageMap, runtime, scopeFactory, etc.) without\n * handlers needing to import the traverser directly. Avoids circular deps.\n */\nexport interface HandlerDeps<TOut = any, TScope = any> {\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  root: StageNode<TOut, TScope>;\n  executionRuntime: IExecutionRuntime;\n  ScopeFactory: ScopeFactory<TScope>;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  streamHandlers?: StreamHandlers;\n  scopeProtectionMode: ScopeProtectionMode;\n  readOnlyContext?: unknown;\n  /** Execution environment — propagates to subflows automatically. */\n  executionEnv?: ExecutionEnv;\n  narrativeGenerator: IControlFlowNarrative;\n  logger: ILogger;\n  signal?: AbortSignal;\n}\n\n/** Options for FlowChartExecutor.run(). */\nexport interface RunOptions {\n  /** AbortSignal for cooperative cancellation. */\n  signal?: AbortSignal;\n  /** Timeout in milliseconds. Creates an internal AbortController. */\n  timeoutMs?: number;\n  /**\n   * Runtime input data for the pipeline.\n   * Becomes the readOnlyContext accessible via `scope.getArgs()`.\n   * Stages cannot overwrite these keys with `setValue()`.\n   */\n  input?: Record<string, unknown>;\n  /**\n   * Execution environment — read-only infrastructure values that propagate\n   * through nested executors (like `process.env` for flowcharts).\n   * Accessible via `scope.getEnv()`. Inherited by subflows automatically.\n   */\n  env?: ExecutionEnv;\n}\n\n// ---------------------------------------------------------------------------\n// Flow Control Narrative\n// ---------------------------------------------------------------------------\n\nexport type FlowControlType = 'next' | 'branch' | 'children' | 'selected' | 'subflow' | 'loop';\n\nexport interface FlowMessage {\n  type: FlowControlType;\n  description: string;\n  targetStage?: string | string[];\n  rationale?: string;\n  count?: number;\n  iteration?: number;\n  timestamp?: number;\n}\n\n// ---------------------------------------------------------------------------\n// Traversal Extractor\n// ---------------------------------------------------------------------------\n\nexport interface RuntimeStructureMetadata {\n  type: 'stage' | 'decider' | 'fork' | 'streaming';\n  subflowId?: string;\n  isSubflowRoot?: boolean;\n  subflowName?: string;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  loopTarget?: string;\n  isDynamic?: boolean;\n  isLoopReference?: boolean;\n  streamId?: string;\n}\n\nexport interface StageSnapshot<TOut = any, TScope = any> {\n  node: StageNode<TOut, TScope>;\n  context: StageContext;\n  stepNumber: number;\n  structureMetadata: RuntimeStructureMetadata;\n  scopeState?: Record<string, unknown>;\n  debugInfo?: {\n    logs: Record<string, unknown>;\n    errors: Record<string, unknown>;\n    metrics: Record<string, unknown>;\n    evals: Record<string, unknown>;\n    flowMessages?: FlowMessage[];\n  };\n  stageOutput?: unknown;\n  errorInfo?: { type: string; message: string };\n  historyIndex?: number;\n}\n\nexport type TraversalExtractor<TResult = unknown> = (snapshot: StageSnapshot) => TResult | undefined | null;\n\nexport interface ExtractorError {\n  stagePath: string;\n  message: string;\n  error: unknown;\n}\n\n// ---------------------------------------------------------------------------\n// Node Result\n// ---------------------------------------------------------------------------\n\nexport type NodeResultType = {\n  id: string;\n  result: unknown;\n  isError?: boolean;\n};\n\n// ---------------------------------------------------------------------------\n// Execution Response\n// ---------------------------------------------------------------------------\n\nexport type BranchResult = {\n  result: string | Error;\n  isError: boolean;\n};\n\nexport type BranchResults = { [branchId: string]: BranchResult };\nexport type TraversalResult = BranchResults | string | Error;\n\n// ---------------------------------------------------------------------------\n// Serialized Pipeline Structure (for visualization)\n// ---------------------------------------------------------------------------\n\nexport interface SerializedPipelineNode {\n  name: string;\n  id?: string;\n  type?: 'stage' | 'decider' | 'fork' | 'streaming' | 'loop' | 'user' | 'tool' | 'function' | 'sequence';\n  description?: string;\n  children?: SerializedPipelineNode[];\n  next?: SerializedPipelineNode;\n  branches?: Record<string, SerializedPipelineNode>;\n  hasDecider?: boolean;\n  hasSelector?: boolean;\n  hasSubtree?: boolean;\n  isStreaming?: boolean;\n  streamId?: string;\n  isSubflowRoot?: boolean;\n  subflowId?: string;\n  subflowName?: string;\n  loopTarget?: string;\n  isLoopReference?: boolean;\n  isParallelChild?: boolean;\n  parallelGroupId?: string;\n  isDynamic?: boolean;\n}\n\n// ---------------------------------------------------------------------------\n// FlowChart (compiled output of FlowChartBuilder)\n// ---------------------------------------------------------------------------\n\nexport type FlowChart<TOut = any, TScope = any> = {\n  root: StageNode<TOut, TScope>;\n  stageMap: Map<string, StageFunction<TOut, TScope>>;\n  extractor?: TraversalExtractor;\n  subflows?: Record<string, { root: StageNode<TOut, TScope> }>;\n  enrichSnapshots?: boolean;\n  enableNarrative?: boolean;\n  logger?: ILogger;\n  buildTimeStructure?: SerializedPipelineStructure;\n  /** Input schema (Zod or JSON Schema) — used for runtime input validation. */\n  inputSchema?: unknown;\n};\n\n/** Alias for SerializedPipelineNode used as full structure */\nexport type SerializedPipelineStructure = SerializedPipelineNode & {\n  branchIds?: string[];\n  subflowStructure?: SerializedPipelineStructure;\n  iterationCount?: number;\n};\n"]}
|
|
@@ -61,4 +61,4 @@ export class ExecutionRuntime {
|
|
|
61
61
|
}
|
|
62
62
|
}
|
|
63
63
|
}
|
|
64
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
64
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRXhlY3V0aW9uUnVudGltZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvcnVubmVyL0V4ZWN1dGlvblJ1bnRpbWUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7R0FVRztBQUVILE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUNqRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDekQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBNEJ6RCxNQUFNLE9BQU8sZ0JBQWdCO0lBSzNCLFlBQVksUUFBZ0IsRUFBRSxNQUFjLEVBQUUsYUFBdUIsRUFBRSxZQUFzQjtRQUMzRixJQUFJLENBQUMsZ0JBQWdCLEdBQUcsSUFBSSxRQUFRLENBQUMsWUFBWSxDQUFDLENBQUM7UUFDbkQsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLFlBQVksQ0FBQyxhQUFhLEVBQUUsWUFBWSxDQUFDLENBQUM7UUFDakUsSUFBSSxDQUFDLGdCQUFnQixHQUFHLElBQUksWUFBWSxDQUFDLEVBQUUsRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLElBQUksQ0FBQyxXQUFXLEVBQUUsRUFBRSxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO0lBQzlHLENBQUM7SUFFRCxZQUFZO1FBQ1YsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUMxQyxPQUFPLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFNBQW9DLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO0lBQ3hGLENBQUM7SUFFRCxhQUFhLENBQUMsSUFBYyxFQUFFLEdBQVcsRUFBRSxLQUFjO1FBQ3ZELElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxTQUFTLENBQUMsSUFBSSxFQUFFLEdBQUcsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUNwRCxDQUFDO0lBRUQsV0FBVztRQUNULE9BQU87WUFDTCxXQUFXLEVBQUUsSUFBSSxDQUFDLFdBQVcsQ0FBQyxRQUFRLEVBQUU7WUFDeEMsYUFBYSxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxXQUFXLEVBQUU7WUFDbEQsU0FBUyxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLEVBQUU7U0FDeEMsQ0FBQztJQUNKLENBQUM7SUFFRCxnQkFBZ0I7UUFDZCxNQUFNLFNBQVMsR0FBcUIsRUFBRSxDQUFDO1FBQ3ZDLElBQUksU0FBUyxHQUFHLENBQUMsQ0FBQztRQUVsQixJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDLE9BQU8sRUFBRSxFQUFFO1lBQ3RELE1BQU0sYUFBYSxHQUFJLE9BQU8sQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDLE9BQW9CLElBQUksRUFBRSxDQUFDO1lBQzNFLE1BQU0sWUFBWSxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsWUFBWSxDQUFDO1lBQ2hELE1BQU0sV0FBVyxHQUFHLFlBQVksQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxZQUFZLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQztZQUUxRSxTQUFTLENBQUMsSUFBSSxDQUFDO2dCQUNiLE9BQU8sRUFBRSxPQUFPLENBQUMsVUFBVSxFQUFFO2dCQUM3QixTQUFTLEVBQUUsT0FBTyxDQUFDLFNBQVM7Z0JBQzVCLGFBQWE7Z0JBQ2IsV0FBVztnQkFDWCxTQUFTLEVBQUUsU0FBUyxFQUFFO2FBQ3ZCLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO1FBRUgsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztJQUVPLGVBQWUsQ0FBQyxPQUFxQixFQUFFLE9BQW9DO1FBQ2pGLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNqQixJQUFJLE9BQU8sQ0FBQyxRQUFRLEVBQUUsQ0FBQztZQUNyQixLQUFLLE1BQU0sS0FBSyxJQUFJLE9BQU8sQ0FBQyxRQUFRLEVBQUUsQ0FBQztnQkFDckMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxLQUFLLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFDdkMsQ0FBQztRQUNILENBQUM7UUFDRCxJQUFJLE9BQU8sQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNqQixJQUFJLENBQUMsZUFBZSxDQUFDLE9BQU8sQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7UUFDOUMsQ0FBQztJQUNILENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogRXhlY3V0aW9uUnVudGltZSDigJQgVGhlIHJ1bnRpbWUgZW52aXJvbm1lbnQgZm9yIG9uZSBmbG93Y2hhcnQgZXhlY3V0aW9uLlxuICpcbiAqIFdpcmVzIHVwIHRoZSB0aHJlZSBtZW1vcnkgcHJpbWl0aXZlcyBpbnRvIGEgc2luZ2xlIGNvbnRhaW5lcjpcbiAqICAgLSBTaGFyZWRNZW1vcnkgKHRoZSBoZWFwIOKAlCBzaGFyZWQgc3RhdGUgYWNyb3NzIGFsbCBzdGFnZXMpXG4gKiAgIC0gU3RhZ2VDb250ZXh0ICAodGhlIGNhbGwgc3RhY2sg4oCUIHBlci1zdGFnZSBleGVjdXRpb24gdHJlZSlcbiAqICAgLSBFdmVudExvZyAgICAgICh0aGUgdHJhbnNhY3Rpb24gbG9nIOKAlCBjb21taXQgaGlzdG9yeSBmb3IgcmVwbGF5KVxuICpcbiAqIFRoZSBlbmdpbmUgKEZsb3djaGFydFRyYXZlcnNlcikgcmVjZWl2ZXMgdGhpcyBhcyBpdHMgcnVudGltZSBwYXJhbWV0ZXIuXG4gKiBBZnRlciBleGVjdXRpb24sIGNvbnN1bWVycyBxdWVyeSBpdCBmb3IgdGhlIGZ1bGwgZXhlY3V0aW9uIHN0YXRlLlxuICovXG5cbmltcG9ydCB7IEV2ZW50TG9nIH0gZnJvbSAnLi4vbWVtb3J5L0V2ZW50TG9nLmpzJztcbmltcG9ydCB7IFNoYXJlZE1lbW9yeSB9IGZyb20gJy4uL21lbW9yeS9TaGFyZWRNZW1vcnkuanMnO1xuaW1wb3J0IHsgU3RhZ2VDb250ZXh0IH0gZnJvbSAnLi4vbWVtb3J5L1N0YWdlQ29udGV4dC5qcyc7XG5pbXBvcnQgdHlwZSB7IENvbW1pdEJ1bmRsZSwgRmxvd01lc3NhZ2UsIFN0YWdlU25hcHNob3QgfSBmcm9tICcuLi9tZW1vcnkvdHlwZXMuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIE5hcnJhdGl2ZUVudHJ5IHtcbiAgc3RhZ2VJZDogc3RyaW5nO1xuICBzdGFnZU5hbWU6IHN0cmluZztcbiAgc3RhZ2VNZXNzYWdlczogc3RyaW5nW107XG4gIGZsb3dNZXNzYWdlPzogRmxvd01lc3NhZ2U7XG4gIHRpbWVJbmRleDogbnVtYmVyO1xufVxuXG4vKiogU25hcHNob3Qgb2YgYSBzaW5nbGUgcmVjb3JkZXIncyBjb2xsZWN0ZWQgZGF0YS4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmVjb3JkZXJTbmFwc2hvdCB7XG4gIGlkOiBzdHJpbmc7XG4gIG5hbWU6IHN0cmluZztcbiAgZGF0YTogdW5rbm93bjtcbn1cblxuZXhwb3J0IHR5cGUgUnVudGltZVNuYXBzaG90ID0ge1xuICBzaGFyZWRTdGF0ZTogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIGV4ZWN1dGlvblRyZWU6IFN0YWdlU25hcHNob3Q7XG4gIGNvbW1pdExvZzogQ29tbWl0QnVuZGxlW107XG4gIC8qKiBQZXItc3ViZmxvdyBleGVjdXRpb24gcmVzdWx0cyAoa2V5ZWQgYnkgc3ViZmxvd0lkKS4gKi9cbiAgc3ViZmxvd1Jlc3VsdHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgLyoqIFNuYXBzaG90IGRhdGEgZnJvbSByZWNvcmRlcnMgdGhhdCBpbXBsZW1lbnQgdG9TbmFwc2hvdCgpLiAqL1xuICByZWNvcmRlcnM/OiBSZWNvcmRlclNuYXBzaG90W107XG59O1xuXG5leHBvcnQgY2xhc3MgRXhlY3V0aW9uUnVudGltZSB7XG4gIHB1YmxpYyBnbG9iYWxTdG9yZTogU2hhcmVkTWVtb3J5O1xuICBwdWJsaWMgcm9vdFN0YWdlQ29udGV4dDogU3RhZ2VDb250ZXh0O1xuICBwdWJsaWMgZXhlY3V0aW9uSGlzdG9yeTogRXZlbnRMb2c7XG5cbiAgY29uc3RydWN0b3Iocm9vdE5hbWU6IHN0cmluZywgcm9vdElkOiBzdHJpbmcsIGRlZmF1bHRWYWx1ZXM/OiB1bmtub3duLCBpbml0aWFsU3RhdGU/OiB1bmtub3duKSB7XG4gICAgdGhpcy5leGVjdXRpb25IaXN0b3J5ID0gbmV3IEV2ZW50TG9nKGluaXRpYWxTdGF0ZSk7XG4gICAgdGhpcy5nbG9iYWxTdG9yZSA9IG5ldyBTaGFyZWRNZW1vcnkoZGVmYXVsdFZhbHVlcywgaW5pdGlhbFN0YXRlKTtcbiAgICB0aGlzLnJvb3RTdGFnZUNvbnRleHQgPSBuZXcgU3RhZ2VDb250ZXh0KCcnLCByb290TmFtZSwgcm9vdElkLCB0aGlzLmdsb2JhbFN0b3JlLCAnJywgdGhpcy5leGVjdXRpb25IaXN0b3J5KTtcbiAgfVxuXG4gIGdldFBpcGVsaW5lcygpOiBzdHJpbmdbXSB7XG4gICAgY29uc3Qgc3RhdGUgPSB0aGlzLmdsb2JhbFN0b3JlLmdldFN0YXRlKCk7XG4gICAgcmV0dXJuIHN0YXRlLnBpcGVsaW5lcyA/IE9iamVjdC5rZXlzKHN0YXRlLnBpcGVsaW5lcyBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgOiBbXTtcbiAgfVxuXG4gIHNldFJvb3RPYmplY3QocGF0aDogc3RyaW5nW10sIGtleTogc3RyaW5nLCB2YWx1ZTogdW5rbm93bikge1xuICAgIHRoaXMucm9vdFN0YWdlQ29udGV4dC5zZXRPYmplY3QocGF0aCwga2V5LCB2YWx1ZSk7XG4gIH1cblxuICBnZXRTbmFwc2hvdCgpOiBSdW50aW1lU25hcHNob3Qge1xuICAgIHJldHVybiB7XG4gICAgICBzaGFyZWRTdGF0ZTogdGhpcy5nbG9iYWxTdG9yZS5nZXRTdGF0ZSgpLFxuICAgICAgZXhlY3V0aW9uVHJlZTogdGhpcy5yb290U3RhZ2VDb250ZXh0LmdldFNuYXBzaG90KCksXG4gICAgICBjb21taXRMb2c6IHRoaXMuZXhlY3V0aW9uSGlzdG9yeS5saXN0KCksXG4gICAgfTtcbiAgfVxuXG4gIGdldEZ1bGxOYXJyYXRpdmUoKTogTmFycmF0aXZlRW50cnlbXSB7XG4gICAgY29uc3QgbmFycmF0aXZlOiBOYXJyYXRpdmVFbnRyeVtdID0gW107XG4gICAgbGV0IHRpbWVJbmRleCA9IDA7XG5cbiAgICB0aGlzLndhbGtDb250ZXh0VHJlZSh0aGlzLnJvb3RTdGFnZUNvbnRleHQsIChjb250ZXh0KSA9PiB7XG4gICAgICBjb25zdCBzdGFnZU1lc3NhZ2VzID0gKGNvbnRleHQuZGVidWcubG9nQ29udGV4dC5tZXNzYWdlIGFzIHN0cmluZ1tdKSB8fCBbXTtcbiAgICAgIGNvbnN0IGZsb3dNZXNzYWdlcyA9IGNvbnRleHQuZGVidWcuZmxvd01lc3NhZ2VzO1xuICAgICAgY29uc3QgZmxvd01lc3NhZ2UgPSBmbG93TWVzc2FnZXMubGVuZ3RoID4gMCA/IGZsb3dNZXNzYWdlc1swXSA6IHVuZGVmaW5lZDtcblxuICAgICAgbmFycmF0aXZlLnB1c2goe1xuICAgICAgICBzdGFnZUlkOiBjb250ZXh0LmdldFN0YWdlSWQoKSxcbiAgICAgICAgc3RhZ2VOYW1lOiBjb250ZXh0LnN0YWdlTmFtZSxcbiAgICAgICAgc3RhZ2VNZXNzYWdlcyxcbiAgICAgICAgZmxvd01lc3NhZ2UsXG4gICAgICAgIHRpbWVJbmRleDogdGltZUluZGV4KyssXG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIHJldHVybiBuYXJyYXRpdmU7XG4gIH1cblxuICBwcml2YXRlIHdhbGtDb250ZXh0VHJlZShjb250ZXh0OiBTdGFnZUNvbnRleHQsIHZpc2l0b3I6IChjdHg6IFN0YWdlQ29udGV4dCkgPT4gdm9pZCk6IHZvaWQge1xuICAgIHZpc2l0b3IoY29udGV4dCk7XG4gICAgaWYgKGNvbnRleHQuY2hpbGRyZW4pIHtcbiAgICAgIGZvciAoY29uc3QgY2hpbGQgb2YgY29udGV4dC5jaGlsZHJlbikge1xuICAgICAgICB0aGlzLndhbGtDb250ZXh0VHJlZShjaGlsZCwgdmlzaXRvcik7XG4gICAgICB9XG4gICAgfVxuICAgIGlmIChjb250ZXh0Lm5leHQpIHtcbiAgICAgIHRoaXMud2Fsa0NvbnRleHRUcmVlKGNvbnRleHQubmV4dCwgdmlzaXRvcik7XG4gICAgfVxuICB9XG59XG4iXX0=
|