footprintjs 4.6.0 → 4.7.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/AGENTS.md CHANGED
@@ -16,12 +16,13 @@ src/lib/
16
16
  ├── scope/ → Per-stage facades + recorders + providers
17
17
  ├── reactive/ → TypedScope<T> deep Proxy (typed property access, $-methods)
18
18
  ├── decide/ → decide()/select() decision evidence capture
19
+ ├── recorder/ → CompositeRecorder, KeyedRecorder<T> base class
19
20
  ├── engine/ → DFS traversal + narrative + handlers
20
21
  ├── runner/ → FlowChartExecutor
21
22
  └── contract/ → I/O schema + OpenAPI
22
23
  ```
23
24
 
24
- Entry points: `footprintjs` (public) and `footprintjs/advanced` (internals).
25
+ Entry points: `footprintjs` (public), `footprintjs/trace` (execution tracing), `footprintjs/advanced` (internals).
25
26
 
26
27
  ## Key API — TypedScope (Recommended)
27
28
 
@@ -108,6 +109,21 @@ executor.setRedactionPolicy({ keys, patterns, fields })
108
109
  - 8 built-in FlowRecorder strategies
109
110
  - Narrative via `executor.recorder(narrative())` at runtime
110
111
 
112
+ ## Execution Tracing (`footprintjs/trace`)
113
+
114
+ Every stage gets a unique `runtimeStageId`: `[subflowPath/]stageId#executionIndex` (e.g., `call-llm#5`, `sf-tools/execute-tool-calls#8`)
115
+
116
+ Use for: debugging (which stage wrote what?), backtracking (who changed a value?), custom recorder storage.
117
+
118
+ ```typescript
119
+ import { parseRuntimeStageId, findLastWriter, findCommit, KeyedRecorder } from 'footprintjs/trace';
120
+
121
+ // commitLog from executor.getSnapshot().commitLog (CommitBundle[])
122
+ // findCommit(commitLog, stageId, key?) → CommitBundle | undefined
123
+ // findLastWriter(commitLog, key, beforeIdx?) → CommitBundle | undefined — search backwards
124
+ // KeyedRecorder<T> — abstract base: store(runtimeStageId, entry), getByKey(), getMap(), values()
125
+ ```
126
+
111
127
  ## Rules
112
128
 
113
129
  - Use `flowChart<T>()` — scopeFactory is auto-embedded
package/CLAUDE.md CHANGED
@@ -16,7 +16,7 @@ src/lib/
16
16
  ├── scope/ → Per-stage facades + recorders + providers
17
17
  ├── reactive/ → TypedScope<T> deep Proxy (typed property access, $-methods, cycle-safe)
18
18
  ├── decide/ → decide()/select() decision evidence capture (filter + function)
19
- ├── recorder/ → CompositeRecorder composition primitives (domain presets)
19
+ ├── recorder/ → CompositeRecorder, KeyedRecorder<T> base class, composition primitives
20
20
  ├── pause/ → Pause/Resume (PauseSignal, FlowchartCheckpoint, PausableHandler)
21
21
  ├── engine/ → DFS traversal + narrative + 13 handlers
22
22
  ├── runner/ → High-level executor (FlowChartExecutor)
@@ -25,9 +25,10 @@ src/lib/
25
25
 
26
26
  Dependency DAG: `memory <- scope <- reactive <- engine <- runner`, `schema <- engine`, `builder (standalone) -> engine`, `contract <- schema`, `decide -> scope`
27
27
 
28
- Two entry points:
28
+ Three entry points:
29
29
  - `import { ... } from 'footprintjs'` — public API
30
- - `import { ... } from 'footprintjs/advanced'` — internals
30
+ - `import { ... } from 'footprintjs/trace'` — execution tracing: runtimeStageId, commitLog queries, KeyedRecorder
31
+ - `import { ... } from 'footprintjs/advanced'` — engine internals (also re-exports trace)
31
32
 
32
33
  ## Key API
33
34
 
@@ -218,6 +219,74 @@ Both use `{ id, hooks } -> dispatcher -> error isolation -> attach/detach`. Inte
218
219
  6. FlowRecorder.onNext/onDecision/onFork — control flow continues
219
220
  ```
220
221
 
222
+ ## Execution Tracing (`footprintjs/trace`)
223
+
224
+ Every stage execution gets a unique `runtimeStageId` — the universal key that links recorder events, commit log entries, and execution tree nodes.
225
+
226
+ **When to use:** Debugging (which stage set a value to something unexpected?), audit trails (trace every write to its source stage), custom recorders (correlate events with specific execution steps), quality trace backtracking (walk backwards to find where data quality dropped).
227
+
228
+ **Format:** `[subflowPath/]stageId#executionIndex`
229
+
230
+ ```
231
+ seed#0 — root stage
232
+ call-llm#5 — 5th execution step
233
+ sf-tools/execute-tool-calls#8 — subflow stage
234
+ call-llm#9 — same stageId, different execution (loop)
235
+ ```
236
+
237
+ **The commitLog:** An ordered array of `CommitBundle` — one per stage commit, recording what each stage wrote to shared state. Get it from `executor.getSnapshot().commitLog`.
238
+
239
+ ```typescript
240
+ import { parseRuntimeStageId, findLastWriter, findCommit } from 'footprintjs/trace';
241
+
242
+ // Parse a runtimeStageId into components
243
+ parseRuntimeStageId('sf-tools/execute-tool-calls#8');
244
+ // → { stageId: 'execute-tool-calls', executionIndex: 8, subflowPath: 'sf-tools' }
245
+
246
+ // Get the commit log after execution
247
+ const snapshot = executor.getSnapshot();
248
+ const commitLog = snapshot.commitLog; // CommitBundle[]
249
+
250
+ // Backtrack: who last wrote 'systemPrompt' before commitLog array index 8?
251
+ // beforeIdx is the CommitBundle.idx (array position), NOT the executionIndex from runtimeStageId.
252
+ const writer = findLastWriter(commitLog, 'systemPrompt', 8);
253
+ // → CommitBundle | undefined (has .stage, .stageId, .runtimeStageId, .trace, .overwrite, .updates)
254
+
255
+ // Find by stageId: use findCommit when you know the stage.
256
+ // Use findLastWriter when you know the key but not which stage wrote it.
257
+ const llmCommit = findCommit(commitLog, 'call-llm', 'adapterRawResponse');
258
+ ```
259
+
260
+ **Exports from `footprintjs/trace`:**
261
+
262
+ | Export | Returns | Use |
263
+ |--------|---------|-----|
264
+ | `buildRuntimeStageId(stageId, idx, subflowPath?)` | `string` | Construct an ID from components |
265
+ | `parseRuntimeStageId(id)` | `{ stageId, executionIndex, subflowPath }` | Decompose an ID |
266
+ | `findCommit(commitLog, stageId, key?)` | `CommitBundle \| undefined` | Find first commit by stageId |
267
+ | `findCommits(commitLog, stageId)` | `CommitBundle[]` | Find all commits by stageId |
268
+ | `findLastWriter(commitLog, key, beforeIdx?)` | `CommitBundle \| undefined` | Search backwards for who wrote a key |
269
+ | `KeyedRecorder<T>` | abstract class | Base for Map-based recorders |
270
+
271
+ **KeyedRecorder<T>** — abstract base class for recorders that store data as `Map<runtimeStageId, T>`:
272
+
273
+ ```typescript
274
+ import { KeyedRecorder } from 'footprintjs/trace';
275
+
276
+ class MyRecorder extends KeyedRecorder<MyEntry> {
277
+ readonly id = 'my-recorder'; // required (abstract)
278
+ onSomeEvent(event) {
279
+ this.store(event.runtimeStageId, { ... }); // protected
280
+ }
281
+ }
282
+ recorder.getByKey('call-llm#5'); // O(1) lookup
283
+ recorder.getMap(); // ReadonlyMap
284
+ recorder.values(); // MyEntry[] in insertion order
285
+ recorder.clear(); // reset
286
+ ```
287
+
288
+ **How runtimeStageId is generated:** A counter starts at 0 and increments by 1 for each stage execution across the entire run, including subflow stages. Subflow child traversers share the parent counter so indices are globally unique. Stages inside subflows have stageIds already prefixed by the builder (e.g., `sf-tools/execute-tool-calls`), so `buildRuntimeStageId` just appends `#index`.
289
+
221
290
  ## Anti-Patterns
222
291
 
223
292
  - Never post-process the tree — use recorders
package/dist/advanced.js CHANGED
@@ -22,7 +22,7 @@
22
22
  */
23
23
  Object.defineProperty(exports, "__esModule", { value: true });
24
24
  exports.createSubflowHandlerDeps = exports.ContinuationResolver = exports.computeNodeType = exports.ChildrenExecutor = exports.applyOutputMapping = exports.NullControlFlowNarrativeGenerator = exports.isStageNodeReturn = exports.FlowchartTraverser = exports.shouldWrapWithProxy = exports.SCOPE_METHOD_NAMES = exports.joinPath = exports.createArrayProxy = exports.buildNestedPatch = exports.BREAK_SETTER = exports.ExecutionRuntime = exports.ZodScopeResolver = exports.isScopeSchema = exports.defineScopeSchema = exports.createScopeProxyFromZod = exports.toScopeFactory = exports.resolveScopeProvider = exports.registerScopeResolver = exports.makeFactoryProvider = exports.makeClassProvider = exports.looksLikeFactory = exports.looksLikeClassCtor = exports.isSubclassOfScopeFacade = exports.attachScopeMethods = exports.ScopeFacade = exports.createProtectedScope = exports.createErrorMessage = exports.createTypedScopeFactory = exports.specToStageNode = exports.SelectorFnList = exports.DeciderList = exports.ArrayMergeMode = exports.updateValue = exports.updateNestedValue = exports.setNestedValue = exports.redactPatch = exports.normalisePath = exports.getRunAndGlobalPaths = exports.getNestedValue = exports.deepSmartMerge = exports.applySmartMerge = exports.DiagnosticCollector = exports.TransactionBuffer = exports.EventLog = exports.StageContext = exports.SharedMemory = void 0;
25
- exports.SubflowExecutor = exports.StageRunner = exports.SelectorHandler = exports.seedSubflowGlobalStore = exports.RuntimeStructureManager = exports.NodeResolver = exports.getInitialScopeValues = exports.extractParentScopeValues = exports.ExtractorRunner = exports.DEFAULT_MAX_ITERATIONS = exports.DeciderHandler = void 0;
25
+ exports.KeyedRecorder = exports.findLastWriter = exports.findCommits = exports.findCommit = exports.parseRuntimeStageId = exports.createExecutionCounter = exports.buildRuntimeStageId = exports.SubflowExecutor = exports.StageRunner = exports.SelectorHandler = exports.seedSubflowGlobalStore = exports.RuntimeStructureManager = exports.NodeResolver = exports.getInitialScopeValues = exports.extractParentScopeValues = exports.ExtractorRunner = exports.DEFAULT_MAX_ITERATIONS = exports.DeciderHandler = void 0;
26
26
  var index_js_1 = require("./lib/memory/index.js");
27
27
  Object.defineProperty(exports, "SharedMemory", { enumerable: true, get: function () { return index_js_1.SharedMemory; } });
28
28
  var index_js_2 = require("./lib/memory/index.js");
