footprintjs 4.2.0 → 4.3.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CLAUDE.md +43 -1
- package/dist/esm/index.js +1 -1
- package/dist/esm/lib/builder/FlowChartBuilder.js +53 -1
- package/dist/esm/lib/builder/types.js +1 -1
- package/dist/esm/lib/engine/graph/StageNode.js +1 -1
- package/dist/esm/lib/engine/handlers/ChildrenExecutor.js +14 -1
- package/dist/esm/lib/engine/handlers/DeciderHandler.js +7 -1
- package/dist/esm/lib/engine/handlers/SelectorHandler.js +7 -1
- package/dist/esm/lib/engine/handlers/StageRunner.js +13 -1
- package/dist/esm/lib/engine/handlers/SubflowExecutor.js +8 -1
- package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +39 -1
- package/dist/esm/lib/engine/narrative/FlowRecorderDispatcher.js +31 -1
- package/dist/esm/lib/engine/narrative/NarrativeFlowRecorder.js +10 -1
- package/dist/esm/lib/engine/narrative/NullControlFlowNarrativeGenerator.js +3 -1
- package/dist/esm/lib/engine/narrative/narrativeTypes.js +1 -1
- package/dist/esm/lib/engine/narrative/types.js +1 -1
- package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +8 -1
- package/dist/esm/lib/engine/types.js +2 -1
- package/dist/esm/lib/pause/index.js +2 -0
- package/dist/esm/lib/pause/types.js +58 -0
- package/dist/esm/lib/reactive/types.js +2 -1
- package/dist/esm/lib/runner/ExecutionRuntime.js +11 -2
- package/dist/esm/lib/runner/FlowChartExecutor.js +232 -8
- package/dist/esm/lib/scope/ScopeFacade.js +21 -1
- package/dist/esm/lib/scope/recorders/DebugRecorder.js +19 -1
- package/dist/esm/lib/scope/recorders/MetricRecorder.js +10 -1
- package/dist/esm/lib/scope/types.js +1 -1
- package/dist/index.js +1 -1
- package/dist/lib/builder/FlowChartBuilder.js +53 -1
- package/dist/lib/builder/types.js +1 -1
- package/dist/lib/engine/graph/StageNode.js +1 -1
- package/dist/lib/engine/handlers/ChildrenExecutor.js +14 -1
- package/dist/lib/engine/handlers/DeciderHandler.js +7 -1
- package/dist/lib/engine/handlers/SelectorHandler.js +7 -1
- package/dist/lib/engine/handlers/StageRunner.js +17 -5
- package/dist/lib/engine/handlers/SubflowExecutor.js +8 -1
- package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +39 -1
- package/dist/lib/engine/narrative/FlowRecorderDispatcher.js +31 -1
- package/dist/lib/engine/narrative/NarrativeFlowRecorder.js +10 -1
- package/dist/lib/engine/narrative/NullControlFlowNarrativeGenerator.js +3 -1
- package/dist/lib/engine/narrative/narrativeTypes.js +1 -1
- package/dist/lib/engine/narrative/types.js +1 -1
- package/dist/lib/engine/traversal/FlowchartTraverser.js +8 -1
- package/dist/lib/engine/types.js +6 -2
- package/dist/lib/pause/index.js +8 -0
- package/dist/lib/pause/types.js +64 -0
- package/dist/lib/reactive/types.js +2 -1
- package/dist/lib/runner/ExecutionRuntime.js +11 -2
- package/dist/lib/runner/FlowChartExecutor.js +232 -8
- package/dist/lib/scope/ScopeFacade.js +21 -1
- package/dist/lib/scope/recorders/DebugRecorder.js +19 -1
- package/dist/lib/scope/recorders/MetricRecorder.js +10 -1
- package/dist/lib/scope/types.js +1 -1
- package/dist/types/index.d.ts +4 -1
- package/dist/types/lib/builder/FlowChartBuilder.d.ts +22 -0
- package/dist/types/lib/builder/types.d.ts +3 -1
- package/dist/types/lib/engine/graph/StageNode.d.ts +9 -0
- package/dist/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +9 -1
- package/dist/types/lib/engine/narrative/FlowRecorderDispatcher.d.ts +2 -0
- package/dist/types/lib/engine/narrative/NarrativeFlowRecorder.d.ts +3 -1
- package/dist/types/lib/engine/narrative/NullControlFlowNarrativeGenerator.d.ts +2 -0
- package/dist/types/lib/engine/narrative/narrativeTypes.d.ts +1 -1
- package/dist/types/lib/engine/narrative/types.d.ts +24 -0
- package/dist/types/lib/engine/types.d.ts +11 -0
- package/dist/types/lib/pause/index.d.ts +2 -0
- package/dist/types/lib/pause/types.d.ts +175 -0
- package/dist/types/lib/runner/ExecutionRuntime.d.ts +4 -0
- package/dist/types/lib/runner/FlowChartExecutor.d.ts +54 -2
- package/dist/types/lib/scope/ScopeFacade.d.ts +4 -0
- package/dist/types/lib/scope/recorders/DebugRecorder.d.ts +4 -2
- package/dist/types/lib/scope/recorders/MetricRecorder.d.ts +4 -1
- package/dist/types/lib/scope/types.d.ts +10 -0
- package/package.json +1 -1
|
@@ -24,6 +24,7 @@ const NarrativeFlowRecorder_js_1 = require("../engine/narrative/NarrativeFlowRec
|
|
|
24
24
|
const ManifestFlowRecorder_js_1 = require("../engine/narrative/recorders/ManifestFlowRecorder.js");
|
|
25
25
|
const FlowchartTraverser_js_1 = require("../engine/traversal/FlowchartTraverser.js");
|
|
26
26
|
const types_js_1 = require("../engine/types.js");
|
|
27
|
+
const types_js_2 = require("../pause/types.js");
|
|
27
28
|
const ScopeFacade_js_1 = require("../scope/ScopeFacade.js");
|
|
28
29
|
const ExecutionRuntime_js_1 = require("./ExecutionRuntime.js");
|
|
29
30
|
const validateInput_js_1 = require("./validateInput.js");
|
|
@@ -91,8 +92,8 @@ class FlowChartExecutor {
|
|
|
91
92
|
};
|
|
92
93
|
this.traverser = this.createTraverser();
|
|
93
94
|
}
|
|
94
|
-
createTraverser(signal, readOnlyContextOverride, env, maxDepth) {
|
|
95
|
-
var _a, _b, _c;
|
|
95
|
+
createTraverser(signal, readOnlyContextOverride, env, maxDepth, overrides) {
|
|
96
|
+
var _a, _b, _c, _d, _e;
|
|
96
97
|
const args = this.flowChartArgs;
|
|
97
98
|
const fc = args.flowChart;
|
|
98
99
|
const narrativeFlag = this.narrativeEnabled || ((_a = fc.enableNarrative) !== null && _a !== void 0 ? _a : false);
|
|
@@ -100,7 +101,10 @@ class FlowChartExecutor {
|
|
|
100
101
|
// Collect all scope modifiers (recorders, redaction) into a single list,
|
|
101
102
|
// then create ONE factory that applies them in a loop. Replaces the
|
|
102
103
|
// previous 4-deep closure nesting with a flat, debuggable composition.
|
|
103
|
-
if (
|
|
104
|
+
if (overrides === null || overrides === void 0 ? void 0 : overrides.preserveRecorders) {
|
|
105
|
+
// Resume mode: keep existing combinedRecorder so narrative accumulates
|
|
106
|
+
}
|
|
107
|
+
else if (narrativeFlag) {
|
|
104
108
|
this.combinedRecorder = new CombinedNarrativeRecorder_js_1.CombinedNarrativeRecorder(this.narrativeOptions);
|
|
105
109
|
}
|
|
106
110
|
else {
|
|
@@ -160,9 +164,25 @@ class FlowChartExecutor {
|
|
|
160
164
|
mod(scope);
|
|
161
165
|
return scope;
|
|
162
166
|
});
|
|
163
|
-
const
|
|
167
|
+
const effectiveRoot = (_b = overrides === null || overrides === void 0 ? void 0 : overrides.root) !== null && _b !== void 0 ? _b : fc.root;
|
|
168
|
+
const effectiveInitialContext = (_c = overrides === null || overrides === void 0 ? void 0 : overrides.initialContext) !== null && _c !== void 0 ? _c : args.initialContext;
|
|
169
|
+
let runtime;
|
|
170
|
+
if (overrides === null || overrides === void 0 ? void 0 : overrides.existingRuntime) {
|
|
171
|
+
// Resume mode: reuse existing runtime so execution tree continues from pause point.
|
|
172
|
+
// Preserve the original root for getSnapshot() (full tree), then advance
|
|
173
|
+
// rootStageContext to a continuation from the leaf (for traversal).
|
|
174
|
+
runtime = overrides.existingRuntime;
|
|
175
|
+
runtime.preserveSnapshotRoot();
|
|
176
|
+
let leaf = runtime.rootStageContext;
|
|
177
|
+
while (leaf.next)
|
|
178
|
+
leaf = leaf.next;
|
|
179
|
+
runtime.rootStageContext = leaf.createNext('', effectiveRoot.name, effectiveRoot.id);
|
|
180
|
+
}
|
|
181
|
+
else {
|
|
182
|
+
runtime = new ExecutionRuntime_js_1.ExecutionRuntime(effectiveRoot.name, effectiveRoot.id, args.defaultValuesForContext, effectiveInitialContext);
|
|
183
|
+
}
|
|
164
184
|
return new FlowchartTraverser_js_1.FlowchartTraverser({
|
|
165
|
-
root:
|
|
185
|
+
root: effectiveRoot,
|
|
166
186
|
stageMap: fc.stageMap,
|
|
167
187
|
scopeFactory,
|
|
168
188
|
executionRuntime: runtime,
|
|
@@ -172,10 +192,10 @@ class FlowChartExecutor {
|
|
|
172
192
|
extractor: fc.extractor,
|
|
173
193
|
scopeProtectionMode: args.scopeProtectionMode,
|
|
174
194
|
subflows: fc.subflows,
|
|
175
|
-
enrichSnapshots: (
|
|
195
|
+
enrichSnapshots: (_d = args.enrichSnapshots) !== null && _d !== void 0 ? _d : fc.enrichSnapshots,
|
|
176
196
|
narrativeEnabled: narrativeFlag,
|
|
177
197
|
buildTimeStructure: fc.buildTimeStructure,
|
|
178
|
-
logger: (
|
|
198
|
+
logger: (_e = fc.logger) !== null && _e !== void 0 ? _e : types_js_1.defaultLogger,
|
|
179
199
|
signal,
|
|
180
200
|
executionEnv: env,
|
|
181
201
|
flowRecorders: this.buildFlowRecordersList(),
|
|
@@ -210,6 +230,190 @@ class FlowChartExecutor {
|
|
|
210
230
|
patterns: ((_b = (_a = this.redactionPolicy) === null || _a === void 0 ? void 0 : _a.patterns) !== null && _b !== void 0 ? _b : []).map((p) => p.source),
|
|
211
231
|
};
|
|
212
232
|
}
|
|
233
|
+
// ─── Pause/Resume ───
|
|
234
|
+
/**
|
|
235
|
+
* Returns the checkpoint from the most recent paused execution, or `undefined`
|
|
236
|
+
* if the last run completed without pausing.
|
|
237
|
+
*
|
|
238
|
+
* The checkpoint is JSON-serializable — store it in Redis, Postgres, localStorage, etc.
|
|
239
|
+
*
|
|
240
|
+
* @example
|
|
241
|
+
* ```typescript
|
|
242
|
+
* const result = await executor.run({ input });
|
|
243
|
+
* if (executor.isPaused()) {
|
|
244
|
+
* const checkpoint = executor.getCheckpoint()!;
|
|
245
|
+
* await redis.set(`session:${id}`, JSON.stringify(checkpoint));
|
|
246
|
+
* }
|
|
247
|
+
* ```
|
|
248
|
+
*/
|
|
249
|
+
getCheckpoint() {
|
|
250
|
+
return this.lastCheckpoint;
|
|
251
|
+
}
|
|
252
|
+
/** Returns `true` if the most recent run() was paused (checkpoint available). */
|
|
253
|
+
isPaused() {
|
|
254
|
+
return this.lastCheckpoint !== undefined;
|
|
255
|
+
}
|
|
256
|
+
/**
|
|
257
|
+
* Resume a paused flowchart from a checkpoint.
|
|
258
|
+
*
|
|
259
|
+
* Restores the scope state, calls the paused stage's `resumeFn` with the
|
|
260
|
+
* provided input, then continues traversal from the next stage.
|
|
261
|
+
*
|
|
262
|
+
* The checkpoint can come from `getCheckpoint()` on a previous run, or from
|
|
263
|
+
* a serialized checkpoint stored in Redis/Postgres/localStorage.
|
|
264
|
+
*
|
|
265
|
+
* **Narrative/recorder state is reset on resume.** To keep a unified narrative
|
|
266
|
+
* across pause/resume cycles, collect it before calling resume.
|
|
267
|
+
*
|
|
268
|
+
* @example
|
|
269
|
+
* ```typescript
|
|
270
|
+
* // After a pause...
|
|
271
|
+
* const checkpoint = executor.getCheckpoint()!;
|
|
272
|
+
* await redis.set(`session:${id}`, JSON.stringify(checkpoint));
|
|
273
|
+
*
|
|
274
|
+
* // Later (possibly different server, same chart)
|
|
275
|
+
* const checkpoint = JSON.parse(await redis.get(`session:${id}`));
|
|
276
|
+
* const executor = new FlowChartExecutor(chart);
|
|
277
|
+
* const result = await executor.resume(checkpoint, { approved: true });
|
|
278
|
+
* ```
|
|
279
|
+
*/
|
|
280
|
+
async resume(checkpoint, resumeInput, options) {
|
|
281
|
+
var _a, _b;
|
|
282
|
+
this.lastCheckpoint = undefined;
|
|
283
|
+
// ── Validate checkpoint structure (may come from untrusted external storage) ──
|
|
284
|
+
if (!checkpoint ||
|
|
285
|
+
typeof checkpoint !== 'object' ||
|
|
286
|
+
typeof checkpoint.sharedState !== 'object' ||
|
|
287
|
+
checkpoint.sharedState === null ||
|
|
288
|
+
Array.isArray(checkpoint.sharedState)) {
|
|
289
|
+
throw new Error('Invalid checkpoint: sharedState must be a plain object.');
|
|
290
|
+
}
|
|
291
|
+
if (typeof checkpoint.pausedStageId !== 'string' || checkpoint.pausedStageId === '') {
|
|
292
|
+
throw new Error('Invalid checkpoint: pausedStageId must be a non-empty string.');
|
|
293
|
+
}
|
|
294
|
+
if (!Array.isArray(checkpoint.subflowPath) ||
|
|
295
|
+
!checkpoint.subflowPath.every((s) => typeof s === 'string')) {
|
|
296
|
+
throw new Error('Invalid checkpoint: subflowPath must be an array of strings.');
|
|
297
|
+
}
|
|
298
|
+
// Find the paused node in the graph
|
|
299
|
+
const pausedNode = this.findNodeInGraph(checkpoint.pausedStageId, checkpoint.subflowPath);
|
|
300
|
+
if (!pausedNode) {
|
|
301
|
+
throw new Error(`Cannot resume: stage '${checkpoint.pausedStageId}' not found in flowchart. ` +
|
|
302
|
+
'The chart may have changed since the checkpoint was created.');
|
|
303
|
+
}
|
|
304
|
+
if (!pausedNode.resumeFn) {
|
|
305
|
+
throw new Error(`Cannot resume: stage '${pausedNode.name}' (${pausedNode.id}) has no resumeFn. ` +
|
|
306
|
+
'Only stages created with addPausableFunction() can be resumed.');
|
|
307
|
+
}
|
|
308
|
+
// Build a synthetic resume node: calls resumeFn with resumeInput, then continues to original next.
|
|
309
|
+
// resumeFn signature is (scope, input) per PausableHandler — wrap to match StageFunction(scope, breakFn).
|
|
310
|
+
const resumeFn = pausedNode.resumeFn;
|
|
311
|
+
const resumeStageFn = (scope) => {
|
|
312
|
+
return resumeFn(scope, resumeInput);
|
|
313
|
+
};
|
|
314
|
+
const resumeNode = {
|
|
315
|
+
name: pausedNode.name,
|
|
316
|
+
id: pausedNode.id,
|
|
317
|
+
description: pausedNode.description,
|
|
318
|
+
fn: resumeStageFn,
|
|
319
|
+
next: pausedNode.next,
|
|
320
|
+
};
|
|
321
|
+
// Don't clear recorders — resume continues from previous state.
|
|
322
|
+
// Narrative, metrics, debug entries accumulate across pause/resume.
|
|
323
|
+
// Reuse the existing runtime so the execution tree continues from the pause point.
|
|
324
|
+
// preserveRecorders keeps the CombinedNarrativeRecorder so narrative accumulates.
|
|
325
|
+
const existingRuntime = this.traverser.getRuntime();
|
|
326
|
+
this.traverser = this.createTraverser(options === null || options === void 0 ? void 0 : options.signal, undefined, options === null || options === void 0 ? void 0 : options.env, options === null || options === void 0 ? void 0 : options.maxDepth, {
|
|
327
|
+
root: resumeNode,
|
|
328
|
+
initialContext: checkpoint.sharedState,
|
|
329
|
+
preserveRecorders: true,
|
|
330
|
+
existingRuntime,
|
|
331
|
+
});
|
|
332
|
+
// Fire onResume event on all recorders (flow + scope)
|
|
333
|
+
const hasInput = resumeInput !== undefined;
|
|
334
|
+
const flowResumeEvent = {
|
|
335
|
+
stageName: pausedNode.name,
|
|
336
|
+
stageId: pausedNode.id,
|
|
337
|
+
hasInput,
|
|
338
|
+
};
|
|
339
|
+
if (this.combinedRecorder)
|
|
340
|
+
this.combinedRecorder.onResume(flowResumeEvent);
|
|
341
|
+
for (const r of this.flowRecorders)
|
|
342
|
+
(_a = r.onResume) === null || _a === void 0 ? void 0 : _a.call(r, flowResumeEvent);
|
|
343
|
+
const scopeResumeEvent = {
|
|
344
|
+
stageName: pausedNode.name,
|
|
345
|
+
stageId: pausedNode.id,
|
|
346
|
+
hasInput,
|
|
347
|
+
pipelineId: '',
|
|
348
|
+
timestamp: Date.now(),
|
|
349
|
+
};
|
|
350
|
+
for (const r of this.scopeRecorders)
|
|
351
|
+
(_b = r.onResume) === null || _b === void 0 ? void 0 : _b.call(r, scopeResumeEvent);
|
|
352
|
+
try {
|
|
353
|
+
return await this.traverser.execute();
|
|
354
|
+
}
|
|
355
|
+
catch (error) {
|
|
356
|
+
if ((0, types_js_2.isPauseSignal)(error)) {
|
|
357
|
+
const snapshot = this.traverser.getSnapshot();
|
|
358
|
+
const sfResults = this.traverser.getSubflowResults();
|
|
359
|
+
this.lastCheckpoint = {
|
|
360
|
+
sharedState: snapshot.sharedState,
|
|
361
|
+
executionTree: snapshot.executionTree,
|
|
362
|
+
pausedStageId: error.stageId,
|
|
363
|
+
subflowPath: error.subflowPath,
|
|
364
|
+
pauseData: error.pauseData,
|
|
365
|
+
...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),
|
|
366
|
+
pausedAt: Date.now(),
|
|
367
|
+
};
|
|
368
|
+
return { paused: true, checkpoint: this.lastCheckpoint };
|
|
369
|
+
}
|
|
370
|
+
throw error;
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
/**
|
|
374
|
+
* Find a StageNode in the compiled graph by ID.
|
|
375
|
+
* Handles subflow paths by drilling into registered subflows.
|
|
376
|
+
*/
|
|
377
|
+
findNodeInGraph(stageId, subflowPath) {
|
|
378
|
+
var _a;
|
|
379
|
+
const fc = this.flowChartArgs.flowChart;
|
|
380
|
+
if (subflowPath.length === 0) {
|
|
381
|
+
// Top-level: DFS from root
|
|
382
|
+
return this.dfsFind(fc.root, stageId);
|
|
383
|
+
}
|
|
384
|
+
// Subflow: drill into the subflow chain, then search from the last subflow's root
|
|
385
|
+
let subflowRoot;
|
|
386
|
+
for (const sfId of subflowPath) {
|
|
387
|
+
const subflow = (_a = fc.subflows) === null || _a === void 0 ? void 0 : _a[sfId];
|
|
388
|
+
if (!subflow)
|
|
389
|
+
return undefined;
|
|
390
|
+
subflowRoot = subflow.root;
|
|
391
|
+
}
|
|
392
|
+
if (!subflowRoot)
|
|
393
|
+
return undefined;
|
|
394
|
+
return this.dfsFind(subflowRoot, stageId);
|
|
395
|
+
}
|
|
396
|
+
/** DFS search for a node by ID in the StageNode graph. Cycle-safe via visited set. */
|
|
397
|
+
dfsFind(node, targetId, visited = new Set()) {
|
|
398
|
+
// Skip loop back-edge references (they share the target's ID but have no fn/resumeFn)
|
|
399
|
+
if (node.isLoopRef)
|
|
400
|
+
return undefined;
|
|
401
|
+
if (visited.has(node.id))
|
|
402
|
+
return undefined;
|
|
403
|
+
visited.add(node.id);
|
|
404
|
+
if (node.id === targetId)
|
|
405
|
+
return node;
|
|
406
|
+
if (node.children) {
|
|
407
|
+
for (const child of node.children) {
|
|
408
|
+
const found = this.dfsFind(child, targetId, visited);
|
|
409
|
+
if (found)
|
|
410
|
+
return found;
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
if (node.next)
|
|
414
|
+
return this.dfsFind(node.next, targetId, visited);
|
|
415
|
+
return undefined;
|
|
416
|
+
}
|
|
213
417
|
// ─── Recorder Management ───
|
|
214
418
|
/**
|
|
215
419
|
* Attach a scope Recorder to observe data operations (reads, writes, commits).
|
|
@@ -351,10 +555,30 @@ class FlowChartExecutor {
|
|
|
351
555
|
for (const r of this.scopeRecorders) {
|
|
352
556
|
(_b = r.clear) === null || _b === void 0 ? void 0 : _b.call(r);
|
|
353
557
|
}
|
|
558
|
+
this.lastCheckpoint = undefined;
|
|
354
559
|
this.traverser = this.createTraverser(signal, validatedInput, options === null || options === void 0 ? void 0 : options.env, options === null || options === void 0 ? void 0 : options.maxDepth);
|
|
355
560
|
try {
|
|
356
561
|
return await this.traverser.execute();
|
|
357
562
|
}
|
|
563
|
+
catch (error) {
|
|
564
|
+
if ((0, types_js_2.isPauseSignal)(error)) {
|
|
565
|
+
// Build checkpoint from current execution state
|
|
566
|
+
const snapshot = this.traverser.getSnapshot();
|
|
567
|
+
const sfResults = this.traverser.getSubflowResults();
|
|
568
|
+
this.lastCheckpoint = {
|
|
569
|
+
sharedState: snapshot.sharedState,
|
|
570
|
+
executionTree: snapshot.executionTree,
|
|
571
|
+
pausedStageId: error.stageId,
|
|
572
|
+
subflowPath: error.subflowPath,
|
|
573
|
+
pauseData: error.pauseData,
|
|
574
|
+
...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),
|
|
575
|
+
pausedAt: Date.now(),
|
|
576
|
+
};
|
|
577
|
+
// Return a PauseResult-shaped value so callers can check without try/catch
|
|
578
|
+
return { paused: true, checkpoint: this.lastCheckpoint };
|
|
579
|
+
}
|
|
580
|
+
throw error;
|
|
581
|
+
}
|
|
358
582
|
finally {
|
|
359
583
|
if (timeoutId !== undefined)
|
|
360
584
|
clearTimeout(timeoutId);
|
|
@@ -437,4 +661,4 @@ class FlowChartExecutor {
|
|
|
437
661
|
}
|
|
438
662
|
}
|
|
439
663
|
exports.FlowChartExecutor = FlowChartExecutor;
|
|
440
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Build a chart with flowChart() and pass the result here:\n *\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *\n *   // No-options form (uses auto-detected TypedScope factory from the chart):\n *   const executor = new FlowChartExecutor(chart);\n *\n *   // Options-object form (preferred when you need to customize behavior):\n *   const executor = new FlowChartExecutor(chart, { scopeFactory: myFactory, enrichSnapshots: true });\n *\n *   // 2-param form (accepts a ScopeFactory directly, for backward compatibility):\n *   const executor = new FlowChartExecutor(chart, myFactory);\n *\n *   const result = await executor.run({ input: data, env: { traceId: 'req-123' } });\n */\n\nimport type { CombinedNarrativeRecorderOptions } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { CombinedNarrativeEntry } from '../engine/narrative/narrativeTypes.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExtractorError,\n  type FlowChart,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { Recorder, RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\n/**\n * Options object for `FlowChartExecutor` — preferred over positional params.\n *\n * ```typescript\n * const ex = new FlowChartExecutor(chart, {\n *   scopeFactory: myFactory,\n *   enrichSnapshots: true,\n * });\n * ```\n *\n * **Sync note for maintainers:** Every field added here must also appear in the\n * `flowChartArgs` private field type and in the constructor's options-resolution\n * block (the `else if` branch that reads from `opts`). Missing any one of the\n * three causes silent omission — the option is accepted but never applied.\n *\n * **TScope inference note:** When using the options-object form with a custom scope,\n * TypeScript cannot infer `TScope` through the options object. Pass the type\n * explicitly: `new FlowChartExecutor<TOut, MyScope>(chart, { scopeFactory })`.\n */\nexport interface FlowChartExecutorOptions<TScope = any> {\n  // ── Common options (most callers need only these) ────────────────────────\n\n  /** Custom scope factory. Defaults to TypedScope or ScopeFacade auto-detection. */\n  scopeFactory?: ScopeFactory<TScope>;\n  /**\n   * Attach a per-stage scope snapshot to each extractor result. When `true`, the\n   * extraction callback receives the full shared state at the point that stage\n   * committed — useful for debugging multi-stage state transitions. Defaults to\n   * `false` (no scope snapshot attached). Can also be set on the chart via\n   * `flowChart(...).enrichSnapshots(true)`.\n   */\n  enrichSnapshots?: boolean;\n\n  // ── Context options ──────────────────────────────────────────────────────\n\n  /**\n   * Default values pre-populated into the shared context before **each** stage\n   * (re-applied every stage, acting as baseline defaults).\n   */\n  defaultValuesForContext?: unknown;\n  /**\n   * Initial context values merged into the shared context **once** at startup\n   * (applied before the first stage, not repeated on subsequent stages).\n   * Distinct from `defaultValuesForContext`, which is re-applied every stage.\n   */\n  initialContext?: unknown;\n  /** Read-only input accessible via `scope.getArgs()` — never tracked or written. */\n  readOnlyContext?: unknown;\n\n  // ── Advanced / escape-hatch options (most callers do not need these) ─────\n\n  /**\n   * Custom error classifier for throttling detection. Return `true` if the\n   * error represents a rate-limit or backpressure condition (the executor will\n   * treat it differently from hard failures). Defaults to no throttling classification.\n   */\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  /** Handlers for streaming stage lifecycle events (see `addStreamingFunction`). */\n  streamHandlers?: StreamHandlers;\n  /** Scope protection mode for TypedScope direct-assignment detection. */\n  scopeProtectionMode?: ScopeProtectionMode;\n}\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private narrativeOptions?: CombinedNarrativeRecorderOptions;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private scopeRecorders: Recorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n  // SYNC REQUIRED: every optional field here must mirror FlowChartExecutorOptions\n  // AND be assigned in the constructor's options-resolution block (the `else if` branch).\n  // Adding a field to only one of the three places causes silent omission.\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  /**\n   * Create a FlowChartExecutor.\n   *\n   * **Options object form** (preferred):\n   * ```typescript\n   * new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots: true })\n   * ```\n   *\n   * **2-param form** (also supported):\n   * ```typescript\n   * new FlowChartExecutor(chart, scopeFactory)\n   * ```\n   *\n   * @param flowChart - The compiled FlowChart returned by `flowChart(...).build()`\n   * @param factoryOrOptions - A `ScopeFactory<TScope>` OR a `FlowChartExecutorOptions<TScope>` options object.\n   */\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    factoryOrOptions?: ScopeFactory<TScope> | FlowChartExecutorOptions<TScope>,\n  ) {\n    // Detect options-object form vs factory form\n    let scopeFactory: ScopeFactory<TScope> | undefined;\n    let defaultValuesForContext: unknown;\n    let initialContext: unknown;\n    let readOnlyContext: unknown;\n    let throttlingErrorChecker: ((error: unknown) => boolean) | undefined;\n    let streamHandlers: StreamHandlers | undefined;\n    let scopeProtectionMode: ScopeProtectionMode | undefined;\n    let enrichSnapshots: boolean | undefined;\n\n    if (typeof factoryOrOptions === 'function') {\n      // 2-param form: new FlowChartExecutor(chart, scopeFactory)\n      scopeFactory = factoryOrOptions;\n    } else if (factoryOrOptions !== undefined) {\n      // Options object form: new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots, ... })\n      const opts = factoryOrOptions;\n      scopeFactory = opts.scopeFactory;\n      defaultValuesForContext = opts.defaultValuesForContext;\n      initialContext = opts.initialContext;\n      readOnlyContext = opts.readOnlyContext;\n      throttlingErrorChecker = opts.throttlingErrorChecker;\n      streamHandlers = opts.streamHandlers;\n      scopeProtectionMode = opts.scopeProtectionMode;\n      enrichSnapshots = opts.enrichSnapshots;\n    }\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? flowChart.scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n    maxDepth?: number,\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // ── Composed scope factory ─────────────────────────────────────────\n    // Collect all scope modifiers (recorders, redaction) into a single list,\n    // then create ONE factory that applies them in a loop. Replaces the\n    // previous 4-deep closure nesting with a flat, debuggable composition.\n\n    if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder(this.narrativeOptions);\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    this.sharedRedactedKeys = new Set<string>();\n    this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n    // Build modifier list — each modifier receives the scope after creation\n    type ScopeModifier = (scope: any) => void;\n    const modifiers: ScopeModifier[] = [];\n\n    // 1. Narrative recorder (if enabled)\n    if (this.combinedRecorder) {\n      const recorder = this.combinedRecorder;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') scope.attachRecorder(recorder);\n      });\n    }\n\n    // 2. User-provided scope recorders\n    if (this.scopeRecorders.length > 0) {\n      const recorders = this.scopeRecorders;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') {\n          for (const r of recorders) scope.attachRecorder(r);\n        }\n      });\n    }\n\n    // 3. Redaction policy (conditional — only when policy is set)\n    if (this.redactionPolicy) {\n      const policy = this.redactionPolicy;\n      modifiers.push((scope) => {\n        if (typeof scope.useRedactionPolicy === 'function') {\n          scope.useRedactionPolicy(policy);\n        }\n      });\n      // Pre-populate executor-level field redaction map from policy\n      // so getRedactionReport() includes field-level redactions.\n      if (policy.fields) {\n        for (const [key, fields] of Object.entries(policy.fields)) {\n          this.sharedRedactedFieldsByKey.set(key, new Set(fields));\n        }\n      }\n    }\n\n    // Compose: base factory + modifiers in a single pass.\n    // Shared redacted keys are ALWAYS wired up (unconditional — ensures cross-stage\n    // propagation even without a policy, because stages can call setValue(key, val, true)\n    // for per-call redaction). Optional modifiers (recorders, policy) are in the list.\n    const baseFactory = args.scopeFactory;\n    const sharedRedactedKeys = this.sharedRedactedKeys;\n    const scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n      const scope = baseFactory(ctx, stageName, readOnly, envArg);\n      // Always wire shared redaction state\n      if (typeof (scope as any).useSharedRedactedKeys === 'function') {\n        (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n      }\n      // Apply optional modifiers\n      for (const mod of modifiers) mod(scope);\n      return scope;\n    }) as ScopeFactory<TScope>;\n\n    const runtime = new ExecutionRuntime(fc.root.name, fc.root.id, args.defaultValuesForContext, args.initialContext);\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: fc.root,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n      ...(maxDepth !== undefined && { maxDepth }),\n    });\n  }\n\n  enableNarrative(options?: CombinedNarrativeRecorderOptions): void {\n    this.narrativeEnabled = true;\n    if (options) this.narrativeOptions = options;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── Recorder Management ───\n\n  /**\n   * Attach a scope Recorder to observe data operations (reads, writes, commits).\n   * Automatically attached to every ScopeFacade created during traversal.\n   * Must be called before run().\n   *\n   * **Idempotent by ID:** If a recorder with the same `id` is already attached,\n   * it is replaced (not duplicated). This prevents double-counting when both\n   * a framework and the user attach the same recorder type.\n   *\n   * Built-in recorders use auto-increment IDs (`metrics-1`, `debug-1`, ...) by\n   * default, so multiple instances with different configs coexist. To override\n   * a framework-attached recorder, pass the same well-known ID.\n   *\n   * @example\n   * ```typescript\n   * // Multiple recorders with different configs — each gets a unique ID\n   * executor.attachRecorder(new MetricRecorder());\n   * executor.attachRecorder(new DebugRecorder({ verbosity: 'minimal' }));\n   *\n   * // Override a framework-attached recorder by passing its well-known ID\n   * executor.attachRecorder(new MetricRecorder('metrics'));\n   *\n   * // Attaching twice with same ID replaces (no double-counting)\n   * executor.attachRecorder(new MetricRecorder('my-metrics'));\n   * executor.attachRecorder(new MetricRecorder('my-metrics')); // replaces previous\n   * ```\n   */\n  attachRecorder(recorder: Recorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== recorder.id);\n    this.scopeRecorders.push(recorder);\n  }\n\n  /** Detach all scope Recorders with the given ID. */\n  detachRecorder(id: string): void {\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached scope Recorders. */\n  getRecorders(): Recorder[] {\n    return [...this.scopeRecorders];\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   *\n   * **Idempotent by ID:** replaces existing recorder with same `id`.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== recorder.id);\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   *\n   * Sentences come from `NarrativeFlowRecorder` (a dedicated flow-only recorder automatically\n   * attached when narrative is enabled). It emits both `onStageExecuted` sentences (one per\n   * stage) AND `onNext` transition sentences (one per stage-to-stage transition), so for a\n   * chart with N stages you will typically get more entries here than from `getNarrative()`.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<TraversalResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // User-attached recorders (flowRecorders + scopeRecorders) are cleared via clear() to prevent\n    // cross-run accumulation. The combinedRecorder is NOT cleared here — createTraverser() always\n    // creates a fresh CombinedNarrativeRecorder instance on each run, so stale state is never an issue.\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n    for (const r of this.scopeRecorders) {\n      r.clear?.();\n    }\n\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env, options?.maxDepth);\n    try {\n      return await this.traverser.execute();\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.scopeRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  /** @internal */\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  /** @internal */\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  /** @internal */\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  /** @internal */\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  /** @internal */\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  /** @internal */\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  /** @internal */\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  /** @internal */\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}
|
|
664
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FlowChartExecutor.js","sourceRoot":"","sources":["../../../src/lib/runner/FlowChartExecutor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * FlowChartExecutor — Public API for executing a compiled FlowChart.\n *\n * Wraps FlowchartTraverser. Build a chart with flowChart() and pass the result here:\n *\n *   const chart = flowChart('entry', entryFn).addFunction('process', processFn).build();\n *\n *   // No-options form (uses auto-detected TypedScope factory from the chart):\n *   const executor = new FlowChartExecutor(chart);\n *\n *   // Options-object form (preferred when you need to customize behavior):\n *   const executor = new FlowChartExecutor(chart, { scopeFactory: myFactory, enrichSnapshots: true });\n *\n *   // 2-param form (accepts a ScopeFactory directly, for backward compatibility):\n *   const executor = new FlowChartExecutor(chart, myFactory);\n *\n *   const result = await executor.run({ input: data, env: { traceId: 'req-123' } });\n */\n\nimport type { CombinedNarrativeRecorderOptions } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { CombinedNarrativeRecorder } from '../engine/narrative/CombinedNarrativeRecorder.js';\nimport { NarrativeFlowRecorder } from '../engine/narrative/NarrativeFlowRecorder.js';\nimport type { CombinedNarrativeEntry } from '../engine/narrative/narrativeTypes.js';\nimport type { ManifestEntry } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport { ManifestFlowRecorder } from '../engine/narrative/recorders/ManifestFlowRecorder.js';\nimport type { FlowRecorder } from '../engine/narrative/types.js';\nimport { FlowchartTraverser } from '../engine/traversal/FlowchartTraverser.js';\nimport {\n  type ExecutorResult,\n  type ExtractorError,\n  type FlowChart,\n  type PausedResult,\n  type RunOptions,\n  type ScopeFactory,\n  type SerializedPipelineStructure,\n  type StageNode,\n  type StreamHandlers,\n  type SubflowResult,\n  type TraversalResult,\n  defaultLogger,\n} from '../engine/types.js';\nimport type { FlowchartCheckpoint } from '../pause/types.js';\nimport { isPauseSignal } from '../pause/types.js';\nimport type { ScopeProtectionMode } from '../scope/protection/types.js';\nimport { ScopeFacade } from '../scope/ScopeFacade.js';\nimport type { Recorder, RedactionPolicy, RedactionReport } from '../scope/types.js';\nimport { type RecorderSnapshot, type RuntimeSnapshot, ExecutionRuntime } from './ExecutionRuntime.js';\nimport { validateInput } from './validateInput.js';\n\n/** Default scope factory — creates a plain ScopeFacade for each stage. */\nconst defaultScopeFactory: ScopeFactory = (ctx, stageName, readOnly, env) =>\n  new ScopeFacade(ctx, stageName, readOnly, env);\n\n/**\n * Options object for `FlowChartExecutor` — preferred over positional params.\n *\n * ```typescript\n * const ex = new FlowChartExecutor(chart, {\n *   scopeFactory: myFactory,\n *   enrichSnapshots: true,\n * });\n * ```\n *\n * **Sync note for maintainers:** Every field added here must also appear in the\n * `flowChartArgs` private field type and in the constructor's options-resolution\n * block (the `else if` branch that reads from `opts`). Missing any one of the\n * three causes silent omission — the option is accepted but never applied.\n *\n * **TScope inference note:** When using the options-object form with a custom scope,\n * TypeScript cannot infer `TScope` through the options object. Pass the type\n * explicitly: `new FlowChartExecutor<TOut, MyScope>(chart, { scopeFactory })`.\n */\nexport interface FlowChartExecutorOptions<TScope = any> {\n  // ── Common options (most callers need only these) ────────────────────────\n\n  /** Custom scope factory. Defaults to TypedScope or ScopeFacade auto-detection. */\n  scopeFactory?: ScopeFactory<TScope>;\n  /**\n   * Attach a per-stage scope snapshot to each extractor result. When `true`, the\n   * extraction callback receives the full shared state at the point that stage\n   * committed — useful for debugging multi-stage state transitions. Defaults to\n   * `false` (no scope snapshot attached). Can also be set on the chart via\n   * `flowChart(...).enrichSnapshots(true)`.\n   */\n  enrichSnapshots?: boolean;\n\n  // ── Context options ──────────────────────────────────────────────────────\n\n  /**\n   * Default values pre-populated into the shared context before **each** stage\n   * (re-applied every stage, acting as baseline defaults).\n   */\n  defaultValuesForContext?: unknown;\n  /**\n   * Initial context values merged into the shared context **once** at startup\n   * (applied before the first stage, not repeated on subsequent stages).\n   * Distinct from `defaultValuesForContext`, which is re-applied every stage.\n   */\n  initialContext?: unknown;\n  /** Read-only input accessible via `scope.getArgs()` — never tracked or written. */\n  readOnlyContext?: unknown;\n\n  // ── Advanced / escape-hatch options (most callers do not need these) ─────\n\n  /**\n   * Custom error classifier for throttling detection. Return `true` if the\n   * error represents a rate-limit or backpressure condition (the executor will\n   * treat it differently from hard failures). Defaults to no throttling classification.\n   */\n  throttlingErrorChecker?: (error: unknown) => boolean;\n  /** Handlers for streaming stage lifecycle events (see `addStreamingFunction`). */\n  streamHandlers?: StreamHandlers;\n  /** Scope protection mode for TypedScope direct-assignment detection. */\n  scopeProtectionMode?: ScopeProtectionMode;\n}\n\nexport class FlowChartExecutor<TOut = any, TScope = any> {\n  private traverser: FlowchartTraverser<TOut, TScope>;\n  private narrativeEnabled = false;\n  private narrativeOptions?: CombinedNarrativeRecorderOptions;\n  private combinedRecorder: CombinedNarrativeRecorder | undefined;\n  private flowRecorders: FlowRecorder[] = [];\n  private scopeRecorders: Recorder[] = [];\n  private redactionPolicy: RedactionPolicy | undefined;\n  private sharedRedactedKeys = new Set<string>();\n  private sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n  private lastCheckpoint: FlowchartCheckpoint | undefined;\n\n  // SYNC REQUIRED: every optional field here must mirror FlowChartExecutorOptions\n  // AND be assigned in the constructor's options-resolution block (the `else if` branch).\n  // Adding a field to only one of the three places causes silent omission.\n  private readonly flowChartArgs: {\n    flowChart: FlowChart<TOut, TScope>;\n    scopeFactory: ScopeFactory<TScope>;\n    defaultValuesForContext?: unknown;\n    initialContext?: unknown;\n    readOnlyContext?: unknown;\n    throttlingErrorChecker?: (error: unknown) => boolean;\n    streamHandlers?: StreamHandlers;\n    scopeProtectionMode?: ScopeProtectionMode;\n    enrichSnapshots?: boolean;\n  };\n\n  /**\n   * Create a FlowChartExecutor.\n   *\n   * **Options object form** (preferred):\n   * ```typescript\n   * new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots: true })\n   * ```\n   *\n   * **2-param form** (also supported):\n   * ```typescript\n   * new FlowChartExecutor(chart, scopeFactory)\n   * ```\n   *\n   * @param flowChart - The compiled FlowChart returned by `flowChart(...).build()`\n   * @param factoryOrOptions - A `ScopeFactory<TScope>` OR a `FlowChartExecutorOptions<TScope>` options object.\n   */\n  constructor(\n    flowChart: FlowChart<TOut, TScope>,\n    factoryOrOptions?: ScopeFactory<TScope> | FlowChartExecutorOptions<TScope>,\n  ) {\n    // Detect options-object form vs factory form\n    let scopeFactory: ScopeFactory<TScope> | undefined;\n    let defaultValuesForContext: unknown;\n    let initialContext: unknown;\n    let readOnlyContext: unknown;\n    let throttlingErrorChecker: ((error: unknown) => boolean) | undefined;\n    let streamHandlers: StreamHandlers | undefined;\n    let scopeProtectionMode: ScopeProtectionMode | undefined;\n    let enrichSnapshots: boolean | undefined;\n\n    if (typeof factoryOrOptions === 'function') {\n      // 2-param form: new FlowChartExecutor(chart, scopeFactory)\n      scopeFactory = factoryOrOptions;\n    } else if (factoryOrOptions !== undefined) {\n      // Options object form: new FlowChartExecutor(chart, { scopeFactory, enrichSnapshots, ... })\n      const opts = factoryOrOptions;\n      scopeFactory = opts.scopeFactory;\n      defaultValuesForContext = opts.defaultValuesForContext;\n      initialContext = opts.initialContext;\n      readOnlyContext = opts.readOnlyContext;\n      throttlingErrorChecker = opts.throttlingErrorChecker;\n      streamHandlers = opts.streamHandlers;\n      scopeProtectionMode = opts.scopeProtectionMode;\n      enrichSnapshots = opts.enrichSnapshots;\n    }\n    this.flowChartArgs = {\n      flowChart,\n      scopeFactory: scopeFactory ?? flowChart.scopeFactory ?? (defaultScopeFactory as ScopeFactory<TScope>),\n      defaultValuesForContext,\n      initialContext,\n      readOnlyContext,\n      throttlingErrorChecker,\n      streamHandlers,\n      scopeProtectionMode,\n      enrichSnapshots,\n    };\n    this.traverser = this.createTraverser();\n  }\n\n  private createTraverser(\n    signal?: AbortSignal,\n    readOnlyContextOverride?: unknown,\n    env?: import('../engine/types').ExecutionEnv,\n    maxDepth?: number,\n    overrides?: {\n      root?: StageNode<TOut, TScope>;\n      initialContext?: unknown;\n      preserveRecorders?: boolean;\n      existingRuntime?: InstanceType<typeof ExecutionRuntime>;\n    },\n  ): FlowchartTraverser<TOut, TScope> {\n    const args = this.flowChartArgs;\n    const fc = args.flowChart;\n    const narrativeFlag = this.narrativeEnabled || (fc.enableNarrative ?? false);\n\n    // ── Composed scope factory ─────────────────────────────────────────\n    // Collect all scope modifiers (recorders, redaction) into a single list,\n    // then create ONE factory that applies them in a loop. Replaces the\n    // previous 4-deep closure nesting with a flat, debuggable composition.\n\n    if (overrides?.preserveRecorders) {\n      // Resume mode: keep existing combinedRecorder so narrative accumulates\n    } else if (narrativeFlag) {\n      this.combinedRecorder = new CombinedNarrativeRecorder(this.narrativeOptions);\n    } else {\n      this.combinedRecorder = undefined;\n    }\n\n    this.sharedRedactedKeys = new Set<string>();\n    this.sharedRedactedFieldsByKey = new Map<string, Set<string>>();\n\n    // Build modifier list — each modifier receives the scope after creation\n    type ScopeModifier = (scope: any) => void;\n    const modifiers: ScopeModifier[] = [];\n\n    // 1. Narrative recorder (if enabled)\n    if (this.combinedRecorder) {\n      const recorder = this.combinedRecorder;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') scope.attachRecorder(recorder);\n      });\n    }\n\n    // 2. User-provided scope recorders\n    if (this.scopeRecorders.length > 0) {\n      const recorders = this.scopeRecorders;\n      modifiers.push((scope) => {\n        if (typeof scope.attachRecorder === 'function') {\n          for (const r of recorders) scope.attachRecorder(r);\n        }\n      });\n    }\n\n    // 3. Redaction policy (conditional — only when policy is set)\n    if (this.redactionPolicy) {\n      const policy = this.redactionPolicy;\n      modifiers.push((scope) => {\n        if (typeof scope.useRedactionPolicy === 'function') {\n          scope.useRedactionPolicy(policy);\n        }\n      });\n      // Pre-populate executor-level field redaction map from policy\n      // so getRedactionReport() includes field-level redactions.\n      if (policy.fields) {\n        for (const [key, fields] of Object.entries(policy.fields)) {\n          this.sharedRedactedFieldsByKey.set(key, new Set(fields));\n        }\n      }\n    }\n\n    // Compose: base factory + modifiers in a single pass.\n    // Shared redacted keys are ALWAYS wired up (unconditional — ensures cross-stage\n    // propagation even without a policy, because stages can call setValue(key, val, true)\n    // for per-call redaction). Optional modifiers (recorders, policy) are in the list.\n    const baseFactory = args.scopeFactory;\n    const sharedRedactedKeys = this.sharedRedactedKeys;\n    const scopeFactory = ((ctx: any, stageName: string, readOnly?: unknown, envArg?: any) => {\n      const scope = baseFactory(ctx, stageName, readOnly, envArg);\n      // Always wire shared redaction state\n      if (typeof (scope as any).useSharedRedactedKeys === 'function') {\n        (scope as any).useSharedRedactedKeys(sharedRedactedKeys);\n      }\n      // Apply optional modifiers\n      for (const mod of modifiers) mod(scope);\n      return scope;\n    }) as ScopeFactory<TScope>;\n\n    const effectiveRoot = overrides?.root ?? fc.root;\n    const effectiveInitialContext = overrides?.initialContext ?? args.initialContext;\n\n    let runtime: ExecutionRuntime;\n    if (overrides?.existingRuntime) {\n      // Resume mode: reuse existing runtime so execution tree continues from pause point.\n      // Preserve the original root for getSnapshot() (full tree), then advance\n      // rootStageContext to a continuation from the leaf (for traversal).\n      runtime = overrides.existingRuntime;\n      runtime.preserveSnapshotRoot();\n      let leaf = runtime.rootStageContext;\n      while (leaf.next) leaf = leaf.next;\n      runtime.rootStageContext = leaf.createNext('', effectiveRoot.name, effectiveRoot.id);\n    } else {\n      runtime = new ExecutionRuntime(\n        effectiveRoot.name,\n        effectiveRoot.id,\n        args.defaultValuesForContext,\n        effectiveInitialContext,\n      );\n    }\n\n    return new FlowchartTraverser<TOut, TScope>({\n      root: effectiveRoot,\n      stageMap: fc.stageMap,\n      scopeFactory,\n      executionRuntime: runtime,\n      readOnlyContext: readOnlyContextOverride ?? args.readOnlyContext,\n      throttlingErrorChecker: args.throttlingErrorChecker,\n      streamHandlers: args.streamHandlers,\n      extractor: fc.extractor,\n      scopeProtectionMode: args.scopeProtectionMode,\n      subflows: fc.subflows,\n      enrichSnapshots: args.enrichSnapshots ?? fc.enrichSnapshots,\n      narrativeEnabled: narrativeFlag,\n      buildTimeStructure: fc.buildTimeStructure,\n      logger: fc.logger ?? defaultLogger,\n      signal,\n      executionEnv: env,\n      flowRecorders: this.buildFlowRecordersList(),\n      ...(maxDepth !== undefined && { maxDepth }),\n    });\n  }\n\n  enableNarrative(options?: CombinedNarrativeRecorderOptions): void {\n    this.narrativeEnabled = true;\n    if (options) this.narrativeOptions = options;\n  }\n\n  /**\n   * Set a declarative redaction policy that applies to all stages.\n   * Must be called before run().\n   */\n  setRedactionPolicy(policy: RedactionPolicy): void {\n    this.redactionPolicy = policy;\n  }\n\n  /**\n   * Returns a compliance-friendly report of all redaction activity from the\n   * most recent run. Never includes actual values.\n   */\n  getRedactionReport(): RedactionReport {\n    const fieldRedactions: Record<string, string[]> = {};\n    for (const [key, fields] of this.sharedRedactedFieldsByKey) {\n      fieldRedactions[key] = [...fields];\n    }\n    return {\n      redactedKeys: [...this.sharedRedactedKeys],\n      fieldRedactions,\n      patterns: (this.redactionPolicy?.patterns ?? []).map((p) => p.source),\n    };\n  }\n\n  // ─── Pause/Resume ───\n\n  /**\n   * Returns the checkpoint from the most recent paused execution, or `undefined`\n   * if the last run completed without pausing.\n   *\n   * The checkpoint is JSON-serializable — store it in Redis, Postgres, localStorage, etc.\n   *\n   * @example\n   * ```typescript\n   * const result = await executor.run({ input });\n   * if (executor.isPaused()) {\n   *   const checkpoint = executor.getCheckpoint()!;\n   *   await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n   * }\n   * ```\n   */\n  getCheckpoint(): FlowchartCheckpoint | undefined {\n    return this.lastCheckpoint;\n  }\n\n  /** Returns `true` if the most recent run() was paused (checkpoint available). */\n  isPaused(): boolean {\n    return this.lastCheckpoint !== undefined;\n  }\n\n  /**\n   * Resume a paused flowchart from a checkpoint.\n   *\n   * Restores the scope state, calls the paused stage's `resumeFn` with the\n   * provided input, then continues traversal from the next stage.\n   *\n   * The checkpoint can come from `getCheckpoint()` on a previous run, or from\n   * a serialized checkpoint stored in Redis/Postgres/localStorage.\n   *\n   * **Narrative/recorder state is reset on resume.** To keep a unified narrative\n   * across pause/resume cycles, collect it before calling resume.\n   *\n   * @example\n   * ```typescript\n   * // After a pause...\n   * const checkpoint = executor.getCheckpoint()!;\n   * await redis.set(`session:${id}`, JSON.stringify(checkpoint));\n   *\n   * // Later (possibly different server, same chart)\n   * const checkpoint = JSON.parse(await redis.get(`session:${id}`));\n   * const executor = new FlowChartExecutor(chart);\n   * const result = await executor.resume(checkpoint, { approved: true });\n   * ```\n   */\n  async resume(\n    checkpoint: FlowchartCheckpoint,\n    resumeInput?: unknown,\n    options?: Pick<RunOptions, 'signal' | 'env' | 'maxDepth'>,\n  ): Promise<ExecutorResult> {\n    this.lastCheckpoint = undefined;\n\n    // ── Validate checkpoint structure (may come from untrusted external storage) ──\n    if (\n      !checkpoint ||\n      typeof checkpoint !== 'object' ||\n      typeof checkpoint.sharedState !== 'object' ||\n      checkpoint.sharedState === null ||\n      Array.isArray(checkpoint.sharedState)\n    ) {\n      throw new Error('Invalid checkpoint: sharedState must be a plain object.');\n    }\n    if (typeof checkpoint.pausedStageId !== 'string' || checkpoint.pausedStageId === '') {\n      throw new Error('Invalid checkpoint: pausedStageId must be a non-empty string.');\n    }\n    if (\n      !Array.isArray(checkpoint.subflowPath) ||\n      !checkpoint.subflowPath.every((s: unknown) => typeof s === 'string')\n    ) {\n      throw new Error('Invalid checkpoint: subflowPath must be an array of strings.');\n    }\n\n    // Find the paused node in the graph\n    const pausedNode = this.findNodeInGraph(checkpoint.pausedStageId, checkpoint.subflowPath);\n    if (!pausedNode) {\n      throw new Error(\n        `Cannot resume: stage '${checkpoint.pausedStageId}' not found in flowchart. ` +\n          'The chart may have changed since the checkpoint was created.',\n      );\n    }\n    if (!pausedNode.resumeFn) {\n      throw new Error(\n        `Cannot resume: stage '${pausedNode.name}' (${pausedNode.id}) has no resumeFn. ` +\n          'Only stages created with addPausableFunction() can be resumed.',\n      );\n    }\n\n    // Build a synthetic resume node: calls resumeFn with resumeInput, then continues to original next.\n    // resumeFn signature is (scope, input) per PausableHandler — wrap to match StageFunction(scope, breakFn).\n    const resumeFn = pausedNode.resumeFn;\n    const resumeStageFn = (scope: TScope) => {\n      return resumeFn(scope, resumeInput);\n    };\n\n    const resumeNode: StageNode<TOut, TScope> = {\n      name: pausedNode.name,\n      id: pausedNode.id,\n      description: pausedNode.description,\n      fn: resumeStageFn,\n      next: pausedNode.next,\n    };\n\n    // Don't clear recorders — resume continues from previous state.\n    // Narrative, metrics, debug entries accumulate across pause/resume.\n\n    // Reuse the existing runtime so the execution tree continues from the pause point.\n    // preserveRecorders keeps the CombinedNarrativeRecorder so narrative accumulates.\n    const existingRuntime = this.traverser.getRuntime() as InstanceType<typeof ExecutionRuntime>;\n    this.traverser = this.createTraverser(options?.signal, undefined, options?.env, options?.maxDepth, {\n      root: resumeNode,\n      initialContext: checkpoint.sharedState,\n      preserveRecorders: true,\n      existingRuntime,\n    });\n\n    // Fire onResume event on all recorders (flow + scope)\n    const hasInput = resumeInput !== undefined;\n    const flowResumeEvent = {\n      stageName: pausedNode.name,\n      stageId: pausedNode.id,\n      hasInput,\n    };\n    if (this.combinedRecorder) this.combinedRecorder.onResume(flowResumeEvent);\n    for (const r of this.flowRecorders) r.onResume?.(flowResumeEvent);\n\n    const scopeResumeEvent = {\n      stageName: pausedNode.name,\n      stageId: pausedNode.id,\n      hasInput,\n      pipelineId: '',\n      timestamp: Date.now(),\n    };\n    for (const r of this.scopeRecorders) r.onResume?.(scopeResumeEvent);\n\n    try {\n      return await this.traverser.execute();\n    } catch (error: unknown) {\n      if (isPauseSignal(error)) {\n        const snapshot = this.traverser.getSnapshot();\n        const sfResults = this.traverser.getSubflowResults();\n        this.lastCheckpoint = {\n          sharedState: snapshot.sharedState,\n          executionTree: snapshot.executionTree,\n          pausedStageId: error.stageId,\n          subflowPath: error.subflowPath,\n          pauseData: error.pauseData,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          pausedAt: Date.now(),\n        };\n        return { paused: true, checkpoint: this.lastCheckpoint } satisfies PausedResult;\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Find a StageNode in the compiled graph by ID.\n   * Handles subflow paths by drilling into registered subflows.\n   */\n  private findNodeInGraph(stageId: string, subflowPath: readonly string[]): StageNode<TOut, TScope> | undefined {\n    const fc = this.flowChartArgs.flowChart;\n\n    if (subflowPath.length === 0) {\n      // Top-level: DFS from root\n      return this.dfsFind(fc.root, stageId);\n    }\n\n    // Subflow: drill into the subflow chain, then search from the last subflow's root\n    let subflowRoot: StageNode<TOut, TScope> | undefined;\n    for (const sfId of subflowPath) {\n      const subflow = fc.subflows?.[sfId];\n      if (!subflow) return undefined;\n      subflowRoot = subflow.root;\n    }\n    if (!subflowRoot) return undefined;\n    return this.dfsFind(subflowRoot, stageId);\n  }\n\n  /** DFS search for a node by ID in the StageNode graph. Cycle-safe via visited set. */\n  private dfsFind(\n    node: StageNode<TOut, TScope>,\n    targetId: string,\n    visited = new Set<string>(),\n  ): StageNode<TOut, TScope> | undefined {\n    // Skip loop back-edge references (they share the target's ID but have no fn/resumeFn)\n    if (node.isLoopRef) return undefined;\n    if (visited.has(node.id)) return undefined;\n    visited.add(node.id);\n    if (node.id === targetId) return node;\n    if (node.children) {\n      for (const child of node.children) {\n        const found = this.dfsFind(child, targetId, visited);\n        if (found) return found;\n      }\n    }\n    if (node.next) return this.dfsFind(node.next, targetId, visited);\n    return undefined;\n  }\n\n  // ─── Recorder Management ───\n\n  /**\n   * Attach a scope Recorder to observe data operations (reads, writes, commits).\n   * Automatically attached to every ScopeFacade created during traversal.\n   * Must be called before run().\n   *\n   * **Idempotent by ID:** If a recorder with the same `id` is already attached,\n   * it is replaced (not duplicated). This prevents double-counting when both\n   * a framework and the user attach the same recorder type.\n   *\n   * Built-in recorders use auto-increment IDs (`metrics-1`, `debug-1`, ...) by\n   * default, so multiple instances with different configs coexist. To override\n   * a framework-attached recorder, pass the same well-known ID.\n   *\n   * @example\n   * ```typescript\n   * // Multiple recorders with different configs — each gets a unique ID\n   * executor.attachRecorder(new MetricRecorder());\n   * executor.attachRecorder(new DebugRecorder({ verbosity: 'minimal' }));\n   *\n   * // Override a framework-attached recorder by passing its well-known ID\n   * executor.attachRecorder(new MetricRecorder('metrics'));\n   *\n   * // Attaching twice with same ID replaces (no double-counting)\n   * executor.attachRecorder(new MetricRecorder('my-metrics'));\n   * executor.attachRecorder(new MetricRecorder('my-metrics')); // replaces previous\n   * ```\n   */\n  attachRecorder(recorder: Recorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== recorder.id);\n    this.scopeRecorders.push(recorder);\n  }\n\n  /** Detach all scope Recorders with the given ID. */\n  detachRecorder(id: string): void {\n    this.scopeRecorders = this.scopeRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached scope Recorders. */\n  getRecorders(): Recorder[] {\n    return [...this.scopeRecorders];\n  }\n\n  // ─── FlowRecorder Management ───\n\n  /**\n   * Attach a FlowRecorder to observe control flow events.\n   * Automatically enables narrative if not already enabled.\n   * Must be called before run() — recorders are passed to the traverser at creation time.\n   *\n   * **Idempotent by ID:** replaces existing recorder with same `id`.\n   */\n  attachFlowRecorder(recorder: FlowRecorder): void {\n    // Replace existing recorder with same ID (idempotent — prevents double-counting)\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== recorder.id);\n    this.flowRecorders.push(recorder);\n    this.narrativeEnabled = true;\n  }\n\n  /** Detach all FlowRecorders with the given ID. */\n  detachFlowRecorder(id: string): void {\n    this.flowRecorders = this.flowRecorders.filter((r) => r.id !== id);\n  }\n\n  /** Returns a defensive copy of attached FlowRecorders. */\n  getFlowRecorders(): FlowRecorder[] {\n    return [...this.flowRecorders];\n  }\n\n  /**\n   * Returns the execution narrative.\n   *\n   * When using ScopeFacade-based scopes, returns a combined narrative that\n   * interleaves flow events (stages, decisions, forks) with data operations\n   * (reads, writes, updates). For plain scopes without attachRecorder support,\n   * returns flow-only narrative sentences.\n   */\n  getNarrative(): string[] {\n    // Combined recorder builds the narrative inline during traversal — just read it\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getNarrative();\n    }\n    return this.traverser.getNarrative();\n  }\n\n  /**\n   * Returns structured narrative entries for programmatic consumption.\n   * Each entry has a type (stage, step, condition, fork, etc.), text, and depth.\n   */\n  getNarrativeEntries(): CombinedNarrativeEntry[] {\n    if (this.combinedRecorder) {\n      return this.combinedRecorder.getEntries();\n    }\n    const flowSentences = this.traverser.getNarrative();\n    return flowSentences.map((text) => ({ type: 'stage' as const, text, depth: 0 }));\n  }\n\n  /**\n   * Returns the combined FlowRecorders list. When narrative is enabled, includes:\n   * - CombinedNarrativeRecorder (builds merged flow+data narrative inline)\n   * - NarrativeFlowRecorder (keeps flow-only sentences for getFlowNarrative())\n   * Plus any user-attached recorders.\n   */\n  private buildFlowRecordersList(): FlowRecorder[] | undefined {\n    const recorders: FlowRecorder[] = [];\n    if (this.combinedRecorder) {\n      recorders.push(this.combinedRecorder);\n      // Keep the default NarrativeFlowRecorder so getFlowNarrative() still works\n      recorders.push(new NarrativeFlowRecorder());\n    }\n    recorders.push(...this.flowRecorders);\n    return recorders.length > 0 ? recorders : undefined;\n  }\n\n  /**\n   * Returns flow-only narrative sentences (without data operations).\n   * Use this when you only want control flow descriptions.\n   *\n   * Sentences come from `NarrativeFlowRecorder` (a dedicated flow-only recorder automatically\n   * attached when narrative is enabled). It emits both `onStageExecuted` sentences (one per\n   * stage) AND `onNext` transition sentences (one per stage-to-stage transition), so for a\n   * chart with N stages you will typically get more entries here than from `getNarrative()`.\n   */\n  getFlowNarrative(): string[] {\n    return this.traverser.getNarrative();\n  }\n\n  async run(options?: RunOptions): Promise<ExecutorResult> {\n    let signal = options?.signal;\n    let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n    // Create an internal AbortController for timeoutMs\n    if (options?.timeoutMs && !signal) {\n      const controller = new AbortController();\n      signal = controller.signal;\n      timeoutId = setTimeout(\n        () => controller.abort(new Error(`Execution timed out after ${options.timeoutMs}ms`)),\n        options.timeoutMs,\n      );\n    }\n\n    // Validate input against inputSchema if both are present\n    let validatedInput = options?.input;\n    if (validatedInput && this.flowChartArgs.flowChart.inputSchema) {\n      validatedInput = validateInput(this.flowChartArgs.flowChart.inputSchema, validatedInput);\n    }\n\n    // User-attached recorders (flowRecorders + scopeRecorders) are cleared via clear() to prevent\n    // cross-run accumulation. The combinedRecorder is NOT cleared here — createTraverser() always\n    // creates a fresh CombinedNarrativeRecorder instance on each run, so stale state is never an issue.\n    for (const r of this.flowRecorders) {\n      r.clear?.();\n    }\n    for (const r of this.scopeRecorders) {\n      r.clear?.();\n    }\n\n    this.lastCheckpoint = undefined;\n    this.traverser = this.createTraverser(signal, validatedInput, options?.env, options?.maxDepth);\n    try {\n      return await this.traverser.execute();\n    } catch (error: unknown) {\n      if (isPauseSignal(error)) {\n        // Build checkpoint from current execution state\n        const snapshot = this.traverser.getSnapshot();\n        const sfResults = this.traverser.getSubflowResults();\n        this.lastCheckpoint = {\n          sharedState: snapshot.sharedState,\n          executionTree: snapshot.executionTree,\n          pausedStageId: error.stageId,\n          subflowPath: error.subflowPath,\n          pauseData: error.pauseData,\n          ...(sfResults.size > 0 && { subflowResults: Object.fromEntries(sfResults) }),\n          pausedAt: Date.now(),\n        };\n        // Return a PauseResult-shaped value so callers can check without try/catch\n        return { paused: true, checkpoint: this.lastCheckpoint } satisfies PausedResult;\n      }\n      throw error;\n    } finally {\n      if (timeoutId !== undefined) clearTimeout(timeoutId);\n    }\n  }\n\n  // ─── Introspection ───\n\n  getSnapshot(): RuntimeSnapshot {\n    const snapshot = this.traverser.getSnapshot() as RuntimeSnapshot;\n    const sfResults = this.traverser.getSubflowResults();\n    if (sfResults.size > 0) {\n      snapshot.subflowResults = Object.fromEntries(sfResults);\n    }\n\n    // Collect snapshot data from recorders that implement toSnapshot()\n    const recorderSnapshots: RecorderSnapshot[] = [];\n    for (const r of this.scopeRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    for (const r of this.flowRecorders) {\n      if (r.toSnapshot) {\n        const { name, data } = r.toSnapshot();\n        recorderSnapshots.push({ id: r.id, name, data });\n      }\n    }\n    if (recorderSnapshots.length > 0) {\n      snapshot.recorders = recorderSnapshots;\n    }\n\n    return snapshot;\n  }\n\n  /** @internal */\n  getRuntime() {\n    return this.traverser.getRuntime();\n  }\n\n  /** @internal */\n  setRootObject(path: string[], key: string, value: unknown): void {\n    this.traverser.setRootObject(path, key, value);\n  }\n\n  /** @internal */\n  getBranchIds() {\n    return this.traverser.getBranchIds();\n  }\n\n  /** @internal */\n  getRuntimeRoot(): StageNode {\n    return this.traverser.getRuntimeRoot();\n  }\n\n  /** @internal */\n  getRuntimeStructure(): SerializedPipelineStructure | undefined {\n    return this.traverser.getRuntimeStructure();\n  }\n\n  /** @internal */\n  getSubflowResults(): Map<string, SubflowResult> {\n    return this.traverser.getSubflowResults();\n  }\n\n  /** @internal */\n  getExtractedResults<TResult = unknown>(): Map<string, TResult> {\n    return this.traverser.getExtractedResults<TResult>();\n  }\n\n  /** @internal */\n  getExtractorErrors(): ExtractorError[] {\n    return this.traverser.getExtractorErrors();\n  }\n\n  /**\n   * Returns the subflow manifest from an attached ManifestFlowRecorder.\n   * Returns empty array if no ManifestFlowRecorder is attached.\n   */\n  getSubflowManifest(): ManifestEntry[] {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getManifest() ?? [];\n  }\n\n  /**\n   * Returns the full spec for a dynamically-registered subflow.\n   * Requires an attached ManifestFlowRecorder that observed the registration.\n   */\n  getSubflowSpec(subflowId: string): unknown | undefined {\n    const recorder = this.flowRecorders.find((r) => r instanceof ManifestFlowRecorder) as\n      | ManifestFlowRecorder\n      | undefined;\n    return recorder?.getSpec(subflowId);\n  }\n}\n"]}
|