@@ -102,4 +102,14 @@ Object.defineProperty(exports, "seedSubflowGlobalStore", { enumerable: true, get
102
102
  Object.defineProperty(exports, "SelectorHandler", { enumerable: true, get: function () { return index_js_16.SelectorHandler; } });
103
103
  Object.defineProperty(exports, "StageRunner", { enumerable: true, get: function () { return index_js_16.StageRunner; } });
104
104
  Object.defineProperty(exports, "SubflowExecutor", { enumerable: true, get: function () { return index_js_16.SubflowExecutor; } });
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"advanced.js","sourceRoot":"","sources":["../src/advanced.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;GAYG;AACH;;;;;;;GAOG;;;;AAcH,kDAAqD;AAA5C,wGAAA,YAAY,OAAA;AACrB,kDAAqD;AAA5C,wGAAA,YAAY,OAAA;AACrB,kDAAiD;AAAxC,oGAAA,QAAQ,OAAA;AACjB,kDAA0D;AAAjD,6GAAA,iBAAiB,OAAA;AAC1B,kDAA4D;AAAnD,+GAAA,mBAAmB,OAAA;AAC5B,kDAU+B;AAT7B,2GAAA,eAAe,OAAA;AACf,0GAAA,cAAc,OAAA;AACd,0GAAA,cAAc,OAAA;AACd,gHAAA,oBAAoB,OAAA;AACpB,yGAAA,aAAa,OAAA;AACb,uGAAA,WAAW,OAAA;AACX,0GAAA,cAAc,OAAA;AACd,6GAAA,iBAAiB,OAAA;AACjB,uGAAA,WAAW,OAAA;AAwBb,mDAAsG;AAA7F,0GAAA,cAAc,OAAA;AAAE,uGAAA,WAAW,OAAA;AAAE,0GAAA,cAAc,OAAA;AAAE,2GAAA,eAAe,OAAA;AACrE,qEAA0E;AAAjE,4HAAA,uBAAuB,OAAA;AAahC,iDAA6F;AAApF,8GAAA,kBAAkB,OAAA;AAAE,gHAAA,oBAAoB,OAAA;AAAE,uGAAA,WAAW,OAAA;AAC9D,iDAU8B;AAT5B,8GAAA,kBAAkB,OAAA;AAClB,mHAAA,uBAAuB,OAAA;AACvB,8GAAA,kBAAkB,OAAA;AAClB,4GAAA,gBAAgB,OAAA;AAChB,6GAAA,iBAAiB,OAAA;AACjB,+GAAA,mBAAmB,OAAA;AACnB,iHAAA,qBAAqB,OAAA;AACrB,gHAAA,oBAAoB,OAAA;AACpB,0GAAA,cAAc,OAAA;AAehB,gBAAgB;AAChB,kDAAmH;AAA1G,oHAAA,uBAAuB,OAAA;AAAE,8GAAA,iBAAiB,OAAA;AAAE,0GAAA,aAAa,OAAA;AAAE,6GAAA,gBAAgB,OAAA;AAOpF,mDAAyD;AAAhD,6GAAA,gBAAgB,OAAA;AAOzB,qDAOiC;AAN/B,yGAAA,YAAY,OAAA;AACZ,6GAAA,gBAAgB,OAAA;AAChB,6GAAA,gBAAgB,OAAA;AAChB,qGAAA,QAAQ,OAAA;AACR,+GAAA,kBAAkB,OAAA;AAClB,gHAAA,mBAAmB,OAAA;AASrB,mDAA2D;AAAlD,+GAAA,kBAAkB,OAAA;AAC3B,mDAA0D;AAAjD,8GAAA,iBAAiB,OAAA;AAsB1B,mDAA0E;AAAjE,8HAAA,iCAAiC,OAAA;AAI1C,mDAiB+B;AAhB7B,+GAAA,kBAAkB,OAAA;AAClB,6GAAA,gBAAgB,OAAA;AAChB,4GAAA,eAAe,OAAA;AACf,iHAAA,oBAAoB,OAAA;AACpB,qHAAA,wBAAwB,OAAA;AACxB,2GAAA,cAAc,OAAA;AACd,mHAAA,sBAAsB,OAAA;AACtB,4GAAA,eAAe,OAAA;AACf,qHAAA,wBAAwB,OAAA;AACxB,kHAAA,qBAAqB,OAAA;AACrB,yGAAA,YAAY,OAAA;AACZ,oHAAA,uBAAuB,OAAA;AACvB,mHAAA,sBAAsB,OAAA;AACtB,4GAAA,eAAe,OAAA;AACf,wGAAA,WAAW,OAAA;AACX,4GAAA,eAAe,OAAA","sourcesContent":["/**\n * footprintjs/advanced — Low-level internals for custom execution engines and testing.\n *\n * Most users never need this. Use `footprintjs` (main) instead.\n * This entry point exposes `SharedMemory`, `StageContext`, `FlowchartTraverser`,\n * and other primitives that power the engine.\n *\n * ```ts\n * import { SharedMemory, StageContext } from 'footprintjs/advanced';\n * ```\n *\n * @module advanced\n */\n/**\n * FootPrint — Advanced / Internal API\n *\n * These exports are for advanced use cases, testing, and building\n * custom execution engines. Most users should use the main 'footprint' entry point.\n *\n * Import via: import { ... } from 'footprint/advanced'\n */\n\n// ============================================================================\n// Memory — Low-level transactional state primitives\n// ============================================================================\n\nexport type {\n  CommitBundle,\n  FlowControlType,\n  FlowMessage,\n  MemoryPatch,\n  StageSnapshot,\n  TraceEntry,\n} from './lib/memory/index.js';\nexport { SharedMemory } from './lib/memory/index.js';\nexport { StageContext } from './lib/memory/index.js';\nexport { EventLog } from './lib/memory/index.js';\nexport { TransactionBuffer } from './lib/memory/index.js';\nexport { DiagnosticCollector } from './lib/memory/index.js';\nexport {\n  applySmartMerge,\n  deepSmartMerge,\n  getNestedValue,\n  getRunAndGlobalPaths,\n  normalisePath,\n  redactPatch,\n  setNestedValue,\n  updateNestedValue,\n  updateValue,\n} from './lib/memory/index.js';\n\n// ============================================================================\n// Builder — Types and internals\n// ============================================================================\n\nexport type {\n  BuildTimeExtractor,\n  BuildTimeNodeMetadata,\n  ExecOptions,\n  FlowChartSpec,\n  ILogger,\n  ScopeProtectionMode,\n  SerializedPipelineStructure,\n  SimplifiedParallelSpec,\n  StageFn,\n  StageNode,\n  StreamCallback,\n  StreamLifecycleHandler,\n  StreamTokenHandler,\n  SubflowMountOptions,\n  SubflowRef,\n} from './lib/builder/index.js';\nexport { ArrayMergeMode, DeciderList, SelectorFnList, specToStageNode } from './lib/builder/index.js';\nexport { createTypedScopeFactory } from './lib/builder/typedFlowChart.js';\n\n// ============================================================================\n// Scope — Providers, protection, recorder options, and event types\n// ============================================================================\n\nexport type {\n  ProviderResolver,\n  ResolveOptions,\n  ScopeProvider,\n  StageContextLike,\n  StrictMode,\n} from './lib/scope/index.js';\nexport { createErrorMessage, createProtectedScope, ScopeFacade } from './lib/scope/index.js';\nexport {\n  attachScopeMethods,\n  isSubclassOfScopeFacade,\n  looksLikeClassCtor,\n  looksLikeFactory,\n  makeClassProvider,\n  makeFactoryProvider,\n  registerScopeResolver,\n  resolveScopeProvider,\n  toScopeFactory,\n} from './lib/scope/index.js';\n\n// Recorder config/option types\nexport type {\n  AggregatedMetrics,\n  DebugEntry,\n  DebugRecorderOptions,\n  DebugVerbosity,\n  DefineScopeOptions,\n  RecorderContext,\n  StageEvent,\n  StageMetrics,\n} from './lib/scope/index.js';\n\n// Zod internals\nexport { createScopeProxyFromZod, defineScopeSchema, isScopeSchema, ZodScopeResolver } from './lib/scope/index.js';\n\n// ============================================================================\n// Runner — Internals\n// ============================================================================\n\nexport type { RuntimeSnapshot } from './lib/runner/index.js';\nexport { ExecutionRuntime } from './lib/runner/index.js';\n\n// ============================================================================\n// Reactive — TypedScope internals (for custom proxy implementations)\n// ============================================================================\n\nexport type { ReactiveOptions, ReactiveTarget } from './lib/reactive/index.js';\nexport {\n  BREAK_SETTER,\n  buildNestedPatch,\n  createArrayProxy,\n  joinPath,\n  SCOPE_METHOD_NAMES,\n  shouldWrapWithProxy,\n} from './lib/reactive/index.js';\n\n// ============================================================================\n// Engine — DFS graph traversal internals\n// ============================================================================\n\nexport type { TraverserOptions } from './lib/engine/index.js';\nexport type { Decider } from './lib/engine/index.js';\nexport { FlowchartTraverser } from './lib/engine/index.js';\nexport { isStageNodeReturn } from './lib/engine/index.js';\n\n// Narrative internals\nexport type { IControlFlowNarrative } from './lib/engine/index.js';\nexport type { CombinedNarrativeEntry, CombinedNarrativeOptions } from './lib/engine/index.js';\nexport type {\n  BranchResult,\n  BranchResults,\n  SerializedPipelineStructure as EngineSerializedPipelineStructure,\n  StageSnapshot as EngineStageSnapshot,\n  ExtractorError,\n  HandlerDeps,\n  IExecutionRuntime,\n  NodeResultType,\n  RuntimeStructureMetadata,\n  ScopeFactory,\n  SerializedPipelineNode,\n  StageFunction,\n  SubflowResult,\n  TraversalExtractor,\n  TraversalResult,\n} from './lib/engine/index.js';\nexport { NullControlFlowNarrativeGenerator } from './lib/engine/index.js';\n\n// Handlers (testing / custom engines)\nexport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './lib/engine/index.js';\nexport {\n  applyOutputMapping,\n  ChildrenExecutor,\n  computeNodeType,\n  ContinuationResolver,\n  createSubflowHandlerDeps,\n  DeciderHandler,\n  DEFAULT_MAX_ITERATIONS,\n  ExtractorRunner,\n  extractParentScopeValues,\n  getInitialScopeValues,\n  NodeResolver,\n  RuntimeStructureManager,\n  seedSubflowGlobalStore,\n  SelectorHandler,\n  StageRunner,\n  SubflowExecutor,\n} from './lib/engine/index.js';\n"]}
105
+ var runtimeStageId_js_1 = require("./lib/engine/runtimeStageId.js");
106
+ Object.defineProperty(exports, "buildRuntimeStageId", { enumerable: true, get: function () { return runtimeStageId_js_1.buildRuntimeStageId; } });
107
+ Object.defineProperty(exports, "createExecutionCounter", { enumerable: true, get: function () { return runtimeStageId_js_1.createExecutionCounter; } });
108
+ Object.defineProperty(exports, "parseRuntimeStageId", { enumerable: true, get: function () { return runtimeStageId_js_1.parseRuntimeStageId; } });
109
+ var commitLogUtils_js_1 = require("./lib/memory/commitLogUtils.js");
110
+ Object.defineProperty(exports, "findCommit", { enumerable: true, get: function () { return commitLogUtils_js_1.findCommit; } });
111
+ Object.defineProperty(exports, "findCommits", { enumerable: true, get: function () { return commitLogUtils_js_1.findCommits; } });
112
+ Object.defineProperty(exports, "findLastWriter", { enumerable: true, get: function () { return commitLogUtils_js_1.findLastWriter; } });
113
+ var KeyedRecorder_js_1 = require("./lib/recorder/KeyedRecorder.js");
114
+ Object.defineProperty(exports, "KeyedRecorder", { enumerable: true, get: function () { return KeyedRecorder_js_1.KeyedRecorder; } });
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"advanced.js","sourceRoot":"","sources":["../src/advanced.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;GAYG;AACH;;;;;;;GAOG;;;;AAcH,kDAAqD;AAA5C,wGAAA,YAAY,OAAA;AACrB,kDAAqD;AAA5C,wGAAA,YAAY,OAAA;AACrB,kDAAiD;AAAxC,oGAAA,QAAQ,OAAA;AACjB,kDAA0D;AAAjD,6GAAA,iBAAiB,OAAA;AAC1B,kDAA4D;AAAnD,+GAAA,mBAAmB,OAAA;AAC5B,kDAU+B;AAT7B,2GAAA,eAAe,OAAA;AACf,0GAAA,cAAc,OAAA;AACd,0GAAA,cAAc,OAAA;AACd,gHAAA,oBAAoB,OAAA;AACpB,yGAAA,aAAa,OAAA;AACb,uGAAA,WAAW,OAAA;AACX,0GAAA,cAAc,OAAA;AACd,6GAAA,iBAAiB,OAAA;AACjB,uGAAA,WAAW,OAAA;AAwBb,mDAAsG;AAA7F,0GAAA,cAAc,OAAA;AAAE,uGAAA,WAAW,OAAA;AAAE,0GAAA,cAAc,OAAA;AAAE,2GAAA,eAAe,OAAA;AACrE,qEAA0E;AAAjE,4HAAA,uBAAuB,OAAA;AAahC,iDAA6F;AAApF,8GAAA,kBAAkB,OAAA;AAAE,gHAAA,oBAAoB,OAAA;AAAE,uGAAA,WAAW,OAAA;AAC9D,iDAU8B;AAT5B,8GAAA,kBAAkB,OAAA;AAClB,mHAAA,uBAAuB,OAAA;AACvB,8GAAA,kBAAkB,OAAA;AAClB,4GAAA,gBAAgB,OAAA;AAChB,6GAAA,iBAAiB,OAAA;AACjB,+GAAA,mBAAmB,OAAA;AACnB,iHAAA,qBAAqB,OAAA;AACrB,gHAAA,oBAAoB,OAAA;AACpB,0GAAA,cAAc,OAAA;AAehB,gBAAgB;AAChB,kDAAmH;AAA1G,oHAAA,uBAAuB,OAAA;AAAE,8GAAA,iBAAiB,OAAA;AAAE,0GAAA,aAAa,OAAA;AAAE,6GAAA,gBAAgB,OAAA;AAOpF,mDAAyD;AAAhD,6GAAA,gBAAgB,OAAA;AAOzB,qDAOiC;AAN/B,yGAAA,YAAY,OAAA;AACZ,6GAAA,gBAAgB,OAAA;AAChB,6GAAA,gBAAgB,OAAA;AAChB,qGAAA,QAAQ,OAAA;AACR,+GAAA,kBAAkB,OAAA;AAClB,gHAAA,mBAAmB,OAAA;AASrB,mDAA2D;AAAlD,+GAAA,kBAAkB,OAAA;AAC3B,mDAA0D;AAAjD,8GAAA,iBAAiB,OAAA;AAsB1B,mDAA0E;AAAjE,8HAAA,iCAAiC,OAAA;AAI1C,mDAiB+B;AAhB7B,+GAAA,kBAAkB,OAAA;AAClB,6GAAA,gBAAgB,OAAA;AAChB,4GAAA,eAAe,OAAA;AACf,iHAAA,oBAAoB,OAAA;AACpB,qHAAA,wBAAwB,OAAA;AACxB,2GAAA,cAAc,OAAA;AACd,mHAAA,sBAAsB,OAAA;AACtB,4GAAA,eAAe,OAAA;AACf,qHAAA,wBAAwB,OAAA;AACxB,kHAAA,qBAAqB,OAAA;AACrB,yGAAA,YAAY,OAAA;AACZ,oHAAA,uBAAuB,OAAA;AACvB,mHAAA,sBAAsB,OAAA;AACtB,4GAAA,eAAe,OAAA;AACf,wGAAA,WAAW,OAAA;AACX,4GAAA,eAAe,OAAA;AAKjB,oEAAkH;AAAzG,wHAAA,mBAAmB,OAAA;AAAE,2HAAA,sBAAsB,OAAA;AAAE,wHAAA,mBAAmB,OAAA;AACzE,oEAAyF;AAAhF,+GAAA,UAAU,OAAA;AAAE,gHAAA,WAAW,OAAA;AAAE,mHAAA,cAAc,OAAA;AAChD,oEAAgE;AAAvD,iHAAA,aAAa,OAAA","sourcesContent":["/**\n * footprintjs/advanced — Low-level internals for custom execution engines and testing.\n *\n * Most users never need this. Use `footprintjs` (main) instead.\n * This entry point exposes `SharedMemory`, `StageContext`, `FlowchartTraverser`,\n * and other primitives that power the engine.\n *\n * ```ts\n * import { SharedMemory, StageContext } from 'footprintjs/advanced';\n * ```\n *\n * @module advanced\n */\n/**\n * FootPrint — Advanced / Internal API\n *\n * These exports are for advanced use cases, testing, and building\n * custom execution engines. Most users should use the main 'footprint' entry point.\n *\n * Import via: import { ... } from 'footprint/advanced'\n */\n\n// ============================================================================\n// Memory — Low-level transactional state primitives\n// ============================================================================\n\nexport type {\n  CommitBundle,\n  FlowControlType,\n  FlowMessage,\n  MemoryPatch,\n  StageSnapshot,\n  TraceEntry,\n} from './lib/memory/index.js';\nexport { SharedMemory } from './lib/memory/index.js';\nexport { StageContext } from './lib/memory/index.js';\nexport { EventLog } from './lib/memory/index.js';\nexport { TransactionBuffer } from './lib/memory/index.js';\nexport { DiagnosticCollector } from './lib/memory/index.js';\nexport {\n  applySmartMerge,\n  deepSmartMerge,\n  getNestedValue,\n  getRunAndGlobalPaths,\n  normalisePath,\n  redactPatch,\n  setNestedValue,\n  updateNestedValue,\n  updateValue,\n} from './lib/memory/index.js';\n\n// ============================================================================\n// Builder — Types and internals\n// ============================================================================\n\nexport type {\n  BuildTimeExtractor,\n  BuildTimeNodeMetadata,\n  ExecOptions,\n  FlowChartSpec,\n  ILogger,\n  ScopeProtectionMode,\n  SerializedPipelineStructure,\n  SimplifiedParallelSpec,\n  StageFn,\n  StageNode,\n  StreamCallback,\n  StreamLifecycleHandler,\n  StreamTokenHandler,\n  SubflowMountOptions,\n  SubflowRef,\n} from './lib/builder/index.js';\nexport { ArrayMergeMode, DeciderList, SelectorFnList, specToStageNode } from './lib/builder/index.js';\nexport { createTypedScopeFactory } from './lib/builder/typedFlowChart.js';\n\n// ============================================================================\n// Scope — Providers, protection, recorder options, and event types\n// ============================================================================\n\nexport type {\n  ProviderResolver,\n  ResolveOptions,\n  ScopeProvider,\n  StageContextLike,\n  StrictMode,\n} from './lib/scope/index.js';\nexport { createErrorMessage, createProtectedScope, ScopeFacade } from './lib/scope/index.js';\nexport {\n  attachScopeMethods,\n  isSubclassOfScopeFacade,\n  looksLikeClassCtor,\n  looksLikeFactory,\n  makeClassProvider,\n  makeFactoryProvider,\n  registerScopeResolver,\n  resolveScopeProvider,\n  toScopeFactory,\n} from './lib/scope/index.js';\n\n// Recorder config/option types\nexport type {\n  AggregatedMetrics,\n  DebugEntry,\n  DebugRecorderOptions,\n  DebugVerbosity,\n  DefineScopeOptions,\n  RecorderContext,\n  StageEvent,\n  StageMetrics,\n} from './lib/scope/index.js';\n\n// Zod internals\nexport { createScopeProxyFromZod, defineScopeSchema, isScopeSchema, ZodScopeResolver } from './lib/scope/index.js';\n\n// ============================================================================\n// Runner — Internals\n// ============================================================================\n\nexport type { RuntimeSnapshot } from './lib/runner/index.js';\nexport { ExecutionRuntime } from './lib/runner/index.js';\n\n// ============================================================================\n// Reactive — TypedScope internals (for custom proxy implementations)\n// ============================================================================\n\nexport type { ReactiveOptions, ReactiveTarget } from './lib/reactive/index.js';\nexport {\n  BREAK_SETTER,\n  buildNestedPatch,\n  createArrayProxy,\n  joinPath,\n  SCOPE_METHOD_NAMES,\n  shouldWrapWithProxy,\n} from './lib/reactive/index.js';\n\n// ============================================================================\n// Engine — DFS graph traversal internals\n// ============================================================================\n\nexport type { TraverserOptions } from './lib/engine/index.js';\nexport type { Decider } from './lib/engine/index.js';\nexport { FlowchartTraverser } from './lib/engine/index.js';\nexport { isStageNodeReturn } from './lib/engine/index.js';\n\n// Narrative internals\nexport type { IControlFlowNarrative } from './lib/engine/index.js';\nexport type { CombinedNarrativeEntry, CombinedNarrativeOptions } from './lib/engine/index.js';\nexport type {\n  BranchResult,\n  BranchResults,\n  SerializedPipelineStructure as EngineSerializedPipelineStructure,\n  StageSnapshot as EngineStageSnapshot,\n  ExtractorError,\n  HandlerDeps,\n  IExecutionRuntime,\n  NodeResultType,\n  RuntimeStructureMetadata,\n  ScopeFactory,\n  SerializedPipelineNode,\n  StageFunction,\n  SubflowResult,\n  TraversalExtractor,\n  TraversalResult,\n} from './lib/engine/index.js';\nexport { NullControlFlowNarrativeGenerator } from './lib/engine/index.js';\n\n// Handlers (testing / custom engines)\nexport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './lib/engine/index.js';\nexport {\n  applyOutputMapping,\n  ChildrenExecutor,\n  computeNodeType,\n  ContinuationResolver,\n  createSubflowHandlerDeps,\n  DeciderHandler,\n  DEFAULT_MAX_ITERATIONS,\n  ExtractorRunner,\n  extractParentScopeValues,\n  getInitialScopeValues,\n  NodeResolver,\n  RuntimeStructureManager,\n  seedSubflowGlobalStore,\n  SelectorHandler,\n  StageRunner,\n  SubflowExecutor,\n} from './lib/engine/index.js';\n\n// Trace utilities — re-exported here for convenience. Canonical path: 'footprintjs/trace'\nexport type { ExecutionCounter } from './lib/engine/runtimeStageId.js';\nexport { buildRuntimeStageId, createExecutionCounter, parseRuntimeStageId } from './lib/engine/runtimeStageId.js';\nexport { findCommit, findCommits, findLastWriter } from './lib/memory/commitLogUtils.js';\nexport { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';\n"]}
@@ -37,4 +37,7 @@ export { FlowchartTraverser } from './lib/engine/index.js';
37
37
  export { isStageNodeReturn } from './lib/engine/index.js';
38
38
  export { NullControlFlowNarrativeGenerator } from './lib/engine/index.js';
39
39
  export { applyOutputMapping, ChildrenExecutor, computeNodeType, ContinuationResolver, createSubflowHandlerDeps, DeciderHandler, DEFAULT_MAX_ITERATIONS, ExtractorRunner, extractParentScopeValues, getInitialScopeValues, NodeResolver, RuntimeStructureManager, seedSubflowGlobalStore, SelectorHandler, StageRunner, SubflowExecutor, } from './lib/engine/index.js';
40
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"advanced.js","sourceRoot":"","sources":["../../src/advanced.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AACH;;;;;;;GAOG;AAcH,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AACjD,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAC1D,OAAO,EAAE,mBAAmB,EAAE,MAAM,uBAAuB,CAAC;AAC5D,OAAO,EACL,eAAe,EACf,cAAc,EACd,cAAc,EACd,oBAAoB,EACpB,aAAa,EACb,WAAW,EACX,cAAc,EACd,iBAAiB,EACjB,WAAW,GACZ,MAAM,uBAAuB,CAAC;AAuB/B,OAAO,EAAE,cAAc,EAAE,WAAW,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACtG,OAAO,EAAE,uBAAuB,EAAE,MAAM,iCAAiC,CAAC;AAa1E,OAAO,EAAE,kBAAkB,EAAE,oBAAoB,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAC7F,OAAO,EACL,kBAAkB,EAClB,uBAAuB,EACvB,kBAAkB,EAClB,gBAAgB,EAChB,iBAAiB,EACjB,mBAAmB,EACnB,qBAAqB,EACrB,oBAAoB,EACpB,cAAc,GACf,MAAM,sBAAsB,CAAC;AAc9B,gBAAgB;AAChB,OAAO,EAAE,uBAAuB,EAAE,iBAAiB,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AAOnH,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AAOzD,OAAO,EACL,YAAY,EACZ,gBAAgB,EAChB,gBAAgB,EAChB,QAAQ,EACR,kBAAkB,EAClB,mBAAmB,GACpB,MAAM,yBAAyB,CAAC;AAQjC,OAAO,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAsB1D,OAAO,EAAE,iCAAiC,EAAE,MAAM,uBAAuB,CAAC;AAI1E,OAAO,EACL,kBAAkB,EAClB,gBAAgB,EAChB,eAAe,EACf,oBAAoB,EACpB,wBAAwB,EACxB,cAAc,EACd,sBAAsB,EACtB,eAAe,EACf,wBAAwB,EACxB,qBAAqB,EACrB,YAAY,EACZ,uBAAuB,EACvB,sBAAsB,EACtB,eAAe,EACf,WAAW,EACX,eAAe,GAChB,MAAM,uBAAuB,CAAC","sourcesContent":["/**\n * footprintjs/advanced — Low-level internals for custom execution engines and testing.\n *\n * Most users never need this. Use `footprintjs` (main) instead.\n * This entry point exposes `SharedMemory`, `StageContext`, `FlowchartTraverser`,\n * and other primitives that power the engine.\n *\n * ```ts\n * import { SharedMemory, StageContext } from 'footprintjs/advanced';\n * ```\n *\n * @module advanced\n */\n/**\n * FootPrint — Advanced / Internal API\n *\n * These exports are for advanced use cases, testing, and building\n * custom execution engines. Most users should use the main 'footprint' entry point.\n *\n * Import via: import { ... } from 'footprint/advanced'\n */\n\n// ============================================================================\n// Memory — Low-level transactional state primitives\n// ============================================================================\n\nexport type {\n  CommitBundle,\n  FlowControlType,\n  FlowMessage,\n  MemoryPatch,\n  StageSnapshot,\n  TraceEntry,\n} from './lib/memory/index.js';\nexport { SharedMemory } from './lib/memory/index.js';\nexport { StageContext } from './lib/memory/index.js';\nexport { EventLog } from './lib/memory/index.js';\nexport { TransactionBuffer } from './lib/memory/index.js';\nexport { DiagnosticCollector } from './lib/memory/index.js';\nexport {\n  applySmartMerge,\n  deepSmartMerge,\n  getNestedValue,\n  getRunAndGlobalPaths,\n  normalisePath,\n  redactPatch,\n  setNestedValue,\n  updateNestedValue,\n  updateValue,\n} from './lib/memory/index.js';\n\n// ============================================================================\n// Builder — Types and internals\n// ============================================================================\n\nexport type {\n  BuildTimeExtractor,\n  BuildTimeNodeMetadata,\n  ExecOptions,\n  FlowChartSpec,\n  ILogger,\n  ScopeProtectionMode,\n  SerializedPipelineStructure,\n  SimplifiedParallelSpec,\n  StageFn,\n  StageNode,\n  StreamCallback,\n  StreamLifecycleHandler,\n  StreamTokenHandler,\n  SubflowMountOptions,\n  SubflowRef,\n} from './lib/builder/index.js';\nexport { ArrayMergeMode, DeciderList, SelectorFnList, specToStageNode } from './lib/builder/index.js';\nexport { createTypedScopeFactory } from './lib/builder/typedFlowChart.js';\n\n// ============================================================================\n// Scope — Providers, protection, recorder options, and event types\n// ============================================================================\n\nexport type {\n  ProviderResolver,\n  ResolveOptions,\n  ScopeProvider,\n  StageContextLike,\n  StrictMode,\n} from './lib/scope/index.js';\nexport { createErrorMessage, createProtectedScope, ScopeFacade } from './lib/scope/index.js';\nexport {\n  attachScopeMethods,\n  isSubclassOfScopeFacade,\n  looksLikeClassCtor,\n  looksLikeFactory,\n  makeClassProvider,\n  makeFactoryProvider,\n  registerScopeResolver,\n  resolveScopeProvider,\n  toScopeFactory,\n} from './lib/scope/index.js';\n\n// Recorder config/option types\nexport type {\n  AggregatedMetrics,\n  DebugEntry,\n  DebugRecorderOptions,\n  DebugVerbosity,\n  DefineScopeOptions,\n  RecorderContext,\n  StageEvent,\n  StageMetrics,\n} from './lib/scope/index.js';\n\n// Zod internals\nexport { createScopeProxyFromZod, defineScopeSchema, isScopeSchema, ZodScopeResolver } from './lib/scope/index.js';\n\n// ============================================================================\n// Runner — Internals\n// ============================================================================\n\nexport type { RuntimeSnapshot } from './lib/runner/index.js';\nexport { ExecutionRuntime } from './lib/runner/index.js';\n\n// ============================================================================\n// Reactive — TypedScope internals (for custom proxy implementations)\n// ============================================================================\n\nexport type { ReactiveOptions, ReactiveTarget } from './lib/reactive/index.js';\nexport {\n  BREAK_SETTER,\n  buildNestedPatch,\n  createArrayProxy,\n  joinPath,\n  SCOPE_METHOD_NAMES,\n  shouldWrapWithProxy,\n} from './lib/reactive/index.js';\n\n// ============================================================================\n// Engine — DFS graph traversal internals\n// ============================================================================\n\nexport type { TraverserOptions } from './lib/engine/index.js';\nexport type { Decider } from './lib/engine/index.js';\nexport { FlowchartTraverser } from './lib/engine/index.js';\nexport { isStageNodeReturn } from './lib/engine/index.js';\n\n// Narrative internals\nexport type { IControlFlowNarrative } from './lib/engine/index.js';\nexport type { CombinedNarrativeEntry, CombinedNarrativeOptions } from './lib/engine/index.js';\nexport type {\n  BranchResult,\n  BranchResults,\n  SerializedPipelineStructure as EngineSerializedPipelineStructure,\n  StageSnapshot as EngineStageSnapshot,\n  ExtractorError,\n  HandlerDeps,\n  IExecutionRuntime,\n  NodeResultType,\n  RuntimeStructureMetadata,\n  ScopeFactory,\n  SerializedPipelineNode,\n  StageFunction,\n  SubflowResult,\n  TraversalExtractor,\n  TraversalResult,\n} from './lib/engine/index.js';\nexport { NullControlFlowNarrativeGenerator } from './lib/engine/index.js';\n\n// Handlers (testing / custom engines)\nexport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './lib/engine/index.js';\nexport {\n  applyOutputMapping,\n  ChildrenExecutor,\n  computeNodeType,\n  ContinuationResolver,\n  createSubflowHandlerDeps,\n  DeciderHandler,\n  DEFAULT_MAX_ITERATIONS,\n  ExtractorRunner,\n  extractParentScopeValues,\n  getInitialScopeValues,\n  NodeResolver,\n  RuntimeStructureManager,\n  seedSubflowGlobalStore,\n  SelectorHandler,\n  StageRunner,\n  SubflowExecutor,\n} from './lib/engine/index.js';\n"]}
40
+ export { buildRuntimeStageId, createExecutionCounter, parseRuntimeStageId } from './lib/engine/runtimeStageId.js';
41
+ export { findCommit, findCommits, findLastWriter } from './lib/memory/commitLogUtils.js';
42
+ export { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';
43
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"advanced.js","sourceRoot":"","sources":["../../src/advanced.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AACH;;;;;;;GAOG;AAcH,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AACrD,OAAO,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AACjD,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAC1D,OAAO,EAAE,mBAAmB,EAAE,MAAM,uBAAuB,CAAC;AAC5D,OAAO,EACL,eAAe,EACf,cAAc,EACd,cAAc,EACd,oBAAoB,EACpB,aAAa,EACb,WAAW,EACX,cAAc,EACd,iBAAiB,EACjB,WAAW,GACZ,MAAM,uBAAuB,CAAC;AAuB/B,OAAO,EAAE,cAAc,EAAE,WAAW,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACtG,OAAO,EAAE,uBAAuB,EAAE,MAAM,iCAAiC,CAAC;AAa1E,OAAO,EAAE,kBAAkB,EAAE,oBAAoB,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAC7F,OAAO,EACL,kBAAkB,EAClB,uBAAuB,EACvB,kBAAkB,EAClB,gBAAgB,EAChB,iBAAiB,EACjB,mBAAmB,EACnB,qBAAqB,EACrB,oBAAoB,EACpB,cAAc,GACf,MAAM,sBAAsB,CAAC;AAc9B,gBAAgB;AAChB,OAAO,EAAE,uBAAuB,EAAE,iBAAiB,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AAOnH,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AAOzD,OAAO,EACL,YAAY,EACZ,gBAAgB,EAChB,gBAAgB,EAChB,QAAQ,EACR,kBAAkB,EAClB,mBAAmB,GACpB,MAAM,yBAAyB,CAAC;AAQjC,OAAO,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAsB1D,OAAO,EAAE,iCAAiC,EAAE,MAAM,uBAAuB,CAAC;AAI1E,OAAO,EACL,kBAAkB,EAClB,gBAAgB,EAChB,eAAe,EACf,oBAAoB,EACpB,wBAAwB,EACxB,cAAc,EACd,sBAAsB,EACtB,eAAe,EACf,wBAAwB,EACxB,qBAAqB,EACrB,YAAY,EACZ,uBAAuB,EACvB,sBAAsB,EACtB,eAAe,EACf,WAAW,EACX,eAAe,GAChB,MAAM,uBAAuB,CAAC;AAI/B,OAAO,EAAE,mBAAmB,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,MAAM,gCAAgC,CAAC;AAClH,OAAO,EAAE,UAAU,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,gCAAgC,CAAC;AACzF,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAC","sourcesContent":["/**\n * footprintjs/advanced — Low-level internals for custom execution engines and testing.\n *\n * Most users never need this. Use `footprintjs` (main) instead.\n * This entry point exposes `SharedMemory`, `StageContext`, `FlowchartTraverser`,\n * and other primitives that power the engine.\n *\n * ```ts\n * import { SharedMemory, StageContext } from 'footprintjs/advanced';\n * ```\n *\n * @module advanced\n */\n/**\n * FootPrint — Advanced / Internal API\n *\n * These exports are for advanced use cases, testing, and building\n * custom execution engines. Most users should use the main 'footprint' entry point.\n *\n * Import via: import { ... } from 'footprint/advanced'\n */\n\n// ============================================================================\n// Memory — Low-level transactional state primitives\n// ============================================================================\n\nexport type {\n  CommitBundle,\n  FlowControlType,\n  FlowMessage,\n  MemoryPatch,\n  StageSnapshot,\n  TraceEntry,\n} from './lib/memory/index.js';\nexport { SharedMemory } from './lib/memory/index.js';\nexport { StageContext } from './lib/memory/index.js';\nexport { EventLog } from './lib/memory/index.js';\nexport { TransactionBuffer } from './lib/memory/index.js';\nexport { DiagnosticCollector } from './lib/memory/index.js';\nexport {\n  applySmartMerge,\n  deepSmartMerge,\n  getNestedValue,\n  getRunAndGlobalPaths,\n  normalisePath,\n  redactPatch,\n  setNestedValue,\n  updateNestedValue,\n  updateValue,\n} from './lib/memory/index.js';\n\n// ============================================================================\n// Builder — Types and internals\n// ============================================================================\n\nexport type {\n  BuildTimeExtractor,\n  BuildTimeNodeMetadata,\n  ExecOptions,\n  FlowChartSpec,\n  ILogger,\n  ScopeProtectionMode,\n  SerializedPipelineStructure,\n  SimplifiedParallelSpec,\n  StageFn,\n  StageNode,\n  StreamCallback,\n  StreamLifecycleHandler,\n  StreamTokenHandler,\n  SubflowMountOptions,\n  SubflowRef,\n} from './lib/builder/index.js';\nexport { ArrayMergeMode, DeciderList, SelectorFnList, specToStageNode } from './lib/builder/index.js';\nexport { createTypedScopeFactory } from './lib/builder/typedFlowChart.js';\n\n// ============================================================================\n// Scope — Providers, protection, recorder options, and event types\n// ============================================================================\n\nexport type {\n  ProviderResolver,\n  ResolveOptions,\n  ScopeProvider,\n  StageContextLike,\n  StrictMode,\n} from './lib/scope/index.js';\nexport { createErrorMessage, createProtectedScope, ScopeFacade } from './lib/scope/index.js';\nexport {\n  attachScopeMethods,\n  isSubclassOfScopeFacade,\n  looksLikeClassCtor,\n  looksLikeFactory,\n  makeClassProvider,\n  makeFactoryProvider,\n  registerScopeResolver,\n  resolveScopeProvider,\n  toScopeFactory,\n} from './lib/scope/index.js';\n\n// Recorder config/option types\nexport type {\n  AggregatedMetrics,\n  DebugEntry,\n  DebugRecorderOptions,\n  DebugVerbosity,\n  DefineScopeOptions,\n  RecorderContext,\n  StageEvent,\n  StageMetrics,\n} from './lib/scope/index.js';\n\n// Zod internals\nexport { createScopeProxyFromZod, defineScopeSchema, isScopeSchema, ZodScopeResolver } from './lib/scope/index.js';\n\n// ============================================================================\n// Runner — Internals\n// ============================================================================\n\nexport type { RuntimeSnapshot } from './lib/runner/index.js';\nexport { ExecutionRuntime } from './lib/runner/index.js';\n\n// ============================================================================\n// Reactive — TypedScope internals (for custom proxy implementations)\n// ============================================================================\n\nexport type { ReactiveOptions, ReactiveTarget } from './lib/reactive/index.js';\nexport {\n  BREAK_SETTER,\n  buildNestedPatch,\n  createArrayProxy,\n  joinPath,\n  SCOPE_METHOD_NAMES,\n  shouldWrapWithProxy,\n} from './lib/reactive/index.js';\n\n// ============================================================================\n// Engine — DFS graph traversal internals\n// ============================================================================\n\nexport type { TraverserOptions } from './lib/engine/index.js';\nexport type { Decider } from './lib/engine/index.js';\nexport { FlowchartTraverser } from './lib/engine/index.js';\nexport { isStageNodeReturn } from './lib/engine/index.js';\n\n// Narrative internals\nexport type { IControlFlowNarrative } from './lib/engine/index.js';\nexport type { CombinedNarrativeEntry, CombinedNarrativeOptions } from './lib/engine/index.js';\nexport type {\n  BranchResult,\n  BranchResults,\n  SerializedPipelineStructure as EngineSerializedPipelineStructure,\n  StageSnapshot as EngineStageSnapshot,\n  ExtractorError,\n  HandlerDeps,\n  IExecutionRuntime,\n  NodeResultType,\n  RuntimeStructureMetadata,\n  ScopeFactory,\n  SerializedPipelineNode,\n  StageFunction,\n  SubflowResult,\n  TraversalExtractor,\n  TraversalResult,\n} from './lib/engine/index.js';\nexport { NullControlFlowNarrativeGenerator } from './lib/engine/index.js';\n\n// Handlers (testing / custom engines)\nexport type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './lib/engine/index.js';\nexport {\n  applyOutputMapping,\n  ChildrenExecutor,\n  computeNodeType,\n  ContinuationResolver,\n  createSubflowHandlerDeps,\n  DeciderHandler,\n  DEFAULT_MAX_ITERATIONS,\n  ExtractorRunner,\n  extractParentScopeValues,\n  getInitialScopeValues,\n  NodeResolver,\n  RuntimeStructureManager,\n  seedSubflowGlobalStore,\n  SelectorHandler,\n  StageRunner,\n  SubflowExecutor,\n} from './lib/engine/index.js';\n\n// Trace utilities — re-exported here for convenience. Canonical path: 'footprintjs/trace'\nexport type { ExecutionCounter } from './lib/engine/runtimeStageId.js';\nexport { buildRuntimeStageId, createExecutionCounter, parseRuntimeStageId } from './lib/engine/runtimeStageId.js';\nexport { findCommit, findCommits, findLastWriter } from './lib/memory/commitLogUtils.js';\nexport { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';\n"]}
@@ -0,0 +1,25 @@
1
+ /**
2
+ * Typed utilities for querying the commit log.
3
+ *
4
+ * The commitLog is an ordered array of CommitBundle — one per stage commit.
5
+ * These helpers provide type-safe queries without (b: any) casts.
6
+ */
7
+ /** Find the first commit by stageId, optionally filtering by a written key. */
8
+ export function findCommit(commitLog, stageId, key) {
9
+ return commitLog.find((b) => b.stageId === stageId && (!key || b.trace.some((t) => t.path === key)));
10
+ }
11
+ /** Find all commits by stageId. */
12
+ export function findCommits(commitLog, stageId) {
13
+ return commitLog.filter((b) => b.stageId === stageId);
14
+ }
15
+ /** Find the last commit that wrote a specific key (for backtracking). */
16
+ export function findLastWriter(commitLog, key, beforeIdx) {
17
+ const end = beforeIdx !== null && beforeIdx !== void 0 ? beforeIdx : commitLog.length;
18
+ for (let i = end - 1; i >= 0; i--) {
19
+ if (commitLog[i].trace.some((t) => t.path === key)) {
20
+ return commitLog[i];
21
+ }
22
+ }
23
+ return undefined;
24
+ }
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbWl0TG9nVXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL21lbW9yeS9jb21taXRMb2dVdGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7R0FLRztBQUlILCtFQUErRTtBQUMvRSxNQUFNLFVBQVUsVUFBVSxDQUFDLFNBQXlCLEVBQUUsT0FBZSxFQUFFLEdBQVk7SUFDakYsT0FBTyxTQUFTLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsT0FBTyxLQUFLLE9BQU8sSUFBSSxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUMsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsSUFBSSxLQUFLLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUN2RyxDQUFDO0FBRUQsbUNBQW1DO0FBQ25DLE1BQU0sVUFBVSxXQUFXLENBQUMsU0FBeUIsRUFBRSxPQUFlO0lBQ3BFLE9BQU8sU0FBUyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLE9BQU8sS0FBSyxPQUFPLENBQUMsQ0FBQztBQUN4RCxDQUFDO0FBRUQseUVBQXlFO0FBQ3pFLE1BQU0sVUFBVSxjQUFjLENBQUMsU0FBeUIsRUFBRSxHQUFXLEVBQUUsU0FBa0I7SUFDdkYsTUFBTSxHQUFHLEdBQUcsU0FBUyxhQUFULFNBQVMsY0FBVCxTQUFTLEdBQUksU0FBUyxDQUFDLE1BQU0sQ0FBQztJQUMxQyxLQUFLLElBQUksQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDO1FBQ2xDLElBQUksU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLEtBQUssR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUNuRCxPQUFPLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN0QixDQUFDO0lBQ0gsQ0FBQztJQUNELE9BQU8sU0FBUyxDQUFDO0FBQ25CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFR5cGVkIHV0aWxpdGllcyBmb3IgcXVlcnlpbmcgdGhlIGNvbW1pdCBsb2cuXG4gKlxuICogVGhlIGNvbW1pdExvZyBpcyBhbiBvcmRlcmVkIGFycmF5IG9mIENvbW1pdEJ1bmRsZSDigJQgb25lIHBlciBzdGFnZSBjb21taXQuXG4gKiBUaGVzZSBoZWxwZXJzIHByb3ZpZGUgdHlwZS1zYWZlIHF1ZXJpZXMgd2l0aG91dCAoYjogYW55KSBjYXN0cy5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7IENvbW1pdEJ1bmRsZSB9IGZyb20gJy4vdHlwZXMuanMnO1xuXG4vKiogRmluZCB0aGUgZmlyc3QgY29tbWl0IGJ5IHN0YWdlSWQsIG9wdGlvbmFsbHkgZmlsdGVyaW5nIGJ5IGEgd3JpdHRlbiBrZXkuICovXG5leHBvcnQgZnVuY3Rpb24gZmluZENvbW1pdChjb21taXRMb2c6IENvbW1pdEJ1bmRsZVtdLCBzdGFnZUlkOiBzdHJpbmcsIGtleT86IHN0cmluZyk6IENvbW1pdEJ1bmRsZSB8IHVuZGVmaW5lZCB7XG4gIHJldHVybiBjb21taXRMb2cuZmluZCgoYikgPT4gYi5zdGFnZUlkID09PSBzdGFnZUlkICYmICgha2V5IHx8IGIudHJhY2Uuc29tZSgodCkgPT4gdC5wYXRoID09PSBrZXkpKSk7XG59XG5cbi8qKiBGaW5kIGFsbCBjb21taXRzIGJ5IHN0YWdlSWQuICovXG5leHBvcnQgZnVuY3Rpb24gZmluZENvbW1pdHMoY29tbWl0TG9nOiBDb21taXRCdW5kbGVbXSwgc3RhZ2VJZDogc3RyaW5nKTogQ29tbWl0QnVuZGxlW10ge1xuICByZXR1cm4gY29tbWl0TG9nLmZpbHRlcigoYikgPT4gYi5zdGFnZUlkID09PSBzdGFnZUlkKTtcbn1cblxuLyoqIEZpbmQgdGhlIGxhc3QgY29tbWl0IHRoYXQgd3JvdGUgYSBzcGVjaWZpYyBrZXkgKGZvciBiYWNrdHJhY2tpbmcpLiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZpbmRMYXN0V3JpdGVyKGNvbW1pdExvZzogQ29tbWl0QnVuZGxlW10sIGtleTogc3RyaW5nLCBiZWZvcmVJZHg/OiBudW1iZXIpOiBDb21taXRCdW5kbGUgfCB1bmRlZmluZWQge1xuICBjb25zdCBlbmQgPSBiZWZvcmVJZHggPz8gY29tbWl0TG9nLmxlbmd0aDtcbiAgZm9yIChsZXQgaSA9IGVuZCAtIDE7IGkgPj0gMDsgaS0tKSB7XG4gICAgaWYgKGNvbW1pdExvZ1tpXS50cmFjZS5zb21lKCh0KSA9PiB0LnBhdGggPT09IGtleSkpIHtcbiAgICAgIHJldHVybiBjb21taXRMb2dbaV07XG4gICAgfVxuICB9XG4gIHJldHVybiB1bmRlZmluZWQ7XG59XG4iXX0=
@@ -0,0 +1,47 @@
1
+ /**
2
+ * KeyedRecorder<T> — base class for Map-based recorders keyed by runtimeStageId.
3
+ *
4
+ * Provides typed key-value storage with O(1) lookup, insertion-ordered iteration,
5
+ * and common accessors. Recorder implementations extend this and call store()
6
+ * from their event hooks.
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * class TokenRecorder extends KeyedRecorder<LLMCallEntry> {
11
+ * onLLMCall(event: LLMCallEvent) {
12
+ * this.store(event.runtimeStageId, { model: event.model, ... });
13
+ * }
14
+ * getStats() { return aggregate(this.values()); }
15
+ * }
16
+ * ```
17
+ */
18
+ export class KeyedRecorder {
19
+ constructor() {
20
+ this.data = new Map();
21
+ }
22
+ /** Store an entry keyed by runtimeStageId. */
23
+ store(runtimeStageId, entry) {
24
+ this.data.set(runtimeStageId, entry);
25
+ }
26
+ /** O(1) lookup by runtimeStageId. */
27
+ getByKey(runtimeStageId) {
28
+ return this.data.get(runtimeStageId);
29
+ }
30
+ /** All entries as a read-only Map (insertion-ordered). */
31
+ getMap() {
32
+ return this.data;
33
+ }
34
+ /** All entries as an array (insertion-ordered). */
35
+ values() {
36
+ return [...this.data.values()];
37
+ }
38
+ /** Number of entries stored. */
39
+ get size() {
40
+ return this.data.size;
41
+ }
42
+ /** Clear all stored data. Called by executor before each run(). */
43
+ clear() {
44
+ this.data.clear();
45
+ }
46
+ }
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiS2V5ZWRSZWNvcmRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvcmVjb3JkZXIvS2V5ZWRSZWNvcmRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7OztHQWdCRztBQUNILE1BQU0sT0FBZ0IsYUFBYTtJQUFuQztRQUdtQixTQUFJLEdBQUcsSUFBSSxHQUFHLEVBQWEsQ0FBQztJQStCL0MsQ0FBQztJQTdCQyw4Q0FBOEM7SUFDcEMsS0FBSyxDQUFDLGNBQXNCLEVBQUUsS0FBUTtRQUM5QyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxjQUFjLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDdkMsQ0FBQztJQUVELHFDQUFxQztJQUNyQyxRQUFRLENBQUMsY0FBc0I7UUFDN0IsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxjQUFjLENBQUMsQ0FBQztJQUN2QyxDQUFDO0lBRUQsMERBQTBEO0lBQzFELE1BQU07UUFDSixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUM7SUFDbkIsQ0FBQztJQUVELG1EQUFtRDtJQUNuRCxNQUFNO1FBQ0osT0FBTyxDQUFDLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO0lBQ2pDLENBQUM7SUFFRCxnQ0FBZ0M7SUFDaEMsSUFBSSxJQUFJO1FBQ04sT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztJQUN4QixDQUFDO0lBRUQsbUVBQW1FO0lBQ25FLEtBQUs7UUFDSCxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ3BCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogS2V5ZWRSZWNvcmRlcjxUPiDigJQgYmFzZSBjbGFzcyBmb3IgTWFwLWJhc2VkIHJlY29yZGVycyBrZXllZCBieSBydW50aW1lU3RhZ2VJZC5cbiAqXG4gKiBQcm92aWRlcyB0eXBlZCBrZXktdmFsdWUgc3RvcmFnZSB3aXRoIE8oMSkgbG9va3VwLCBpbnNlcnRpb24tb3JkZXJlZCBpdGVyYXRpb24sXG4gKiBhbmQgY29tbW9uIGFjY2Vzc29ycy4gUmVjb3JkZXIgaW1wbGVtZW50YXRpb25zIGV4dGVuZCB0aGlzIGFuZCBjYWxsIHN0b3JlKClcbiAqIGZyb20gdGhlaXIgZXZlbnQgaG9va3MuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNsYXNzIFRva2VuUmVjb3JkZXIgZXh0ZW5kcyBLZXllZFJlY29yZGVyPExMTUNhbGxFbnRyeT4ge1xuICogICBvbkxMTUNhbGwoZXZlbnQ6IExMTUNhbGxFdmVudCkge1xuICogICAgIHRoaXMuc3RvcmUoZXZlbnQucnVudGltZVN0YWdlSWQsIHsgbW9kZWw6IGV2ZW50Lm1vZGVsLCAuLi4gfSk7XG4gKiAgIH1cbiAqICAgZ2V0U3RhdHMoKSB7IHJldHVybiBhZ2dyZWdhdGUodGhpcy52YWx1ZXMoKSk7IH1cbiAqIH1cbiAqIGBgYFxuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgS2V5ZWRSZWNvcmRlcjxUPiB7XG4gIGFic3RyYWN0IHJlYWRvbmx5IGlkOiBzdHJpbmc7XG5cbiAgcHJpdmF0ZSByZWFkb25seSBkYXRhID0gbmV3IE1hcDxzdHJpbmcsIFQ+KCk7XG5cbiAgLyoqIFN0b3JlIGFuIGVudHJ5IGtleWVkIGJ5IHJ1bnRpbWVTdGFnZUlkLiAqL1xuICBwcm90ZWN0ZWQgc3RvcmUocnVudGltZVN0YWdlSWQ6IHN0cmluZywgZW50cnk6IFQpOiB2b2lkIHtcbiAgICB0aGlzLmRhdGEuc2V0KHJ1bnRpbWVTdGFnZUlkLCBlbnRyeSk7XG4gIH1cblxuICAvKiogTygxKSBsb29rdXAgYnkgcnVudGltZVN0YWdlSWQuICovXG4gIGdldEJ5S2V5KHJ1bnRpbWVTdGFnZUlkOiBzdHJpbmcpOiBUIHwgdW5kZWZpbmVkIHtcbiAgICByZXR1cm4gdGhpcy5kYXRhLmdldChydW50aW1lU3RhZ2VJZCk7XG4gIH1cblxuICAvKiogQWxsIGVudHJpZXMgYXMgYSByZWFkLW9ubHkgTWFwIChpbnNlcnRpb24tb3JkZXJlZCkuICovXG4gIGdldE1hcCgpOiBSZWFkb25seU1hcDxzdHJpbmcsIFQ+IHtcbiAgICByZXR1cm4gdGhpcy5kYXRhO1xuICB9XG5cbiAgLyoqIEFsbCBlbnRyaWVzIGFzIGFuIGFycmF5IChpbnNlcnRpb24tb3JkZXJlZCkuICovXG4gIHZhbHVlcygpOiBUW10ge1xuICAgIHJldHVybiBbLi4udGhpcy5kYXRhLnZhbHVlcygpXTtcbiAgfVxuXG4gIC8qKiBOdW1iZXIgb2YgZW50cmllcyBzdG9yZWQuICovXG4gIGdldCBzaXplKCk6IG51bWJlciB7XG4gICAgcmV0dXJuIHRoaXMuZGF0YS5zaXplO1xuICB9XG5cbiAgLyoqIENsZWFyIGFsbCBzdG9yZWQgZGF0YS4gQ2FsbGVkIGJ5IGV4ZWN1dG9yIGJlZm9yZSBlYWNoIHJ1bigpLiAqL1xuICBjbGVhcigpOiB2b2lkIHtcbiAgICB0aGlzLmRhdGEuY2xlYXIoKTtcbiAgfVxufVxuIl19
@@ -1,2 +1,3 @@
1
1
  export { CompositeRecorder } from './CompositeRecorder.js';
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL3JlY29yZGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHdCQUF3QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgeyBDb21wb3NpdGVTbmFwc2hvdCB9IGZyb20gJy4vQ29tcG9zaXRlUmVjb3JkZXIuanMnO1xuZXhwb3J0IHsgQ29tcG9zaXRlUmVjb3JkZXIgfSBmcm9tICcuL0NvbXBvc2l0ZVJlY29yZGVyLmpzJztcbiJdfQ==
2
+ export { KeyedRecorder } from './KeyedRecorder.js';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL3JlY29yZGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQzNELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIHsgQ29tcG9zaXRlU25hcHNob3QgfSBmcm9tICcuL0NvbXBvc2l0ZVJlY29yZGVyLmpzJztcbmV4cG9ydCB7IENvbXBvc2l0ZVJlY29yZGVyIH0gZnJvbSAnLi9Db21wb3NpdGVSZWNvcmRlci5qcyc7XG5leHBvcnQgeyBLZXllZFJlY29yZGVyIH0gZnJvbSAnLi9LZXllZFJlY29yZGVyLmpzJztcbiJdfQ==
@@ -0,0 +1,25 @@
1
+ /**
2
+ * footprintjs/trace — Execution tracing, debugging, and backtracking utilities.
3
+ *
4
+ * Runtime stage IDs, commit log queries, and keyed recorder base class.
5
+ *
6
+ * @example
7
+ * ```typescript
8
+ * import { parseRuntimeStageId, findLastWriter, KeyedRecorder } from 'footprintjs/trace';
9
+ *
10
+ * // Parse a runtimeStageId
11
+ * const { stageId, executionIndex } = parseRuntimeStageId('call-llm#5');
12
+ *
13
+ * // Backtrack: who wrote 'systemPrompt' before stage at idx 8?
14
+ * const writer = findLastWriter(commitLog, 'systemPrompt', 8);
15
+ *
16
+ * // Build a keyed recorder
17
+ * class MyRecorder extends KeyedRecorder<MyEntry> { ... }
18
+ * ```
19
+ */
20
+ export { buildRuntimeStageId, createExecutionCounter, parseRuntimeStageId } from './lib/engine/runtimeStageId.js';
21
+ // Commit log queries — typed utilities for backtracking
22
+ export { findCommit, findCommits, findLastWriter } from './lib/memory/commitLogUtils.js';
23
+ // KeyedRecorder — base class for Map-based recorders
24
+ export { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHJhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQWtCRztBQUlILE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxzQkFBc0IsRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBRWxILHdEQUF3RDtBQUN4RCxPQUFPLEVBQUUsVUFBVSxFQUFFLFdBQVcsRUFBRSxjQUFjLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUV6RixxREFBcUQ7QUFDckQsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGlDQUFpQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBmb290cHJpbnRqcy90cmFjZSDigJQgRXhlY3V0aW9uIHRyYWNpbmcsIGRlYnVnZ2luZywgYW5kIGJhY2t0cmFja2luZyB1dGlsaXRpZXMuXG4gKlxuICogUnVudGltZSBzdGFnZSBJRHMsIGNvbW1pdCBsb2cgcXVlcmllcywgYW5kIGtleWVkIHJlY29yZGVyIGJhc2UgY2xhc3MuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IHBhcnNlUnVudGltZVN0YWdlSWQsIGZpbmRMYXN0V3JpdGVyLCBLZXllZFJlY29yZGVyIH0gZnJvbSAnZm9vdHByaW50anMvdHJhY2UnO1xuICpcbiAqIC8vIFBhcnNlIGEgcnVudGltZVN0YWdlSWRcbiAqIGNvbnN0IHsgc3RhZ2VJZCwgZXhlY3V0aW9uSW5kZXggfSA9IHBhcnNlUnVudGltZVN0YWdlSWQoJ2NhbGwtbGxtIzUnKTtcbiAqXG4gKiAvLyBCYWNrdHJhY2s6IHdobyB3cm90ZSAnc3lzdGVtUHJvbXB0JyBiZWZvcmUgc3RhZ2UgYXQgaWR4IDg/XG4gKiBjb25zdCB3cml0ZXIgPSBmaW5kTGFzdFdyaXRlcihjb21taXRMb2csICdzeXN0ZW1Qcm9tcHQnLCA4KTtcbiAqXG4gKiAvLyBCdWlsZCBhIGtleWVkIHJlY29yZGVyXG4gKiBjbGFzcyBNeVJlY29yZGVyIGV4dGVuZHMgS2V5ZWRSZWNvcmRlcjxNeUVudHJ5PiB7IC4uLiB9XG4gKiBgYGBcbiAqL1xuXG4vLyBSdW50aW1lIHN0YWdlIElEIOKAlCB1bmlxdWUgZXhlY3V0aW9uIHN0ZXAgaWRlbnRpZmllcnNcbmV4cG9ydCB0eXBlIHsgRXhlY3V0aW9uQ291bnRlciB9IGZyb20gJy4vbGliL2VuZ2luZS9ydW50aW1lU3RhZ2VJZC5qcyc7XG5leHBvcnQgeyBidWlsZFJ1bnRpbWVTdGFnZUlkLCBjcmVhdGVFeGVjdXRpb25Db3VudGVyLCBwYXJzZVJ1bnRpbWVTdGFnZUlkIH0gZnJvbSAnLi9saWIvZW5naW5lL3J1bnRpbWVTdGFnZUlkLmpzJztcblxuLy8gQ29tbWl0IGxvZyBxdWVyaWVzIOKAlCB0eXBlZCB1dGlsaXRpZXMgZm9yIGJhY2t0cmFja2luZ1xuZXhwb3J0IHsgZmluZENvbW1pdCwgZmluZENvbW1pdHMsIGZpbmRMYXN0V3JpdGVyIH0gZnJvbSAnLi9saWIvbWVtb3J5L2NvbW1pdExvZ1V0aWxzLmpzJztcblxuLy8gS2V5ZWRSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgTWFwLWJhc2VkIHJlY29yZGVyc1xuZXhwb3J0IHsgS2V5ZWRSZWNvcmRlciB9IGZyb20gJy4vbGliL3JlY29yZGVyL0tleWVkUmVjb3JkZXIuanMnO1xuIl19
@@ -0,0 +1,31 @@
1
+ "use strict";
2
+ /**
3
+ * Typed utilities for querying the commit log.
4
+ *
5
+ * The commitLog is an ordered array of CommitBundle — one per stage commit.
6
+ * These helpers provide type-safe queries without (b: any) casts.
7
+ */
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.findLastWriter = exports.findCommits = exports.findCommit = void 0;
10
+ /** Find the first commit by stageId, optionally filtering by a written key. */
11
+ function findCommit(commitLog, stageId, key) {
12
+ return commitLog.find((b) => b.stageId === stageId && (!key || b.trace.some((t) => t.path === key)));
13
+ }
14
+ exports.findCommit = findCommit;
15
+ /** Find all commits by stageId. */
16
+ function findCommits(commitLog, stageId) {
17
+ return commitLog.filter((b) => b.stageId === stageId);
18
+ }
19
+ exports.findCommits = findCommits;
20
+ /** Find the last commit that wrote a specific key (for backtracking). */
21
+ function findLastWriter(commitLog, key, beforeIdx) {
22
+ const end = beforeIdx !== null && beforeIdx !== void 0 ? beforeIdx : commitLog.length;
23
+ for (let i = end - 1; i >= 0; i--) {
24
+ if (commitLog[i].trace.some((t) => t.path === key)) {
25
+ return commitLog[i];
26
+ }
27
+ }
28
+ return undefined;
29
+ }
30
+ exports.findLastWriter = findLastWriter;
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbWl0TG9nVXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL21lbW9yeS9jb21taXRMb2dVdGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7O0dBS0c7OztBQUlILCtFQUErRTtBQUMvRSxTQUFnQixVQUFVLENBQUMsU0FBeUIsRUFBRSxPQUFlLEVBQUUsR0FBWTtJQUNqRixPQUFPLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxPQUFPLEtBQUssT0FBTyxJQUFJLENBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLEtBQUssR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO0FBQ3ZHLENBQUM7QUFGRCxnQ0FFQztBQUVELG1DQUFtQztBQUNuQyxTQUFnQixXQUFXLENBQUMsU0FBeUIsRUFBRSxPQUFlO0lBQ3BFLE9BQU8sU0FBUyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLE9BQU8sS0FBSyxPQUFPLENBQUMsQ0FBQztBQUN4RCxDQUFDO0FBRkQsa0NBRUM7QUFFRCx5RUFBeUU7QUFDekUsU0FBZ0IsY0FBYyxDQUFDLFNBQXlCLEVBQUUsR0FBVyxFQUFFLFNBQWtCO0lBQ3ZGLE1BQU0sR0FBRyxHQUFHLFNBQVMsYUFBVCxTQUFTLGNBQVQsU0FBUyxHQUFJLFNBQVMsQ0FBQyxNQUFNLENBQUM7SUFDMUMsS0FBSyxJQUFJLENBQUMsR0FBRyxHQUFHLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQztRQUNsQyxJQUFJLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsSUFBSSxLQUFLLEdBQUcsQ0FBQyxFQUFFLENBQUM7WUFDbkQsT0FBTyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDdEIsQ0FBQztJQUNILENBQUM7SUFDRCxPQUFPLFNBQVMsQ0FBQztBQUNuQixDQUFDO0FBUkQsd0NBUUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFR5cGVkIHV0aWxpdGllcyBmb3IgcXVlcnlpbmcgdGhlIGNvbW1pdCBsb2cuXG4gKlxuICogVGhlIGNvbW1pdExvZyBpcyBhbiBvcmRlcmVkIGFycmF5IG9mIENvbW1pdEJ1bmRsZSDigJQgb25lIHBlciBzdGFnZSBjb21taXQuXG4gKiBUaGVzZSBoZWxwZXJzIHByb3ZpZGUgdHlwZS1zYWZlIHF1ZXJpZXMgd2l0aG91dCAoYjogYW55KSBjYXN0cy5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7IENvbW1pdEJ1bmRsZSB9IGZyb20gJy4vdHlwZXMuanMnO1xuXG4vKiogRmluZCB0aGUgZmlyc3QgY29tbWl0IGJ5IHN0YWdlSWQsIG9wdGlvbmFsbHkgZmlsdGVyaW5nIGJ5IGEgd3JpdHRlbiBrZXkuICovXG5leHBvcnQgZnVuY3Rpb24gZmluZENvbW1pdChjb21taXRMb2c6IENvbW1pdEJ1bmRsZVtdLCBzdGFnZUlkOiBzdHJpbmcsIGtleT86IHN0cmluZyk6IENvbW1pdEJ1bmRsZSB8IHVuZGVmaW5lZCB7XG4gIHJldHVybiBjb21taXRMb2cuZmluZCgoYikgPT4gYi5zdGFnZUlkID09PSBzdGFnZUlkICYmICgha2V5IHx8IGIudHJhY2Uuc29tZSgodCkgPT4gdC5wYXRoID09PSBrZXkpKSk7XG59XG5cbi8qKiBGaW5kIGFsbCBjb21taXRzIGJ5IHN0YWdlSWQuICovXG5leHBvcnQgZnVuY3Rpb24gZmluZENvbW1pdHMoY29tbWl0TG9nOiBDb21taXRCdW5kbGVbXSwgc3RhZ2VJZDogc3RyaW5nKTogQ29tbWl0QnVuZGxlW10ge1xuICByZXR1cm4gY29tbWl0TG9nLmZpbHRlcigoYikgPT4gYi5zdGFnZUlkID09PSBzdGFnZUlkKTtcbn1cblxuLyoqIEZpbmQgdGhlIGxhc3QgY29tbWl0IHRoYXQgd3JvdGUgYSBzcGVjaWZpYyBrZXkgKGZvciBiYWNrdHJhY2tpbmcpLiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZpbmRMYXN0V3JpdGVyKGNvbW1pdExvZzogQ29tbWl0QnVuZGxlW10sIGtleTogc3RyaW5nLCBiZWZvcmVJZHg/OiBudW1iZXIpOiBDb21taXRCdW5kbGUgfCB1bmRlZmluZWQge1xuICBjb25zdCBlbmQgPSBiZWZvcmVJZHggPz8gY29tbWl0TG9nLmxlbmd0aDtcbiAgZm9yIChsZXQgaSA9IGVuZCAtIDE7IGkgPj0gMDsgaS0tKSB7XG4gICAgaWYgKGNvbW1pdExvZ1tpXS50cmFjZS5zb21lKCh0KSA9PiB0LnBhdGggPT09IGtleSkpIHtcbiAgICAgIHJldHVybiBjb21taXRMb2dbaV07XG4gICAgfVxuICB9XG4gIHJldHVybiB1bmRlZmluZWQ7XG59XG4iXX0=
@@ -0,0 +1,51 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.KeyedRecorder = void 0;
4
+ /**
5
+ * KeyedRecorder<T> — base class for Map-based recorders keyed by runtimeStageId.
6
+ *
7
+ * Provides typed key-value storage with O(1) lookup, insertion-ordered iteration,
8
+ * and common accessors. Recorder implementations extend this and call store()
9
+ * from their event hooks.
10
+ *
11
+ * @example
12
+ * ```typescript
13
+ * class TokenRecorder extends KeyedRecorder<LLMCallEntry> {
14
+ * onLLMCall(event: LLMCallEvent) {
15
+ * this.store(event.runtimeStageId, { model: event.model, ... });
16
+ * }
17
+ * getStats() { return aggregate(this.values()); }
18
+ * }
19
+ * ```
20
+ */
21
+ class KeyedRecorder {
22
+ constructor() {
23
+ this.data = new Map();
24
+ }
25
+ /** Store an entry keyed by runtimeStageId. */
26
+ store(runtimeStageId, entry) {
27
+ this.data.set(runtimeStageId, entry);
28
+ }
29
+ /** O(1) lookup by runtimeStageId. */
30
+ getByKey(runtimeStageId) {
31
+ return this.data.get(runtimeStageId);
32
+ }
33
+ /** All entries as a read-only Map (insertion-ordered). */
34
+ getMap() {
35
+ return this.data;
36
+ }
37
+ /** All entries as an array (insertion-ordered). */
38
+ values() {
39
+ return [...this.data.values()];
40
+ }
41
+ /** Number of entries stored. */
42
+ get size() {
43
+ return this.data.size;
44
+ }
45
+ /** Clear all stored data. Called by executor before each run(). */
46
+ clear() {
47
+ this.data.clear();
48
+ }
49
+ }
50
+ exports.KeyedRecorder = KeyedRecorder;
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiS2V5ZWRSZWNvcmRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9saWIvcmVjb3JkZXIvS2V5ZWRSZWNvcmRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQTs7Ozs7Ozs7Ozs7Ozs7OztHQWdCRztBQUNILE1BQXNCLGFBQWE7SUFBbkM7UUFHbUIsU0FBSSxHQUFHLElBQUksR0FBRyxFQUFhLENBQUM7SUErQi9DLENBQUM7SUE3QkMsOENBQThDO0lBQ3BDLEtBQUssQ0FBQyxjQUFzQixFQUFFLEtBQVE7UUFDOUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsY0FBYyxFQUFFLEtBQUssQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFRCxxQ0FBcUM7SUFDckMsUUFBUSxDQUFDLGNBQXNCO1FBQzdCLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsY0FBYyxDQUFDLENBQUM7SUFDdkMsQ0FBQztJQUVELDBEQUEwRDtJQUMxRCxNQUFNO1FBQ0osT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDO0lBQ25CLENBQUM7SUFFRCxtREFBbUQ7SUFDbkQsTUFBTTtRQUNKLE9BQU8sQ0FBQyxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztJQUNqQyxDQUFDO0lBRUQsZ0NBQWdDO0lBQ2hDLElBQUksSUFBSTtRQUNOLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUM7SUFDeEIsQ0FBQztJQUVELG1FQUFtRTtJQUNuRSxLQUFLO1FBQ0gsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUNwQixDQUFDO0NBQ0Y7QUFsQ0Qsc0NBa0NDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBLZXllZFJlY29yZGVyPFQ+IOKAlCBiYXNlIGNsYXNzIGZvciBNYXAtYmFzZWQgcmVjb3JkZXJzIGtleWVkIGJ5IHJ1bnRpbWVTdGFnZUlkLlxuICpcbiAqIFByb3ZpZGVzIHR5cGVkIGtleS12YWx1ZSBzdG9yYWdlIHdpdGggTygxKSBsb29rdXAsIGluc2VydGlvbi1vcmRlcmVkIGl0ZXJhdGlvbixcbiAqIGFuZCBjb21tb24gYWNjZXNzb3JzLiBSZWNvcmRlciBpbXBsZW1lbnRhdGlvbnMgZXh0ZW5kIHRoaXMgYW5kIGNhbGwgc3RvcmUoKVxuICogZnJvbSB0aGVpciBldmVudCBob29rcy5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogY2xhc3MgVG9rZW5SZWNvcmRlciBleHRlbmRzIEtleWVkUmVjb3JkZXI8TExNQ2FsbEVudHJ5PiB7XG4gKiAgIG9uTExNQ2FsbChldmVudDogTExNQ2FsbEV2ZW50KSB7XG4gKiAgICAgdGhpcy5zdG9yZShldmVudC5ydW50aW1lU3RhZ2VJZCwgeyBtb2RlbDogZXZlbnQubW9kZWwsIC4uLiB9KTtcbiAqICAgfVxuICogICBnZXRTdGF0cygpIHsgcmV0dXJuIGFnZ3JlZ2F0ZSh0aGlzLnZhbHVlcygpKTsgfVxuICogfVxuICogYGBgXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBLZXllZFJlY29yZGVyPFQ+IHtcbiAgYWJzdHJhY3QgcmVhZG9ubHkgaWQ6IHN0cmluZztcblxuICBwcml2YXRlIHJlYWRvbmx5IGRhdGEgPSBuZXcgTWFwPHN0cmluZywgVD4oKTtcblxuICAvKiogU3RvcmUgYW4gZW50cnkga2V5ZWQgYnkgcnVudGltZVN0YWdlSWQuICovXG4gIHByb3RlY3RlZCBzdG9yZShydW50aW1lU3RhZ2VJZDogc3RyaW5nLCBlbnRyeTogVCk6IHZvaWQge1xuICAgIHRoaXMuZGF0YS5zZXQocnVudGltZVN0YWdlSWQsIGVudHJ5KTtcbiAgfVxuXG4gIC8qKiBPKDEpIGxvb2t1cCBieSBydW50aW1lU3RhZ2VJZC4gKi9cbiAgZ2V0QnlLZXkocnVudGltZVN0YWdlSWQ6IHN0cmluZyk6IFQgfCB1bmRlZmluZWQge1xuICAgIHJldHVybiB0aGlzLmRhdGEuZ2V0KHJ1bnRpbWVTdGFnZUlkKTtcbiAgfVxuXG4gIC8qKiBBbGwgZW50cmllcyBhcyBhIHJlYWQtb25seSBNYXAgKGluc2VydGlvbi1vcmRlcmVkKS4gKi9cbiAgZ2V0TWFwKCk6IFJlYWRvbmx5TWFwPHN0cmluZywgVD4ge1xuICAgIHJldHVybiB0aGlzLmRhdGE7XG4gIH1cblxuICAvKiogQWxsIGVudHJpZXMgYXMgYW4gYXJyYXkgKGluc2VydGlvbi1vcmRlcmVkKS4gKi9cbiAgdmFsdWVzKCk6IFRbXSB7XG4gICAgcmV0dXJuIFsuLi50aGlzLmRhdGEudmFsdWVzKCldO1xuICB9XG5cbiAgLyoqIE51bWJlciBvZiBlbnRyaWVzIHN0b3JlZC4gKi9cbiAgZ2V0IHNpemUoKTogbnVtYmVyIHtcbiAgICByZXR1cm4gdGhpcy5kYXRhLnNpemU7XG4gIH1cblxuICAvKiogQ2xlYXIgYWxsIHN0b3JlZCBkYXRhLiBDYWxsZWQgYnkgZXhlY3V0b3IgYmVmb3JlIGVhY2ggcnVuKCkuICovXG4gIGNsZWFyKCk6IHZvaWQge1xuICAgIHRoaXMuZGF0YS5jbGVhcigpO1xuICB9XG59XG4iXX0=
@@ -1,6 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.CompositeRecorder = void 0;
3
+ exports.KeyedRecorder = exports.CompositeRecorder = void 0;
4
4
  var CompositeRecorder_js_1 = require("./CompositeRecorder.js");
5
5
  Object.defineProperty(exports, "CompositeRecorder", { enumerable: true, get: function () { return CompositeRecorder_js_1.CompositeRecorder; } });
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL3JlY29yZGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUNBLCtEQUEyRDtBQUFsRCx5SEFBQSxpQkFBaUIsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIHsgQ29tcG9zaXRlU25hcHNob3QgfSBmcm9tICcuL0NvbXBvc2l0ZVJlY29yZGVyLmpzJztcbmV4cG9ydCB7IENvbXBvc2l0ZVJlY29yZGVyIH0gZnJvbSAnLi9Db21wb3NpdGVSZWNvcmRlci5qcyc7XG4iXX0=
6
+ var KeyedRecorder_js_1 = require("./KeyedRecorder.js");
7
+ Object.defineProperty(exports, "KeyedRecorder", { enumerable: true, get: function () { return KeyedRecorder_js_1.KeyedRecorder; } });
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL3JlY29yZGVyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUNBLCtEQUEyRDtBQUFsRCx5SEFBQSxpQkFBaUIsT0FBQTtBQUMxQix1REFBbUQ7QUFBMUMsaUhBQUEsYUFBYSxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgeyBDb21wb3NpdGVTbmFwc2hvdCB9IGZyb20gJy4vQ29tcG9zaXRlUmVjb3JkZXIuanMnO1xuZXhwb3J0IHsgQ29tcG9zaXRlUmVjb3JkZXIgfSBmcm9tICcuL0NvbXBvc2l0ZVJlY29yZGVyLmpzJztcbmV4cG9ydCB7IEtleWVkUmVjb3JkZXIgfSBmcm9tICcuL0tleWVkUmVjb3JkZXIuanMnO1xuIl19
package/dist/trace.js ADDED
@@ -0,0 +1,35 @@
1
+ "use strict";
2
+ /**
3
+ * footprintjs/trace — Execution tracing, debugging, and backtracking utilities.
4
+ *
5
+ * Runtime stage IDs, commit log queries, and keyed recorder base class.
6
+ *
7
+ * @example
8
+ * ```typescript
9
+ * import { parseRuntimeStageId, findLastWriter, KeyedRecorder } from 'footprintjs/trace';
10
+ *
11
+ * // Parse a runtimeStageId
12
+ * const { stageId, executionIndex } = parseRuntimeStageId('call-llm#5');
13
+ *
14
+ * // Backtrack: who wrote 'systemPrompt' before stage at idx 8?
15
+ * const writer = findLastWriter(commitLog, 'systemPrompt', 8);
16
+ *
17
+ * // Build a keyed recorder
18
+ * class MyRecorder extends KeyedRecorder<MyEntry> { ... }
19
+ * ```
20
+ */
21
+ Object.defineProperty(exports, "__esModule", { value: true });
22
+ exports.KeyedRecorder = exports.findLastWriter = exports.findCommits = exports.findCommit = exports.parseRuntimeStageId = exports.createExecutionCounter = exports.buildRuntimeStageId = void 0;
23
+ var runtimeStageId_js_1 = require("./lib/engine/runtimeStageId.js");
24
+ Object.defineProperty(exports, "buildRuntimeStageId", { enumerable: true, get: function () { return runtimeStageId_js_1.buildRuntimeStageId; } });
25
+ Object.defineProperty(exports, "createExecutionCounter", { enumerable: true, get: function () { return runtimeStageId_js_1.createExecutionCounter; } });
26
+ Object.defineProperty(exports, "parseRuntimeStageId", { enumerable: true, get: function () { return runtimeStageId_js_1.parseRuntimeStageId; } });
27
+ // Commit log queries — typed utilities for backtracking
28
+ var commitLogUtils_js_1 = require("./lib/memory/commitLogUtils.js");
29
+ Object.defineProperty(exports, "findCommit", { enumerable: true, get: function () { return commitLogUtils_js_1.findCommit; } });
30
+ Object.defineProperty(exports, "findCommits", { enumerable: true, get: function () { return commitLogUtils_js_1.findCommits; } });
31
+ Object.defineProperty(exports, "findLastWriter", { enumerable: true, get: function () { return commitLogUtils_js_1.findLastWriter; } });
32
+ // KeyedRecorder — base class for Map-based recorders
33
+ var KeyedRecorder_js_1 = require("./lib/recorder/KeyedRecorder.js");
34
+ Object.defineProperty(exports, "KeyedRecorder", { enumerable: true, get: function () { return KeyedRecorder_js_1.KeyedRecorder; } });
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHJhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FrQkc7OztBQUlILG9FQUFrSDtBQUF6Ryx3SEFBQSxtQkFBbUIsT0FBQTtBQUFFLDJIQUFBLHNCQUFzQixPQUFBO0FBQUUsd0hBQUEsbUJBQW1CLE9BQUE7QUFFekUsd0RBQXdEO0FBQ3hELG9FQUF5RjtBQUFoRiwrR0FBQSxVQUFVLE9BQUE7QUFBRSxnSEFBQSxXQUFXLE9BQUE7QUFBRSxtSEFBQSxjQUFjLE9BQUE7QUFFaEQscURBQXFEO0FBQ3JELG9FQUFnRTtBQUF2RCxpSEFBQSxhQUFhLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIGZvb3RwcmludGpzL3RyYWNlIOKAlCBFeGVjdXRpb24gdHJhY2luZywgZGVidWdnaW5nLCBhbmQgYmFja3RyYWNraW5nIHV0aWxpdGllcy5cbiAqXG4gKiBSdW50aW1lIHN0YWdlIElEcywgY29tbWl0IGxvZyBxdWVyaWVzLCBhbmQga2V5ZWQgcmVjb3JkZXIgYmFzZSBjbGFzcy5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaW1wb3J0IHsgcGFyc2VSdW50aW1lU3RhZ2VJZCwgZmluZExhc3RXcml0ZXIsIEtleWVkUmVjb3JkZXIgfSBmcm9tICdmb290cHJpbnRqcy90cmFjZSc7XG4gKlxuICogLy8gUGFyc2UgYSBydW50aW1lU3RhZ2VJZFxuICogY29uc3QgeyBzdGFnZUlkLCBleGVjdXRpb25JbmRleCB9ID0gcGFyc2VSdW50aW1lU3RhZ2VJZCgnY2FsbC1sbG0jNScpO1xuICpcbiAqIC8vIEJhY2t0cmFjazogd2hvIHdyb3RlICdzeXN0ZW1Qcm9tcHQnIGJlZm9yZSBzdGFnZSBhdCBpZHggOD9cbiAqIGNvbnN0IHdyaXRlciA9IGZpbmRMYXN0V3JpdGVyKGNvbW1pdExvZywgJ3N5c3RlbVByb21wdCcsIDgpO1xuICpcbiAqIC8vIEJ1aWxkIGEga2V5ZWQgcmVjb3JkZXJcbiAqIGNsYXNzIE15UmVjb3JkZXIgZXh0ZW5kcyBLZXllZFJlY29yZGVyPE15RW50cnk+IHsgLi4uIH1cbiAqIGBgYFxuICovXG5cbi8vIFJ1bnRpbWUgc3RhZ2UgSUQg4oCUIHVuaXF1ZSBleGVjdXRpb24gc3RlcCBpZGVudGlmaWVyc1xuZXhwb3J0IHR5cGUgeyBFeGVjdXRpb25Db3VudGVyIH0gZnJvbSAnLi9saWIvZW5naW5lL3J1bnRpbWVTdGFnZUlkLmpzJztcbmV4cG9ydCB7IGJ1aWxkUnVudGltZVN0YWdlSWQsIGNyZWF0ZUV4ZWN1dGlvbkNvdW50ZXIsIHBhcnNlUnVudGltZVN0YWdlSWQgfSBmcm9tICcuL2xpYi9lbmdpbmUvcnVudGltZVN0YWdlSWQuanMnO1xuXG4vLyBDb21taXQgbG9nIHF1ZXJpZXMg4oCUIHR5cGVkIHV0aWxpdGllcyBmb3IgYmFja3RyYWNraW5nXG5leHBvcnQgeyBmaW5kQ29tbWl0LCBmaW5kQ29tbWl0cywgZmluZExhc3RXcml0ZXIgfSBmcm9tICcuL2xpYi9tZW1vcnkvY29tbWl0TG9nVXRpbHMuanMnO1xuXG4vLyBLZXllZFJlY29yZGVyIOKAlCBiYXNlIGNsYXNzIGZvciBNYXAtYmFzZWQgcmVjb3JkZXJzXG5leHBvcnQgeyBLZXllZFJlY29yZGVyIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvS2V5ZWRSZWNvcmRlci5qcyc7XG4iXX0=
@@ -48,3 +48,7 @@ export type { BranchResult, BranchResults, SerializedPipelineStructure as Engine
48
48
  export { NullControlFlowNarrativeGenerator } from './lib/engine/index.js';
49
49
  export type { CallExtractorFn, ExecuteNodeFn, GetStagePathFn, RunStageFn } from './lib/engine/index.js';
50
50
  export { applyOutputMapping, ChildrenExecutor, computeNodeType, ContinuationResolver, createSubflowHandlerDeps, DeciderHandler, DEFAULT_MAX_ITERATIONS, ExtractorRunner, extractParentScopeValues, getInitialScopeValues, NodeResolver, RuntimeStructureManager, seedSubflowGlobalStore, SelectorHandler, StageRunner, SubflowExecutor, } from './lib/engine/index.js';
51
+ export type { ExecutionCounter } from './lib/engine/runtimeStageId.js';
52
+ export { buildRuntimeStageId, createExecutionCounter, parseRuntimeStageId } from './lib/engine/runtimeStageId.js';
53
+ export { findCommit, findCommits, findLastWriter } from './lib/memory/commitLogUtils.js';
54
+ export { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Typed utilities for querying the commit log.
3
+ *
4
+ * The commitLog is an ordered array of CommitBundle — one per stage commit.
5
+ * These helpers provide type-safe queries without (b: any) casts.
6
+ */
7
+ import type { CommitBundle } from './types.js';
8
+ /** Find the first commit by stageId, optionally filtering by a written key. */
9
+ export declare function findCommit(commitLog: CommitBundle[], stageId: string, key?: string): CommitBundle | undefined;
10
+ /** Find all commits by stageId. */
11
+ export declare function findCommits(commitLog: CommitBundle[], stageId: string): CommitBundle[];
12
+ /** Find the last commit that wrote a specific key (for backtracking). */
13
+ export declare function findLastWriter(commitLog: CommitBundle[], key: string, beforeIdx?: number): CommitBundle | undefined;
@@ -0,0 +1,33 @@
1
+ /**
2
+ * KeyedRecorder<T> — base class for Map-based recorders keyed by runtimeStageId.
3
+ *
4
+ * Provides typed key-value storage with O(1) lookup, insertion-ordered iteration,
5
+ * and common accessors. Recorder implementations extend this and call store()
6
+ * from their event hooks.
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * class TokenRecorder extends KeyedRecorder<LLMCallEntry> {
11
+ * onLLMCall(event: LLMCallEvent) {
12
+ * this.store(event.runtimeStageId, { model: event.model, ... });
13
+ * }
14
+ * getStats() { return aggregate(this.values()); }
15
+ * }
16
+ * ```
17
+ */
18
+ export declare abstract class KeyedRecorder<T> {
19
+ abstract readonly id: string;
20
+ private readonly data;
21
+ /** Store an entry keyed by runtimeStageId. */
22
+ protected store(runtimeStageId: string, entry: T): void;
23
+ /** O(1) lookup by runtimeStageId. */
24
+ getByKey(runtimeStageId: string): T | undefined;
25
+ /** All entries as a read-only Map (insertion-ordered). */
26
+ getMap(): ReadonlyMap<string, T>;
27
+ /** All entries as an array (insertion-ordered). */
28
+ values(): T[];
29
+ /** Number of entries stored. */
30
+ get size(): number;
31
+ /** Clear all stored data. Called by executor before each run(). */
32
+ clear(): void;
33
+ }
@@ -1,2 +1,3 @@
1
1
  export type { CompositeSnapshot } from './CompositeRecorder.js';
2
2
  export { CompositeRecorder } from './CompositeRecorder.js';
3
+ export { KeyedRecorder } from './KeyedRecorder.js';
@@ -0,0 +1,23 @@
1
+ /**
2
+ * footprintjs/trace — Execution tracing, debugging, and backtracking utilities.
3
+ *
4
+ * Runtime stage IDs, commit log queries, and keyed recorder base class.
5
+ *
6
+ * @example
7
+ * ```typescript
8
+ * import { parseRuntimeStageId, findLastWriter, KeyedRecorder } from 'footprintjs/trace';
9
+ *
10
+ * // Parse a runtimeStageId
11
+ * const { stageId, executionIndex } = parseRuntimeStageId('call-llm#5');
12
+ *
13
+ * // Backtrack: who wrote 'systemPrompt' before stage at idx 8?
14
+ * const writer = findLastWriter(commitLog, 'systemPrompt', 8);
15
+ *
16
+ * // Build a keyed recorder
17
+ * class MyRecorder extends KeyedRecorder<MyEntry> { ... }
18
+ * ```
19
+ */
20
+ export type { ExecutionCounter } from './lib/engine/runtimeStageId.js';
21
+ export { buildRuntimeStageId, createExecutionCounter, parseRuntimeStageId } from './lib/engine/runtimeStageId.js';
22
+ export { findCommit, findCommits, findLastWriter } from './lib/memory/commitLogUtils.js';
23
+ export { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "footprintjs",
3
- "version": "4.6.0",
3
+ "version": "4.7.0",
4
4
  "description": "Explainable backend flows — automatic causal traces, decision evidence, and MCP tool generation for AI agents",
5
5
  "license": "MIT",
6
6
  "author": "Sanjay Krishna Anbalagan",
@@ -82,6 +82,11 @@
82
82
  "types": "./dist/types/recorders.d.ts",
83
83
  "import": "./dist/esm/recorders.js",
84
84
  "require": "./dist/recorders.js"
85
+ },
86
+ "./trace": {
87
+ "types": "./dist/types/trace.d.ts",
88
+ "import": "./dist/esm/trace.js",
89
+ "require": "./dist/trace.js"
85
90
  }
86
91
  },
87
92
  "lint-staged": {