footprintjs 4.7.0 → 4.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CLAUDE.md CHANGED
@@ -279,10 +279,12 @@ class MyRecorder extends KeyedRecorder<MyEntry> {
279
279
  this.store(event.runtimeStageId, { ... }); // protected
280
280
  }
281
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
282
+ // Three operations on auto-collected data:
283
+ recorder.getByKey('call-llm#5'); // Translate: per-step value
284
+ recorder.aggregate((sum, e) => sum + e.value, 0); // Aggregate: grand total
285
+ recorder.accumulate((sum, e) => sum + e.value, 0, visibleKeys); // Accumulate: progressive up to slider
286
+ recorder.filterByKeys(visibleKeys); // Filter: entries up to slider
287
+ recorder.getMap(); // Raw Map access
286
288
  ```
287
289
 
288
290
  **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`.
@@ -233,6 +233,7 @@ export class StageContext {
233
233
  getSnapshot() {
234
234
  const snapshot = {
235
235
  id: this.stageId,
236
+ runtimeStageId: this.runtimeStageId || undefined,
236
237
  name: this.stageName,
237
238
  isDecider: this.isDecider,
238
239
  isFork: this.isFork,
@@ -270,4 +271,4 @@ export class StageContext {
270
271
  return snapshot;
271
272
  }
272
273
  }
273
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"StageContext.js","sourceRoot":"","sources":["../../../../src/lib/memory/StageContext.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AAG/D,OAAO,EAAE,iBAAiB,EAAE,MAAM,wBAAwB,CAAC;AAE3D,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAEzC,MAAM,OAAO,YAAY;IAoCvB,YACE,KAAa,EACb,IAAY,EACZ,OAAe,EACf,YAA0B,EAC1B,QAAiB,EACjB,QAAmB,EACnB,SAAmB;QAtCd,cAAS,GAAG,EAAE,CAAC;QAGtB,qFAAqF;QAC9E,mBAAc,GAAG,EAAE,CAAC;QAcpB,UAAK,GAAwB,IAAI,mBAAmB,EAAE,CAAC;QAE9D,iFAAiF;QACzE,iBAAY,GAA+E,EAAE,CAAC;QAEtG,mEAAmE;QAC3D,gBAAW,GAA4B,EAAE,CAAC;QAgBhD,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;QACjC,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC,SAAS,CAAC;QAC7B,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;IACtB,CAAC;IAED,iEAAiE;IACjE,eAAe;QACb,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED,mFAAmF;IACnF,oBAAoB;QAClB,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,IAAI,CAAC,MAAM,GAAG,IAAI,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC,CAAC;QACpE,CAAC;QACD,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED,wEAAwE;IAChE,aAAa,CAAC,IAAc,EAAE,GAAW;QAC/C,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,KAAK,EAAE,EAAE,CAAC;YACrC,OAAO,CAAC,GAAG,IAAI,EAAE,GAAG,CAAC,CAAC;QACxB,CAAC;QACD,OAAO,CAAC,MAAM,EAAE,IAAI,CAAC,KAAK,EAAE,GAAG,IAAI,EAAE,GAAG,CAAC,CAAC;IAC5C,CAAC;IAED,0EAA0E;IAE1E,KAAK,CAAC,IAAc,EAAE,GAAW,EAAE,KAAc,EAAE,YAAY,GAAG,KAAK;QACrE,IAAI,CAAC,oBAAoB,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,KAAK,EAAE,YAAY,CAAC,CAAC;IACtF,CAAC;IAED,GAAG,CAAC,IAAc,EAAE,GAAW,EAAE,KAAc;QAC7C,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,KAAK,CAAC,IAAc,EAAE,GAAW,EAAE,KAAc;QAC/C,IAAI,CAAC,oBAAoB,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;IAC1E,CAAC;IAED,SAAS,CACP,IAAc,EACd,GAAW,EACX,KAAc,EACd,YAAsB,EACtB,WAAoB,EACpB,iBAAoC;QAEpC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,GAAG,EAAE,KAAK,EAAE,YAAY,aAAZ,YAAY,cAAZ,YAAY,GAAI,KAAK,CAAC,CAAC;QACpD,oEAAoE;QACpE,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,EAAE,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;QACjE,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG;YAC3B,KAAK,EAAE,YAAY,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,eAAe,CAAC,KAAK,CAAC;YAC3D,SAAS,EAAE,iBAAiB,aAAjB,iBAAiB,cAAjB,iBAAiB,GAAI,KAAK;SACtC,CAAC;QACF,IAAI,WAAW,EAAE,CAAC;YAChB,MAAM,MAAM,GAAG,WAAW,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,WAAW,WAAW,EAAE,CAAC;YACpF,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;QACvC,CAAC;IACH,CAAC;IAED,YAAY,CAAC,IAAc,EAAE,GAAW,EAAE,KAAc,EAAE,WAAoB,EAAE,YAAsB;QACpG,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC;QAC7B,oEAAoE;QACpE,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,EAAE,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;QACjE,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG;YAC3B,KAAK,EAAE,YAAY,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,eAAe,CAAC,KAAK,CAAC;YAC3D,SAAS,EAAE,QAAQ;SACpB,CAAC;QACF,IAAI,WAAW,EAAE,CAAC;YAChB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;QAC5C,CAAC;IACH,CAAC;IAED,OAAO,CAAC,GAAW,EAAE,KAAc;QACjC,IAAI,CAAC,KAAK,CAAC,EAAE,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC;IAC7B,CAAC;IAED,SAAS,CAAC,GAAW,EAAE,KAAc,EAAE,WAAoB;QACzD,IAAI,CAAC,oBAAoB,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;QAC9C,IAAI,WAAW,EAAE,CAAC;YAChB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;QAC5C,CAAC;IACH,CAAC;IAED,mBAAmB,CAAC,GAAW,EAAE,KAAc;QAC7C,IAAI,CAAC,oBAAoB,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;IAChD,CAAC;IAED,aAAa,CAAC,IAAc,EAAE,GAAW,EAAE,KAAgB,EAAE,WAAoB;QAC/E,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;QAC1C,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,QAAQ,EAAE,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC;QAC9E,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,CAAC,CAAC;IACxD,CAAC;IAED,WAAW,CAAC,IAAc,EAAE,GAAW,EAAE,GAA4B,EAAE,WAAoB;QACzF,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;QAC1C,MAAM,MAAM,GACV,QAAQ,IAAI,OAAO,QAAQ,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC;YAClE,CAAC,CAAC,EAAE,GAAI,QAAoC,EAAE,GAAG,GAAG,EAAE;YACtD,CAAC,CAAC,EAAE,GAAG,GAAG,EAAE,CAAC;QACjB,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,CAAC,CAAC;IACxD,CAAC;IAED,0EAA0E;IAE1E,QAAQ,CAAC,IAAc,EAAE,GAAY,EAAE,WAAoB;QACzD,MAAM,GAAG,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QACxC,MAAM,SAAS,GAAG,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,GAAa,CAAC,CAAC,CAAC;QACnE,MAAM,KAAK,GAAG,OAAO,SAAS,KAAK,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,EAAE,GAAG,CAAC,CAAC;QAC/G,wDAAwD;QACxD,IAAI,GAAG,KAAK,SAAS,EAAE,CAAC;YACtB,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,EAAE,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;YACjE,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,GAAG,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QACvF,CAAC;QACD,IAAI,WAAW,EAAE,CAAC;YAChB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,EAAE,UAAU,WAAW,EAAE,CAAC,CAAC;QACxD,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED;oFACgF;IAChF,cAAc,CAAC,IAAc,EAAE,GAAY;QACzC,MAAM,GAAG,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QACxC,MAAM,SAAS,GAAG,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,GAAa,CAAC,CAAC,CAAC;QACnE,OAAO,OAAO,SAAS,KAAK,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,EAAE,GAAG,CAAC,CAAC;IAC1G,CAAC;IAED,OAAO,CAAC,GAAW;QACjB,OAAO,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,EAAE,EAAE,EAAE,GAAG,CAAC,CAAC;IACzD,CAAC;IAED,SAAS,CAAC,GAAW;QACnB,OAAO,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,GAAG,CAAC,CAAC;IACjD,CAAC;IAED,QAAQ;QACN,OAAO,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC;IACtC,CAAC;IAED,QAAQ;QACN,OAAO,IAAI,CAAC,KAAK,CAAC;IACpB,CAAC;IAED,0EAA0E;IAE1E;oEACgE;IAChE,iBAAiB,CACf,QAAyG;QAEzG,IAAI,CAAC,eAAe,GAAG,QAAQ,CAAC;IAClC,CAAC;IAED,MAAM;;QACJ,MAAM,GAAG,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QACxC,MAAM,MAAM,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC;QAC5B,MAAM,YAAY,GAAG;YACnB,GAAG,MAAM;YACT,KAAK,EAAE,IAAI,CAAC,SAAS;YACrB,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,cAAc,EAAE,IAAI,CAAC,cAAc;SACpC,CAAC;QAEF,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,YAAY,CAAC,SAAS,EAAE,YAAY,CAAC,OAAO,EAAE,YAAY,CAAC,KAAK,CAAC,CAAC;QAE/F,MAAM,iBAAiB,GAAG,WAAW,CAAC,YAAY,CAAC,SAAS,EAAE,YAAY,CAAC,aAAa,CAAC,CAAC;QAC1F,MAAM,eAAe,GAAG,WAAW,CAAC,YAAY,CAAC,OAAO,EAAE,YAAY,CAAC,aAAa,CAAC,CAAC;QACtF,MAAA,IAAI,CAAC,QAAQ,0CAAE,MAAM,CAAC;YACpB,GAAG,YAAY;YACf,aAAa,EAAE,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC;YAC9D,SAAS,EAAE,iBAAiB;YAC5B,OAAO,EAAE,eAAe;SACzB,CAAC,CAAC;QAEH,uDAAuD;QACvD,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,IAAI,CAAC,eAAe,CAAC,EAAE,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC;QACjD,CAAC;IACH,CAAC;IAED,0EAA0E;IAE1E,UAAU,CAAC,IAAY,EAAE,SAAiB,EAAE,OAAe,EAAE,SAAS,GAAG,KAAK;QAC5E,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACf,IAAI,CAAC,IAAI,GAAG,IAAI,YAAY,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,CAAC,YAAY,EAAE,EAAE,EAAE,IAAI,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC;YACxG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QAC1B,CAAC;QACD,OAAO,IAAI,CAAC,IAAI,CAAC;IACnB,CAAC;IAED,WAAW,CAAC,KAAa,EAAE,QAAgB,EAAE,SAAiB,EAAE,OAAe,EAAE,SAAS,GAAG,KAAK;QAChG,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACnB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACrB,CAAC;QACD,MAAM,KAAK,GAAG,IAAI,YAAY,CAAC,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,CAAC,YAAY,EAAE,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC;QACjH,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,OAAO,KAAK,CAAC;IACf,CAAC;IAED,aAAa,CAAC,IAAY,EAAE,SAAiB,EAAE,OAAe;QAC5D,OAAO,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;IACzD,CAAC;IAED,YAAY;QACV,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,OAAO,IAAI,CAAC;IACd,CAAC;IAED,SAAS;QACP,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACnB,OAAO,IAAI,CAAC;IACd,CAAC;IAED,0EAA0E;IAE1E,MAAM,CAAC,GAAW,EAAE,KAAc,EAAE,IAAe;QACjD,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACtC,CAAC;IAED,MAAM,CAAC,GAAW,EAAE,KAAc,EAAE,IAAe;QACjD,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACtC,CAAC;IAED,SAAS,CAAC,GAAW,EAAE,KAAc,EAAE,IAAe;QACpD,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACzC,CAAC;IAED,SAAS,CAAC,GAAW,EAAE,KAAc,EAAE,IAAe;QACpD,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACzC,CAAC;IAED,OAAO,CAAC,GAAW,EAAE,KAAc,EAAE,IAAe;QAClD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACvC,CAAC;IAED,OAAO,CAAC,GAAW,EAAE,KAAc,EAAE,IAAe;QAClD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACvC,CAAC;IAED,QAAQ,CAAC,GAAW,EAAE,KAAc,EAAE,IAAe;QACnD,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;IACxC,CAAC;IAED,mBAAmB,CACjB,IAAqB,EACrB,WAAmB,EACnB,OAAqG;QAErG,MAAM,WAAW,GAAgB,EAAE,IAAI,EAAE,WAAW,EAAE,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,GAAG,OAAO,EAAE,CAAC;QAC1F,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;IACzC,CAAC;IAED,0EAA0E;IAE1E,UAAU;QACR,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,KAAK,EAAE;YAAE,OAAO,IAAI,CAAC,SAAS,CAAC;QAC5D,OAAO,GAAG,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;IAC3C,CAAC;IAED,WAAW;QACT,MAAM,QAAQ,GAAkB;YAC9B,EAAE,EAAE,IAAI,CAAC,OAAO;YAChB,IAAI,EAAE,IAAI,CAAC,SAAS;YACpB,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU;YAC3B,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,YAAY;YAC/B,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,aAAa;YACjC,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,WAAW;SAC9B,CAAC;QACF,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC9C,kEAAkE;YAClE,MAAM,MAAM,GAA4B,EAAE,CAAC;YAC3C,KAAK,MAAM,CAAC,CAAC,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC;gBAC3D,MAAM,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;YAC1B,CAAC;YACD,QAAQ,CAAC,WAAW,GAAG,MAAM,CAAC;QAChC,CAAC;QACD,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC7C,QAAQ,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC;QACzC,CAAC;QACD,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,QAAQ,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QAC1C,CAAC;QACD,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,QAAQ,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;QACtC,CAAC;QACD,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvC,QAAQ,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC;QAClD,CAAC;QACD,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACd,QAAQ,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;QAC1C,CAAC;QACD,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAClB,QAAQ,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;QAChE,CAAC;QACD,OAAO,QAAQ,CAAC;IAClB,CAAC;CACF","sourcesContent":["/**\n * StageContext — Execution context for a single stage in a flowchart run\n *\n * Like a stack frame in a compiler/runtime:\n * - Reference to SharedMemory (accessing heap memory)\n * - TransactionBuffer for staging mutations (transaction buffer)\n * - Links to parent/child/next contexts (call stack frames)\n * - DiagnosticCollector for logs, errors, metrics\n */\n\nimport { DiagnosticCollector } from './DiagnosticCollector.js';\nimport { EventLog } from './EventLog.js';\nimport { SharedMemory } from './SharedMemory.js';\nimport { TransactionBuffer } from './TransactionBuffer.js';\nimport type { FlowControlType, FlowMessage, StageSnapshot } from './types.js';\nimport { redactPatch } from './utils.js';\n\nexport class StageContext {\n  private sharedMemory: SharedMemory;\n  private buffer?: TransactionBuffer;\n  private eventLog?: EventLog;\n\n  public stageName = '';\n  /** Unique stage identifier from the builder (matches spec node id). */\n  public stageId: string;\n  /** Unique per-execution-step identifier. Set by traverser before stage execution. */\n  public runtimeStageId = '';\n  public runId: string;\n  public branchId?: string;\n  public isDecider: boolean;\n  public isFork: boolean;\n  /** Human-readable description from builder (set by traverser before execution). */\n  public description?: string;\n  /** Subflow identifier (set by traverser when this is a subflow entry point). */\n  public subflowId?: string;\n\n  public parent?: StageContext;\n  public next?: StageContext;\n  public children?: StageContext[];\n\n  public debug: DiagnosticCollector = new DiagnosticCollector();\n\n  /** Tracks user-level writes (pre-namespace) for the memory view and onCommit. */\n  private _stageWrites: Record<string, { value: unknown; operation: 'set' | 'update' | 'delete' }> = {};\n\n  /** Tracks user-level reads (pre-namespace) for the memory view. */\n  private _stageReads: Record<string, unknown> = {};\n\n  /** Observer called after commit() — used by ScopeFacade to fire Recorder.onCommit. */\n  private _commitObserver?: (\n    mutations: Record<string, { value: unknown; operation: 'set' | 'update' | 'delete' }>,\n  ) => void;\n\n  constructor(\n    runId: string,\n    name: string,\n    stageId: string,\n    sharedMemory: SharedMemory,\n    branchId?: string,\n    eventLog?: EventLog,\n    isDecider?: boolean,\n  ) {\n    this.runId = runId;\n    this.stageName = name;\n    this.stageId = stageId;\n    this.sharedMemory = sharedMemory;\n    this.branchId = branchId;\n    this.eventLog = eventLog;\n    this.isDecider = !!isDecider;\n    this.isFork = false;\n  }\n\n  /** Returns the SharedMemory instance (needed by scope layer). */\n  getSharedMemory(): SharedMemory {\n    return this.sharedMemory;\n  }\n\n  /** Lazily creates the transaction buffer (pay clone cost only if stage writes). */\n  getTransactionBuffer(): TransactionBuffer {\n    if (!this.buffer) {\n      this.buffer = new TransactionBuffer(this.sharedMemory.getState());\n    }\n    return this.buffer;\n  }\n\n  /** Builds an absolute path inside the shared memory (run namespace). */\n  private withNamespace(path: string[], key: string): string[] {\n    if (!this.runId || this.runId === '') {\n      return [...path, key];\n    }\n    return ['runs', this.runId, ...path, key];\n  }\n\n  // ── Write operations ───────────────────────────────────────────────────\n\n  patch(path: string[], key: string, value: unknown, shouldRedact = false) {\n    this.getTransactionBuffer().set(this.withNamespace(path, key), value, shouldRedact);\n  }\n\n  set(path: string[], key: string, value: unknown) {\n    this.patch(path, key, value);\n  }\n\n  merge(path: string[], key: string, value: unknown) {\n    this.getTransactionBuffer().merge(this.withNamespace(path, key), value);\n  }\n\n  setObject(\n    path: string[],\n    key: string,\n    value: unknown,\n    shouldRedact?: boolean,\n    description?: string,\n    operationOverride?: 'set' | 'delete',\n  ) {\n    this.patch(path, key, value, shouldRedact ?? false);\n    // Track user-level write (pre-namespace) for memory view + onCommit\n    const userKey = path.length > 0 ? [...path, key].join('.') : key;\n    this._stageWrites[userKey] = {\n      value: shouldRedact ? '[REDACTED]' : structuredClone(value),\n      operation: operationOverride ?? 'set',\n    };\n    if (description) {\n      const tagged = description.startsWith('[') ? description : `[WRITE] ${description}`;\n      this.debug.addLog('message', tagged);\n    }\n  }\n\n  updateObject(path: string[], key: string, value: unknown, description?: string, shouldRedact?: boolean) {\n    this.merge(path, key, value);\n    // Track user-level write (pre-namespace) for memory view + onCommit\n    const userKey = path.length > 0 ? [...path, key].join('.') : key;\n    this._stageWrites[userKey] = {\n      value: shouldRedact ? '[REDACTED]' : structuredClone(value),\n      operation: 'update',\n    };\n    if (description) {\n      this.debug.addLog('message', description);\n    }\n  }\n\n  setRoot(key: string, value: unknown) {\n    this.patch([], key, value);\n  }\n\n  setGlobal(key: string, value: unknown, description?: string) {\n    this.getTransactionBuffer().set([key], value);\n    if (description) {\n      this.debug.addLog('message', description);\n    }\n  }\n\n  updateGlobalContext(key: string, value: unknown) {\n    this.getTransactionBuffer().set([key], value);\n  }\n\n  appendToArray(path: string[], key: string, items: unknown[], description?: string) {\n    const existing = this.getValue(path, key);\n    const merged = Array.isArray(existing) ? [...existing, ...items] : [...items];\n    this.setObject(path, key, merged, false, description);\n  }\n\n  mergeObject(path: string[], key: string, obj: Record<string, unknown>, description?: string) {\n    const existing = this.getValue(path, key);\n    const merged =\n      existing && typeof existing === 'object' && !Array.isArray(existing)\n        ? { ...(existing as Record<string, unknown>), ...obj }\n        : { ...obj };\n    this.setObject(path, key, merged, false, description);\n  }\n\n  // ── Read operations ────────────────────────────────────────────────────\n\n  getValue(path: string[], key?: string, description?: string) {\n    const buf = this.getTransactionBuffer();\n    const fromPatch = buf.get(this.withNamespace(path, key as string));\n    const value = typeof fromPatch !== 'undefined' ? fromPatch : this.sharedMemory.getValue(this.runId, path, key);\n    // Track user-level read (pre-namespace) for memory view\n    if (key !== undefined) {\n      const userKey = path.length > 0 ? [...path, key].join('.') : key;\n      this._stageReads[userKey] = value !== undefined ? structuredClone(value) : undefined;\n    }\n    if (description) {\n      this.debug.addLog('message', `[READ] ${description}`);\n    }\n    return value;\n  }\n\n  /** Read state without tracking in _stageReads or paying structuredClone cost.\n   *  Used by ScopeFacade.getValueSilent() for array proxy internal operations. */\n  getValueDirect(path: string[], key?: string): unknown {\n    const buf = this.getTransactionBuffer();\n    const fromPatch = buf.get(this.withNamespace(path, key as string));\n    return typeof fromPatch !== 'undefined' ? fromPatch : this.sharedMemory.getValue(this.runId, path, key);\n  }\n\n  getRoot(key: string) {\n    return this.sharedMemory.getValue(this.runId, [], key);\n  }\n\n  getGlobal(key: string) {\n    return this.sharedMemory.getValue('', [], key);\n  }\n\n  getScope(): Record<string, unknown> {\n    return this.sharedMemory.getState();\n  }\n\n  getRunId(): string {\n    return this.runId;\n  }\n\n  // ── Commit ─────────────────────────────────────────────────────────────\n\n  /** Register an observer that fires after commit() applies patches.\n   *  Used by ScopeFacade to dispatch Recorder.onCommit events. */\n  setCommitObserver(\n    observer: (mutations: Record<string, { value: unknown; operation: 'set' | 'update' | 'delete' }>) => void,\n  ): void {\n    this._commitObserver = observer;\n  }\n\n  commit(): void {\n    const buf = this.getTransactionBuffer();\n    const bundle = buf.commit();\n    const commitBundle = {\n      ...bundle,\n      stage: this.stageName,\n      stageId: this.stageId,\n      runtimeStageId: this.runtimeStageId,\n    };\n\n    this.sharedMemory.applyPatch(commitBundle.overwrite, commitBundle.updates, commitBundle.trace);\n\n    const redactedOverwrite = redactPatch(commitBundle.overwrite, commitBundle.redactedPaths);\n    const redactedUpdates = redactPatch(commitBundle.updates, commitBundle.redactedPaths);\n    this.eventLog?.record({\n      ...commitBundle,\n      redactedPaths: Array.from(commitBundle.redactedPaths.values()),\n      overwrite: redactedOverwrite,\n      updates: redactedUpdates,\n    });\n\n    // Notify observer (ScopeFacade) with tracked mutations\n    if (this._commitObserver) {\n      this._commitObserver({ ...this._stageWrites });\n    }\n  }\n\n  // ── Tree navigation ────────────────────────────────────────────────────\n\n  createNext(path: string, stageName: string, stageId: string, isDecider = false): StageContext {\n    if (!this.next) {\n      this.next = new StageContext(path, stageName, stageId, this.sharedMemory, '', this.eventLog, isDecider);\n      this.next.parent = this;\n    }\n    return this.next;\n  }\n\n  createChild(runId: string, branchId: string, stageName: string, stageId: string, isDecider = false): StageContext {\n    if (!this.children) {\n      this.children = [];\n    }\n    const child = new StageContext(runId, stageName, stageId, this.sharedMemory, branchId, this.eventLog, isDecider);\n    child.parent = this;\n    this.children.push(child);\n    return child;\n  }\n\n  createDecider(path: string, stageName: string, stageId: string): StageContext {\n    return this.createNext(path, stageName, stageId, true);\n  }\n\n  setAsDecider(): StageContext {\n    this.isDecider = true;\n    return this;\n  }\n\n  setAsFork(): StageContext {\n    this.isFork = true;\n    return this;\n  }\n\n  // ── Diagnostics delegation ─────────────────────────────────────────────\n\n  addLog(key: string, value: unknown, path?: string[]) {\n    this.debug.addLog(key, value, path);\n  }\n\n  setLog(key: string, value: unknown, path?: string[]) {\n    this.debug.setLog(key, value, path);\n  }\n\n  addMetric(key: string, value: unknown, path?: string[]) {\n    this.debug.addMetric(key, value, path);\n  }\n\n  setMetric(key: string, value: unknown, path?: string[]) {\n    this.debug.setMetric(key, value, path);\n  }\n\n  addEval(key: string, value: unknown, path?: string[]) {\n    this.debug.addEval(key, value, path);\n  }\n\n  setEval(key: string, value: unknown, path?: string[]) {\n    this.debug.setEval(key, value, path);\n  }\n\n  addError(key: string, value: unknown, path?: string[]) {\n    this.debug.addError(key, value, path);\n  }\n\n  addFlowDebugMessage(\n    type: FlowControlType,\n    description: string,\n    options?: { targetStage?: string | string[]; rationale?: string; count?: number; iteration?: number },\n  ) {\n    const flowMessage: FlowMessage = { type, description, timestamp: Date.now(), ...options };\n    this.debug.addFlowMessage(flowMessage);\n  }\n\n  // ── Snapshot ───────────────────────────────────────────────────────────\n\n  getStageId(): string {\n    if (!this.runId || this.runId === '') return this.stageName;\n    return `${this.runId}.${this.stageName}`;\n  }\n\n  getSnapshot(): StageSnapshot {\n    const snapshot: StageSnapshot = {\n      id: this.stageId,\n      name: this.stageName,\n      isDecider: this.isDecider,\n      isFork: this.isFork,\n      logs: this.debug.logContext,\n      errors: this.debug.errorContext,\n      metrics: this.debug.metricContext,\n      evals: this.debug.evalContext,\n    };\n    if (Object.keys(this._stageWrites).length > 0) {\n      // Extract values only for the snapshot (strip operation metadata)\n      const writes: Record<string, unknown> = {};\n      for (const [k, entry] of Object.entries(this._stageWrites)) {\n        writes[k] = entry.value;\n      }\n      snapshot.stageWrites = writes;\n    }\n    if (Object.keys(this._stageReads).length > 0) {\n      snapshot.stageReads = this._stageReads;\n    }\n    if (this.description) {\n      snapshot.description = this.description;\n    }\n    if (this.subflowId) {\n      snapshot.subflowId = this.subflowId;\n    }\n    if (this.debug.flowMessages.length > 0) {\n      snapshot.flowMessages = this.debug.flowMessages;\n    }\n    if (this.next) {\n      snapshot.next = this.next.getSnapshot();\n    }\n    if (this.children) {\n      snapshot.children = this.children.map((c) => c.getSnapshot());\n    }\n    return snapshot;\n  }\n}\n"]}
274
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"StageContext.js","sourceRoot":"","sources":["../../../../src/lib/memory/StageContext.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * StageContext — Execution context for a single stage in a flowchart run\n *\n * Like a stack frame in a compiler/runtime:\n * - Reference to SharedMemory (accessing heap memory)\n * - TransactionBuffer for staging mutations (transaction buffer)\n * - Links to parent/child/next contexts (call stack frames)\n * - DiagnosticCollector for logs, errors, metrics\n */\n\nimport { DiagnosticCollector } from './DiagnosticCollector.js';\nimport { EventLog } from './EventLog.js';\nimport { SharedMemory } from './SharedMemory.js';\nimport { TransactionBuffer } from './TransactionBuffer.js';\nimport type { FlowControlType, FlowMessage, StageSnapshot } from './types.js';\nimport { redactPatch } from './utils.js';\n\nexport class StageContext {\n  private sharedMemory: SharedMemory;\n  private buffer?: TransactionBuffer;\n  private eventLog?: EventLog;\n\n  public stageName = '';\n  /** Unique stage identifier from the builder (matches spec node id). */\n  public stageId: string;\n  /** Unique per-execution-step identifier. Set by traverser before stage execution. */\n  public runtimeStageId = '';\n  public runId: string;\n  public branchId?: string;\n  public isDecider: boolean;\n  public isFork: boolean;\n  /** Human-readable description from builder (set by traverser before execution). */\n  public description?: string;\n  /** Subflow identifier (set by traverser when this is a subflow entry point). */\n  public subflowId?: string;\n\n  public parent?: StageContext;\n  public next?: StageContext;\n  public children?: StageContext[];\n\n  public debug: DiagnosticCollector = new DiagnosticCollector();\n\n  /** Tracks user-level writes (pre-namespace) for the memory view and onCommit. */\n  private _stageWrites: Record<string, { value: unknown; operation: 'set' | 'update' | 'delete' }> = {};\n\n  /** Tracks user-level reads (pre-namespace) for the memory view. */\n  private _stageReads: Record<string, unknown> = {};\n\n  /** Observer called after commit() — used by ScopeFacade to fire Recorder.onCommit. */\n  private _commitObserver?: (\n    mutations: Record<string, { value: unknown; operation: 'set' | 'update' | 'delete' }>,\n  ) => void;\n\n  constructor(\n    runId: string,\n    name: string,\n    stageId: string,\n    sharedMemory: SharedMemory,\n    branchId?: string,\n    eventLog?: EventLog,\n    isDecider?: boolean,\n  ) {\n    this.runId = runId;\n    this.stageName = name;\n    this.stageId = stageId;\n    this.sharedMemory = sharedMemory;\n    this.branchId = branchId;\n    this.eventLog = eventLog;\n    this.isDecider = !!isDecider;\n    this.isFork = false;\n  }\n\n  /** Returns the SharedMemory instance (needed by scope layer). */\n  getSharedMemory(): SharedMemory {\n    return this.sharedMemory;\n  }\n\n  /** Lazily creates the transaction buffer (pay clone cost only if stage writes). */\n  getTransactionBuffer(): TransactionBuffer {\n    if (!this.buffer) {\n      this.buffer = new TransactionBuffer(this.sharedMemory.getState());\n    }\n    return this.buffer;\n  }\n\n  /** Builds an absolute path inside the shared memory (run namespace). */\n  private withNamespace(path: string[], key: string): string[] {\n    if (!this.runId || this.runId === '') {\n      return [...path, key];\n    }\n    return ['runs', this.runId, ...path, key];\n  }\n\n  // ── Write operations ───────────────────────────────────────────────────\n\n  patch(path: string[], key: string, value: unknown, shouldRedact = false) {\n    this.getTransactionBuffer().set(this.withNamespace(path, key), value, shouldRedact);\n  }\n\n  set(path: string[], key: string, value: unknown) {\n    this.patch(path, key, value);\n  }\n\n  merge(path: string[], key: string, value: unknown) {\n    this.getTransactionBuffer().merge(this.withNamespace(path, key), value);\n  }\n\n  setObject(\n    path: string[],\n    key: string,\n    value: unknown,\n    shouldRedact?: boolean,\n    description?: string,\n    operationOverride?: 'set' | 'delete',\n  ) {\n    this.patch(path, key, value, shouldRedact ?? false);\n    // Track user-level write (pre-namespace) for memory view + onCommit\n    const userKey = path.length > 0 ? [...path, key].join('.') : key;\n    this._stageWrites[userKey] = {\n      value: shouldRedact ? '[REDACTED]' : structuredClone(value),\n      operation: operationOverride ?? 'set',\n    };\n    if (description) {\n      const tagged = description.startsWith('[') ? description : `[WRITE] ${description}`;\n      this.debug.addLog('message', tagged);\n    }\n  }\n\n  updateObject(path: string[], key: string, value: unknown, description?: string, shouldRedact?: boolean) {\n    this.merge(path, key, value);\n    // Track user-level write (pre-namespace) for memory view + onCommit\n    const userKey = path.length > 0 ? [...path, key].join('.') : key;\n    this._stageWrites[userKey] = {\n      value: shouldRedact ? '[REDACTED]' : structuredClone(value),\n      operation: 'update',\n    };\n    if (description) {\n      this.debug.addLog('message', description);\n    }\n  }\n\n  setRoot(key: string, value: unknown) {\n    this.patch([], key, value);\n  }\n\n  setGlobal(key: string, value: unknown, description?: string) {\n    this.getTransactionBuffer().set([key], value);\n    if (description) {\n      this.debug.addLog('message', description);\n    }\n  }\n\n  updateGlobalContext(key: string, value: unknown) {\n    this.getTransactionBuffer().set([key], value);\n  }\n\n  appendToArray(path: string[], key: string, items: unknown[], description?: string) {\n    const existing = this.getValue(path, key);\n    const merged = Array.isArray(existing) ? [...existing, ...items] : [...items];\n    this.setObject(path, key, merged, false, description);\n  }\n\n  mergeObject(path: string[], key: string, obj: Record<string, unknown>, description?: string) {\n    const existing = this.getValue(path, key);\n    const merged =\n      existing && typeof existing === 'object' && !Array.isArray(existing)\n        ? { ...(existing as Record<string, unknown>), ...obj }\n        : { ...obj };\n    this.setObject(path, key, merged, false, description);\n  }\n\n  // ── Read operations ────────────────────────────────────────────────────\n\n  getValue(path: string[], key?: string, description?: string) {\n    const buf = this.getTransactionBuffer();\n    const fromPatch = buf.get(this.withNamespace(path, key as string));\n    const value = typeof fromPatch !== 'undefined' ? fromPatch : this.sharedMemory.getValue(this.runId, path, key);\n    // Track user-level read (pre-namespace) for memory view\n    if (key !== undefined) {\n      const userKey = path.length > 0 ? [...path, key].join('.') : key;\n      this._stageReads[userKey] = value !== undefined ? structuredClone(value) : undefined;\n    }\n    if (description) {\n      this.debug.addLog('message', `[READ] ${description}`);\n    }\n    return value;\n  }\n\n  /** Read state without tracking in _stageReads or paying structuredClone cost.\n   *  Used by ScopeFacade.getValueSilent() for array proxy internal operations. */\n  getValueDirect(path: string[], key?: string): unknown {\n    const buf = this.getTransactionBuffer();\n    const fromPatch = buf.get(this.withNamespace(path, key as string));\n    return typeof fromPatch !== 'undefined' ? fromPatch : this.sharedMemory.getValue(this.runId, path, key);\n  }\n\n  getRoot(key: string) {\n    return this.sharedMemory.getValue(this.runId, [], key);\n  }\n\n  getGlobal(key: string) {\n    return this.sharedMemory.getValue('', [], key);\n  }\n\n  getScope(): Record<string, unknown> {\n    return this.sharedMemory.getState();\n  }\n\n  getRunId(): string {\n    return this.runId;\n  }\n\n  // ── Commit ─────────────────────────────────────────────────────────────\n\n  /** Register an observer that fires after commit() applies patches.\n   *  Used by ScopeFacade to dispatch Recorder.onCommit events. */\n  setCommitObserver(\n    observer: (mutations: Record<string, { value: unknown; operation: 'set' | 'update' | 'delete' }>) => void,\n  ): void {\n    this._commitObserver = observer;\n  }\n\n  commit(): void {\n    const buf = this.getTransactionBuffer();\n    const bundle = buf.commit();\n    const commitBundle = {\n      ...bundle,\n      stage: this.stageName,\n      stageId: this.stageId,\n      runtimeStageId: this.runtimeStageId,\n    };\n\n    this.sharedMemory.applyPatch(commitBundle.overwrite, commitBundle.updates, commitBundle.trace);\n\n    const redactedOverwrite = redactPatch(commitBundle.overwrite, commitBundle.redactedPaths);\n    const redactedUpdates = redactPatch(commitBundle.updates, commitBundle.redactedPaths);\n    this.eventLog?.record({\n      ...commitBundle,\n      redactedPaths: Array.from(commitBundle.redactedPaths.values()),\n      overwrite: redactedOverwrite,\n      updates: redactedUpdates,\n    });\n\n    // Notify observer (ScopeFacade) with tracked mutations\n    if (this._commitObserver) {\n      this._commitObserver({ ...this._stageWrites });\n    }\n  }\n\n  // ── Tree navigation ────────────────────────────────────────────────────\n\n  createNext(path: string, stageName: string, stageId: string, isDecider = false): StageContext {\n    if (!this.next) {\n      this.next = new StageContext(path, stageName, stageId, this.sharedMemory, '', this.eventLog, isDecider);\n      this.next.parent = this;\n    }\n    return this.next;\n  }\n\n  createChild(runId: string, branchId: string, stageName: string, stageId: string, isDecider = false): StageContext {\n    if (!this.children) {\n      this.children = [];\n    }\n    const child = new StageContext(runId, stageName, stageId, this.sharedMemory, branchId, this.eventLog, isDecider);\n    child.parent = this;\n    this.children.push(child);\n    return child;\n  }\n\n  createDecider(path: string, stageName: string, stageId: string): StageContext {\n    return this.createNext(path, stageName, stageId, true);\n  }\n\n  setAsDecider(): StageContext {\n    this.isDecider = true;\n    return this;\n  }\n\n  setAsFork(): StageContext {\n    this.isFork = true;\n    return this;\n  }\n\n  // ── Diagnostics delegation ─────────────────────────────────────────────\n\n  addLog(key: string, value: unknown, path?: string[]) {\n    this.debug.addLog(key, value, path);\n  }\n\n  setLog(key: string, value: unknown, path?: string[]) {\n    this.debug.setLog(key, value, path);\n  }\n\n  addMetric(key: string, value: unknown, path?: string[]) {\n    this.debug.addMetric(key, value, path);\n  }\n\n  setMetric(key: string, value: unknown, path?: string[]) {\n    this.debug.setMetric(key, value, path);\n  }\n\n  addEval(key: string, value: unknown, path?: string[]) {\n    this.debug.addEval(key, value, path);\n  }\n\n  setEval(key: string, value: unknown, path?: string[]) {\n    this.debug.setEval(key, value, path);\n  }\n\n  addError(key: string, value: unknown, path?: string[]) {\n    this.debug.addError(key, value, path);\n  }\n\n  addFlowDebugMessage(\n    type: FlowControlType,\n    description: string,\n    options?: { targetStage?: string | string[]; rationale?: string; count?: number; iteration?: number },\n  ) {\n    const flowMessage: FlowMessage = { type, description, timestamp: Date.now(), ...options };\n    this.debug.addFlowMessage(flowMessage);\n  }\n\n  // ── Snapshot ───────────────────────────────────────────────────────────\n\n  getStageId(): string {\n    if (!this.runId || this.runId === '') return this.stageName;\n    return `${this.runId}.${this.stageName}`;\n  }\n\n  getSnapshot(): StageSnapshot {\n    const snapshot: StageSnapshot = {\n      id: this.stageId,\n      runtimeStageId: this.runtimeStageId || undefined,\n      name: this.stageName,\n      isDecider: this.isDecider,\n      isFork: this.isFork,\n      logs: this.debug.logContext,\n      errors: this.debug.errorContext,\n      metrics: this.debug.metricContext,\n      evals: this.debug.evalContext,\n    };\n    if (Object.keys(this._stageWrites).length > 0) {\n      // Extract values only for the snapshot (strip operation metadata)\n      const writes: Record<string, unknown> = {};\n      for (const [k, entry] of Object.entries(this._stageWrites)) {\n        writes[k] = entry.value;\n      }\n      snapshot.stageWrites = writes;\n    }\n    if (Object.keys(this._stageReads).length > 0) {\n      snapshot.stageReads = this._stageReads;\n    }\n    if (this.description) {\n      snapshot.description = this.description;\n    }\n    if (this.subflowId) {\n      snapshot.subflowId = this.subflowId;\n    }\n    if (this.debug.flowMessages.length > 0) {\n      snapshot.flowMessages = this.debug.flowMessages;\n    }\n    if (this.next) {\n      snapshot.next = this.next.getSnapshot();\n    }\n    if (this.children) {\n      snapshot.children = this.children.map((c) => c.getSnapshot());\n    }\n    return snapshot;\n  }\n}\n"]}
@@ -4,4 +4,4 @@
4
4
  * Zero dependencies on old code or other libraries in this package.
5
5
  */
6
6
  export {};
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL21lbW9yeS90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7OztHQUlHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiB0eXBlcy50cyDigJQgQ29yZSB0eXBlIGRlZmluaXRpb25zIGZvciB0aGUgbWVtb3J5IGxpYnJhcnlcbiAqXG4gKiBaZXJvIGRlcGVuZGVuY2llcyBvbiBvbGQgY29kZSBvciBvdGhlciBsaWJyYXJpZXMgaW4gdGhpcyBwYWNrYWdlLlxuICovXG5cbi8vIOKUgOKUgCBQYXRjaCAmIFRyYWNlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4vKiogQSBmbGF0IGtleS12YWx1ZSBiYWcgcmVwcmVzZW50aW5nIGEgc3RhdGUgcGF0Y2ggKG92ZXJ3cml0ZSBvciBtZXJnZSkuICovXG5leHBvcnQgaW50ZXJmYWNlIE1lbW9yeVBhdGNoIHtcbiAgW2tleTogc3RyaW5nXTogYW55O1xufVxuXG4vKiogQSBzaW5nbGUgZW50cnkgaW4gdGhlIGNocm9ub2xvZ2ljYWwgb3BlcmF0aW9uIHRyYWNlLiAqL1xuZXhwb3J0IGludGVyZmFjZSBUcmFjZUVudHJ5IHtcbiAgLyoqIENhbm9uaWNhbCBwYXRoIHN0cmluZyAoc2VnbWVudHMgam9pbmVkIGJ5IERFTElNKS4gKi9cbiAgcGF0aDogc3RyaW5nO1xuICAvKiogJ3NldCcgPSBoYXJkIG92ZXJ3cml0ZSwgJ21lcmdlJyA9IGRlZXAgdW5pb24gbWVyZ2UuICovXG4gIHZlcmI6ICdzZXQnIHwgJ21lcmdlJztcbn1cblxuLyoqIFRoZSBhdG9taWMgYnVuZGxlIHByb2R1Y2VkIGJ5IFRyYW5zYWN0aW9uQnVmZmVyLmNvbW1pdCgpLiAqL1xuZXhwb3J0IGludGVyZmFjZSBDb21taXRCdW5kbGUge1xuICAvKiogQXV0by1hc3NpZ25lZCBzdGVwIGluZGV4IChzZXQgYnkgRXZlbnRMb2cucmVjb3JkKS4gKi9cbiAgaWR4PzogbnVtYmVyO1xuICAvKiogSHVtYW4tcmVhZGFibGUgc3RhZ2UgbmFtZS4gKi9cbiAgc3RhZ2U6IHN0cmluZztcbiAgLyoqIFN0YWJsZSBzdGFnZSBpZGVudGlmaWVyIChtYXRjaGVzIHNwZWMgbm9kZSBpZCkuICovXG4gIHN0YWdlSWQ6IHN0cmluZztcbiAgLyoqIFVuaXF1ZSBwZXItZXhlY3V0aW9uLXN0ZXAgaWRlbnRpZmllci4gRm9ybWF0OiBbc3ViZmxvd1BhdGgvXXN0YWdlSWQjZXhlY3V0aW9uSW5kZXggKi9cbiAgcnVudGltZVN0YWdlSWQ6IHN0cmluZztcbiAgLyoqIENocm9ub2xvZ2ljYWwgd3JpdGUgbG9nIGZvciBkZXRlcm1pbmlzdGljIHJlcGxheS4gKi9cbiAgdHJhY2U6IFRyYWNlRW50cnlbXTtcbiAgLyoqIFBhdGhzIHRoYXQgc2hvdWxkIGJlIHJlZGFjdGVkIGluIFVJIChzZW5zaXRpdmUgZGF0YSkuICovXG4gIHJlZGFjdGVkUGF0aHM6IHN0cmluZ1tdO1xuICAvKiogSGFyZCBvdmVyd3JpdGUgcGF0Y2hlcy4gKi9cbiAgb3ZlcndyaXRlOiBNZW1vcnlQYXRjaDtcbiAgLyoqIERlZXAgbWVyZ2UgcGF0Y2hlcy4gKi9cbiAgdXBkYXRlczogTWVtb3J5UGF0Y2g7XG59XG5cbi8vIOKUgOKUgCBGbG93IENvbnRyb2wgTmFycmF0aXZlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4vKiogVHlwZXMgb2YgY29udHJvbCBmbG93IGRlY2lzaW9ucyBjYXB0dXJlZCBieSB0aGUgZXhlY3V0aW9uIGVuZ2luZS4gKi9cbmV4cG9ydCB0eXBlIEZsb3dDb250cm9sVHlwZSA9ICduZXh0JyB8ICdicmFuY2gnIHwgJ2NoaWxkcmVuJyB8ICdzZWxlY3RlZCcgfCAnc3ViZmxvdycgfCAnbG9vcCc7XG5cbi8qKiBBIHNpbmdsZSBmbG93IGNvbnRyb2wgbmFycmF0aXZlIGVudHJ5LiAqL1xuZXhwb3J0IGludGVyZmFjZSBGbG93TWVzc2FnZSB7XG4gIHR5cGU6IEZsb3dDb250cm9sVHlwZTtcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgdGFyZ2V0U3RhZ2U/OiBzdHJpbmcgfCBzdHJpbmdbXTtcbiAgcmF0aW9uYWxlPzogc3RyaW5nO1xuICBjb3VudD86IG51bWJlcjtcbiAgaXRlcmF0aW9uPzogbnVtYmVyO1xuICB0aW1lc3RhbXA/OiBudW1iZXI7XG59XG5cbi8vIOKUgOKUgCBTdGFnZSBTbmFwc2hvdCDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcblxuLyoqIFNlcmlhbGlzYWJsZSByZXByZXNlbnRhdGlvbiBvZiBhIHN0YWdlJ3Mgc3RhdGUgKGZvciBkZWJ1Z2dpbmcgLyB2aXN1YWxpc2F0aW9uKS4gKi9cbmV4cG9ydCB0eXBlIFN0YWdlU25hcHNob3QgPSB7XG4gIGlkOiBzdHJpbmc7XG4gIG5hbWU/OiBzdHJpbmc7XG4gIC8qKiBIdW1hbi1yZWFkYWJsZSBkZXNjcmlwdGlvbiBvZiB3aGF0IHRoaXMgc3RhZ2UgZG9lcyAoZnJvbSBidWlsZGVyKS4gKi9cbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIC8qKiBTdWJmbG93IGlkZW50aWZpZXIg4oCUIHByZXNlbnQgd2hlbiB0aGlzIHN0YWdlIGlzIGEgc3ViZmxvdyBlbnRyeSBwb2ludC4gKi9cbiAgc3ViZmxvd0lkPzogc3RyaW5nO1xuICBpc0RlY2lkZXI/OiBib29sZWFuO1xuICBpc0Zvcms/OiBib29sZWFuO1xuICAvKiogVXNlci1sZXZlbCB3cml0ZXMgbWFkZSBieSB0aGlzIHN0YWdlIChwcmUtbmFtZXNwYWNlIGtleXMg4oaSIHZhbHVlcykuICovXG4gIHN0YWdlV3JpdGVzPzogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIC8qKiBVc2VyLWxldmVsIHJlYWRzIG1hZGUgYnkgdGhpcyBzdGFnZSAocHJlLW5hbWVzcGFjZSBrZXlzIOKGkiB2YWx1ZXMgYXQgcmVhZCB0aW1lKS4gKi9cbiAgc3RhZ2VSZWFkcz86IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBsb2dzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgZXJyb3JzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgbWV0cmljczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIGV2YWxzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgZmxvd01lc3NhZ2VzPzogRmxvd01lc3NhZ2VbXTtcbiAgbmV4dD86IFN0YWdlU25hcHNob3Q7XG4gIGNoaWxkcmVuPzogU3RhZ2VTbmFwc2hvdFtdO1xufTtcblxuLy8g4pSA4pSAIFNjb3BlIEZhY3Rvcnkg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbi8qKiBGb3J3YXJkLWRlY2xhcmVkIHNvIFN0YWdlQ29udGV4dCBjYW4gYWNjZXB0IGl0IHdpdGhvdXQgaW1wb3J0aW5nIHNjb3BlLy4gKi9cbmV4cG9ydCB0eXBlIFNjb3BlRmFjdG9yeTxUU2NvcGU+ID0gKGNvcmU6IFN0YWdlQ29udGV4dCwgc3RhZ2VOYW1lOiBzdHJpbmcsIHJlYWRPbmx5Q29udGV4dD86IHVua25vd24pID0+IFRTY29wZTtcblxuLy8g4pSA4pSAIFN0YWdlQ29udGV4dCAoZm9yd2FyZCByZWZlcmVuY2UgZm9yIFNjb3BlRmFjdG9yeSkg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbi8vIFRoZSBhY3R1YWwgY2xhc3MgbGl2ZXMgaW4gU3RhZ2VDb250ZXh0LnRzOyB3ZSBqdXN0IG5lZWQgdGhlIHR5cGUgaGVyZSBmb3Jcbi8vIHRoZSBTY29wZUZhY3RvcnkgZ2VuZXJpYy4gVHlwZVNjcmlwdCdzIGltcG9ydC10eXBlIGhhbmRsZXMgdGhpczpcbmltcG9ydCB0eXBlIHsgU3RhZ2VDb250ZXh0IH0gZnJvbSAnLi9TdGFnZUNvbnRleHQuanMnO1xuIl19
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL21lbW9yeS90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7OztHQUlHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiB0eXBlcy50cyDigJQgQ29yZSB0eXBlIGRlZmluaXRpb25zIGZvciB0aGUgbWVtb3J5IGxpYnJhcnlcbiAqXG4gKiBaZXJvIGRlcGVuZGVuY2llcyBvbiBvbGQgY29kZSBvciBvdGhlciBsaWJyYXJpZXMgaW4gdGhpcyBwYWNrYWdlLlxuICovXG5cbi8vIOKUgOKUgCBQYXRjaCAmIFRyYWNlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4vKiogQSBmbGF0IGtleS12YWx1ZSBiYWcgcmVwcmVzZW50aW5nIGEgc3RhdGUgcGF0Y2ggKG92ZXJ3cml0ZSBvciBtZXJnZSkuICovXG5leHBvcnQgaW50ZXJmYWNlIE1lbW9yeVBhdGNoIHtcbiAgW2tleTogc3RyaW5nXTogYW55O1xufVxuXG4vKiogQSBzaW5nbGUgZW50cnkgaW4gdGhlIGNocm9ub2xvZ2ljYWwgb3BlcmF0aW9uIHRyYWNlLiAqL1xuZXhwb3J0IGludGVyZmFjZSBUcmFjZUVudHJ5IHtcbiAgLyoqIENhbm9uaWNhbCBwYXRoIHN0cmluZyAoc2VnbWVudHMgam9pbmVkIGJ5IERFTElNKS4gKi9cbiAgcGF0aDogc3RyaW5nO1xuICAvKiogJ3NldCcgPSBoYXJkIG92ZXJ3cml0ZSwgJ21lcmdlJyA9IGRlZXAgdW5pb24gbWVyZ2UuICovXG4gIHZlcmI6ICdzZXQnIHwgJ21lcmdlJztcbn1cblxuLyoqIFRoZSBhdG9taWMgYnVuZGxlIHByb2R1Y2VkIGJ5IFRyYW5zYWN0aW9uQnVmZmVyLmNvbW1pdCgpLiAqL1xuZXhwb3J0IGludGVyZmFjZSBDb21taXRCdW5kbGUge1xuICAvKiogQXV0by1hc3NpZ25lZCBzdGVwIGluZGV4IChzZXQgYnkgRXZlbnRMb2cucmVjb3JkKS4gKi9cbiAgaWR4PzogbnVtYmVyO1xuICAvKiogSHVtYW4tcmVhZGFibGUgc3RhZ2UgbmFtZS4gKi9cbiAgc3RhZ2U6IHN0cmluZztcbiAgLyoqIFN0YWJsZSBzdGFnZSBpZGVudGlmaWVyIChtYXRjaGVzIHNwZWMgbm9kZSBpZCkuICovXG4gIHN0YWdlSWQ6IHN0cmluZztcbiAgLyoqIFVuaXF1ZSBwZXItZXhlY3V0aW9uLXN0ZXAgaWRlbnRpZmllci4gRm9ybWF0OiBbc3ViZmxvd1BhdGgvXXN0YWdlSWQjZXhlY3V0aW9uSW5kZXggKi9cbiAgcnVudGltZVN0YWdlSWQ6IHN0cmluZztcbiAgLyoqIENocm9ub2xvZ2ljYWwgd3JpdGUgbG9nIGZvciBkZXRlcm1pbmlzdGljIHJlcGxheS4gKi9cbiAgdHJhY2U6IFRyYWNlRW50cnlbXTtcbiAgLyoqIFBhdGhzIHRoYXQgc2hvdWxkIGJlIHJlZGFjdGVkIGluIFVJIChzZW5zaXRpdmUgZGF0YSkuICovXG4gIHJlZGFjdGVkUGF0aHM6IHN0cmluZ1tdO1xuICAvKiogSGFyZCBvdmVyd3JpdGUgcGF0Y2hlcy4gKi9cbiAgb3ZlcndyaXRlOiBNZW1vcnlQYXRjaDtcbiAgLyoqIERlZXAgbWVyZ2UgcGF0Y2hlcy4gKi9cbiAgdXBkYXRlczogTWVtb3J5UGF0Y2g7XG59XG5cbi8vIOKUgOKUgCBGbG93IENvbnRyb2wgTmFycmF0aXZlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4vKiogVHlwZXMgb2YgY29udHJvbCBmbG93IGRlY2lzaW9ucyBjYXB0dXJlZCBieSB0aGUgZXhlY3V0aW9uIGVuZ2luZS4gKi9cbmV4cG9ydCB0eXBlIEZsb3dDb250cm9sVHlwZSA9ICduZXh0JyB8ICdicmFuY2gnIHwgJ2NoaWxkcmVuJyB8ICdzZWxlY3RlZCcgfCAnc3ViZmxvdycgfCAnbG9vcCc7XG5cbi8qKiBBIHNpbmdsZSBmbG93IGNvbnRyb2wgbmFycmF0aXZlIGVudHJ5LiAqL1xuZXhwb3J0IGludGVyZmFjZSBGbG93TWVzc2FnZSB7XG4gIHR5cGU6IEZsb3dDb250cm9sVHlwZTtcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgdGFyZ2V0U3RhZ2U/OiBzdHJpbmcgfCBzdHJpbmdbXTtcbiAgcmF0aW9uYWxlPzogc3RyaW5nO1xuICBjb3VudD86IG51bWJlcjtcbiAgaXRlcmF0aW9uPzogbnVtYmVyO1xuICB0aW1lc3RhbXA/OiBudW1iZXI7XG59XG5cbi8vIOKUgOKUgCBTdGFnZSBTbmFwc2hvdCDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIDilIBcblxuLyoqIFNlcmlhbGlzYWJsZSByZXByZXNlbnRhdGlvbiBvZiBhIHN0YWdlJ3Mgc3RhdGUgKGZvciBkZWJ1Z2dpbmcgLyB2aXN1YWxpc2F0aW9uKS4gKi9cbmV4cG9ydCB0eXBlIFN0YWdlU25hcHNob3QgPSB7XG4gIGlkOiBzdHJpbmc7XG4gIC8qKiBVbmlxdWUgcGVyLWV4ZWN1dGlvbi1zdGVwIGlkZW50aWZpZXIuIEZvcm1hdDogW3N1YmZsb3dQYXRoL11zdGFnZUlkI2V4ZWN1dGlvbkluZGV4ICovXG4gIHJ1bnRpbWVTdGFnZUlkPzogc3RyaW5nO1xuICBuYW1lPzogc3RyaW5nO1xuICAvKiogSHVtYW4tcmVhZGFibGUgZGVzY3JpcHRpb24gb2Ygd2hhdCB0aGlzIHN0YWdlIGRvZXMgKGZyb20gYnVpbGRlcikuICovXG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICAvKiogU3ViZmxvdyBpZGVudGlmaWVyIOKAlCBwcmVzZW50IHdoZW4gdGhpcyBzdGFnZSBpcyBhIHN1YmZsb3cgZW50cnkgcG9pbnQuICovXG4gIHN1YmZsb3dJZD86IHN0cmluZztcbiAgaXNEZWNpZGVyPzogYm9vbGVhbjtcbiAgaXNGb3JrPzogYm9vbGVhbjtcbiAgLyoqIFVzZXItbGV2ZWwgd3JpdGVzIG1hZGUgYnkgdGhpcyBzdGFnZSAocHJlLW5hbWVzcGFjZSBrZXlzIOKGkiB2YWx1ZXMpLiAqL1xuICBzdGFnZVdyaXRlcz86IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICAvKiogVXNlci1sZXZlbCByZWFkcyBtYWRlIGJ5IHRoaXMgc3RhZ2UgKHByZS1uYW1lc3BhY2Uga2V5cyDihpIgdmFsdWVzIGF0IHJlYWQgdGltZSkuICovXG4gIHN0YWdlUmVhZHM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgbG9nczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIGVycm9yczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIG1ldHJpY3M6IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBldmFsczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIGZsb3dNZXNzYWdlcz86IEZsb3dNZXNzYWdlW107XG4gIG5leHQ/OiBTdGFnZVNuYXBzaG90O1xuICBjaGlsZHJlbj86IFN0YWdlU25hcHNob3RbXTtcbn07XG5cbi8vIOKUgOKUgCBTY29wZSBGYWN0b3J5IOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4vKiogRm9yd2FyZC1kZWNsYXJlZCBzbyBTdGFnZUNvbnRleHQgY2FuIGFjY2VwdCBpdCB3aXRob3V0IGltcG9ydGluZyBzY29wZS8uICovXG5leHBvcnQgdHlwZSBTY29wZUZhY3Rvcnk8VFNjb3BlPiA9IChjb3JlOiBTdGFnZUNvbnRleHQsIHN0YWdlTmFtZTogc3RyaW5nLCByZWFkT25seUNvbnRleHQ/OiB1bmtub3duKSA9PiBUU2NvcGU7XG5cbi8vIOKUgOKUgCBTdGFnZUNvbnRleHQgKGZvcndhcmQgcmVmZXJlbmNlIGZvciBTY29wZUZhY3RvcnkpIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4vLyBUaGUgYWN0dWFsIGNsYXNzIGxpdmVzIGluIFN0YWdlQ29udGV4dC50czsgd2UganVzdCBuZWVkIHRoZSB0eXBlIGhlcmUgZm9yXG4vLyB0aGUgU2NvcGVGYWN0b3J5IGdlbmVyaWMuIFR5cGVTY3JpcHQncyBpbXBvcnQtdHlwZSBoYW5kbGVzIHRoaXM6XG5pbXBvcnQgdHlwZSB7IFN0YWdlQ29udGV4dCB9IGZyb20gJy4vU3RhZ2VDb250ZXh0LmpzJztcbiJdfQ==
@@ -2,16 +2,29 @@
2
2
  * KeyedRecorder<T> — base class for Map-based recorders keyed by runtimeStageId.
3
3
  *
4
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.
5
+ * and three standard operations on auto-collected traversal data:
6
+ *
7
+ * - **Translate** (raw): `getByKey(id)` — per-step value
8
+ * - **Accumulate** (progressive): `accumulate(fn, initial, keys?)` — running total up to a point
9
+ * - **Aggregate** (summary): `aggregate(fn, initial)` — reduce all entries
10
+ *
11
+ * Data is automatically collected during the single DFS traversal.
12
+ * The consumer chooses the operation at read time.
7
13
  *
8
14
  * @example
9
15
  * ```typescript
10
16
  * class TokenRecorder extends KeyedRecorder<LLMCallEntry> {
11
- * onLLMCall(event: LLMCallEvent) {
12
- * this.store(event.runtimeStageId, { model: event.model, ... });
13
- * }
14
- * getStats() { return aggregate(this.values()); }
17
+ * readonly id = 'tokens';
18
+ * onLLMCall(event) { this.store(event.runtimeStageId, { tokens: event.usage }); }
19
+ *
20
+ * // Translate: per-step
21
+ * getForStep(id: string) { return this.getByKey(id); }
22
+ *
23
+ * // Aggregate: total
24
+ * getTotalTokens() { return this.aggregate((sum, e) => sum + e.tokens, 0); }
25
+ *
26
+ * // Accumulate: progressive up to slider position
27
+ * getTokensUpTo(keys: Set<string>) { return this.accumulate((sum, e) => sum + e.tokens, 0, keys); }
15
28
  * }
16
29
  * ```
17
30
  */
@@ -23,6 +36,7 @@ export class KeyedRecorder {
23
36
  store(runtimeStageId, entry) {
24
37
  this.data.set(runtimeStageId, entry);
25
38
  }
39
+ // ── Translate (raw per-step) ──────────────────────────────
26
40
  /** O(1) lookup by runtimeStageId. */
27
41
  getByKey(runtimeStageId) {
28
42
  return this.data.get(runtimeStageId);
@@ -39,9 +53,44 @@ export class KeyedRecorder {
39
53
  get size() {
40
54
  return this.data.size;
41
55
  }
56
+ // ── Aggregate (reduce all entries) ────────────────────────
57
+ /** Reduce ALL entries to a single value. For dashboards, totals, summaries. */
58
+ aggregate(fn, initial) {
59
+ let acc = initial;
60
+ for (const [key, entry] of this.data) {
61
+ acc = fn(acc, entry, key);
62
+ }
63
+ return acc;
64
+ }
65
+ // ── Accumulate (progressive reduce) ───────────────────────
66
+ /**
67
+ * Reduce entries, optionally filtered by a set of keys.
68
+ * For time-travel progressive view: pass the runtimeStageIds visible at the current slider position.
69
+ * Without keys, reduces all entries (same as aggregate).
70
+ */
71
+ accumulate(fn, initial, keys) {
72
+ let acc = initial;
73
+ for (const [key, entry] of this.data) {
74
+ if (keys && !keys.has(key))
75
+ continue;
76
+ acc = fn(acc, entry, key);
77
+ }
78
+ return acc;
79
+ }
80
+ // ── Filter (subset by keys) ───────────────────────────────
81
+ /** Return entries whose keys are in the set, preserving insertion order. */
82
+ filterByKeys(keys) {
83
+ const result = [];
84
+ for (const [key, entry] of this.data) {
85
+ if (keys.has(key))
86
+ result.push(entry);
87
+ }
88
+ return result;
89
+ }
90
+ // ── Lifecycle ─────────────────────────────────────────────
42
91
  /** Clear all stored data. Called by executor before each run(). */
43
92
  clear() {
44
93
  this.data.clear();
45
94
  }
46
95
  }
47
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiS2V5ZWRSZWNvcmRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvcmVjb3JkZXIvS2V5ZWRSZWNvcmRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7OztHQWdCRztBQUNILE1BQU0sT0FBZ0IsYUFBYTtJQUFuQztRQUdtQixTQUFJLEdBQUcsSUFBSSxHQUFHLEVBQWEsQ0FBQztJQStCL0MsQ0FBQztJQTdCQyw4Q0FBOEM7SUFDcEMsS0FBSyxDQUFDLGNBQXNCLEVBQUUsS0FBUTtRQUM5QyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxjQUFjLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDdkMsQ0FBQztJQUVELHFDQUFxQztJQUNyQyxRQUFRLENBQUMsY0FBc0I7UUFDN0IsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxjQUFjLENBQUMsQ0FBQztJQUN2QyxDQUFDO0lBRUQsMERBQTBEO0lBQzFELE1BQU07UUFDSixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUM7SUFDbkIsQ0FBQztJQUVELG1EQUFtRDtJQUNuRCxNQUFNO1FBQ0osT0FBTyxDQUFDLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO0lBQ2pDLENBQUM7SUFFRCxnQ0FBZ0M7SUFDaEMsSUFBSSxJQUFJO1FBQ04sT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztJQUN4QixDQUFDO0lBRUQsbUVBQW1FO0lBQ25FLEtBQUs7UUFDSCxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ3BCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogS2V5ZWRSZWNvcmRlcjxUPiDigJQgYmFzZSBjbGFzcyBmb3IgTWFwLWJhc2VkIHJlY29yZGVycyBrZXllZCBieSBydW50aW1lU3RhZ2VJZC5cbiAqXG4gKiBQcm92aWRlcyB0eXBlZCBrZXktdmFsdWUgc3RvcmFnZSB3aXRoIE8oMSkgbG9va3VwLCBpbnNlcnRpb24tb3JkZXJlZCBpdGVyYXRpb24sXG4gKiBhbmQgY29tbW9uIGFjY2Vzc29ycy4gUmVjb3JkZXIgaW1wbGVtZW50YXRpb25zIGV4dGVuZCB0aGlzIGFuZCBjYWxsIHN0b3JlKClcbiAqIGZyb20gdGhlaXIgZXZlbnQgaG9va3MuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGNsYXNzIFRva2VuUmVjb3JkZXIgZXh0ZW5kcyBLZXllZFJlY29yZGVyPExMTUNhbGxFbnRyeT4ge1xuICogICBvbkxMTUNhbGwoZXZlbnQ6IExMTUNhbGxFdmVudCkge1xuICogICAgIHRoaXMuc3RvcmUoZXZlbnQucnVudGltZVN0YWdlSWQsIHsgbW9kZWw6IGV2ZW50Lm1vZGVsLCAuLi4gfSk7XG4gKiAgIH1cbiAqICAgZ2V0U3RhdHMoKSB7IHJldHVybiBhZ2dyZWdhdGUodGhpcy52YWx1ZXMoKSk7IH1cbiAqIH1cbiAqIGBgYFxuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgS2V5ZWRSZWNvcmRlcjxUPiB7XG4gIGFic3RyYWN0IHJlYWRvbmx5IGlkOiBzdHJpbmc7XG5cbiAgcHJpdmF0ZSByZWFkb25seSBkYXRhID0gbmV3IE1hcDxzdHJpbmcsIFQ+KCk7XG5cbiAgLyoqIFN0b3JlIGFuIGVudHJ5IGtleWVkIGJ5IHJ1bnRpbWVTdGFnZUlkLiAqL1xuICBwcm90ZWN0ZWQgc3RvcmUocnVudGltZVN0YWdlSWQ6IHN0cmluZywgZW50cnk6IFQpOiB2b2lkIHtcbiAgICB0aGlzLmRhdGEuc2V0KHJ1bnRpbWVTdGFnZUlkLCBlbnRyeSk7XG4gIH1cblxuICAvKiogTygxKSBsb29rdXAgYnkgcnVudGltZVN0YWdlSWQuICovXG4gIGdldEJ5S2V5KHJ1bnRpbWVTdGFnZUlkOiBzdHJpbmcpOiBUIHwgdW5kZWZpbmVkIHtcbiAgICByZXR1cm4gdGhpcy5kYXRhLmdldChydW50aW1lU3RhZ2VJZCk7XG4gIH1cblxuICAvKiogQWxsIGVudHJpZXMgYXMgYSByZWFkLW9ubHkgTWFwIChpbnNlcnRpb24tb3JkZXJlZCkuICovXG4gIGdldE1hcCgpOiBSZWFkb25seU1hcDxzdHJpbmcsIFQ+IHtcbiAgICByZXR1cm4gdGhpcy5kYXRhO1xuICB9XG5cbiAgLyoqIEFsbCBlbnRyaWVzIGFzIGFuIGFycmF5IChpbnNlcnRpb24tb3JkZXJlZCkuICovXG4gIHZhbHVlcygpOiBUW10ge1xuICAgIHJldHVybiBbLi4udGhpcy5kYXRhLnZhbHVlcygpXTtcbiAgfVxuXG4gIC8qKiBOdW1iZXIgb2YgZW50cmllcyBzdG9yZWQuICovXG4gIGdldCBzaXplKCk6IG51bWJlciB7XG4gICAgcmV0dXJuIHRoaXMuZGF0YS5zaXplO1xuICB9XG5cbiAgLyoqIENsZWFyIGFsbCBzdG9yZWQgZGF0YS4gQ2FsbGVkIGJ5IGV4ZWN1dG9yIGJlZm9yZSBlYWNoIHJ1bigpLiAqL1xuICBjbGVhcigpOiB2b2lkIHtcbiAgICB0aGlzLmRhdGEuY2xlYXIoKTtcbiAgfVxufVxuIl19
96
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiS2V5ZWRSZWNvcmRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvcmVjb3JkZXIvS2V5ZWRSZWNvcmRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0E2Qkc7QUFDSCxNQUFNLE9BQWdCLGFBQWE7SUFBbkM7UUFHbUIsU0FBSSxHQUFHLElBQUksR0FBRyxFQUFhLENBQUM7SUF5RS9DLENBQUM7SUF2RUMsOENBQThDO0lBQ3BDLEtBQUssQ0FBQyxjQUFzQixFQUFFLEtBQVE7UUFDOUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsY0FBYyxFQUFFLEtBQUssQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFRCw2REFBNkQ7SUFFN0QscUNBQXFDO0lBQ3JDLFFBQVEsQ0FBQyxjQUFzQjtRQUM3QixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFRCwwREFBMEQ7SUFDMUQsTUFBTTtRQUNKLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQztJQUNuQixDQUFDO0lBRUQsbURBQW1EO0lBQ25ELE1BQU07UUFDSixPQUFPLENBQUMsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUM7SUFDakMsQ0FBQztJQUVELGdDQUFnQztJQUNoQyxJQUFJLElBQUk7UUFDTixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDO0lBQ3hCLENBQUM7SUFFRCw2REFBNkQ7SUFFN0QsK0VBQStFO0lBQy9FLFNBQVMsQ0FBSSxFQUF3QyxFQUFFLE9BQVU7UUFDL0QsSUFBSSxHQUFHLEdBQUcsT0FBTyxDQUFDO1FBQ2xCLEtBQUssTUFBTSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsSUFBSSxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDckMsR0FBRyxHQUFHLEVBQUUsQ0FBQyxHQUFHLEVBQUUsS0FBSyxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBQzVCLENBQUM7UUFDRCxPQUFPLEdBQUcsQ0FBQztJQUNiLENBQUM7SUFFRCw2REFBNkQ7SUFFN0Q7Ozs7T0FJRztJQUNILFVBQVUsQ0FBSSxFQUF3QyxFQUFFLE9BQVUsRUFBRSxJQUEwQjtRQUM1RixJQUFJLEdBQUcsR0FBRyxPQUFPLENBQUM7UUFDbEIsS0FBSyxNQUFNLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNyQyxJQUFJLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDO2dCQUFFLFNBQVM7WUFDckMsR0FBRyxHQUFHLEVBQUUsQ0FBQyxHQUFHLEVBQUUsS0FBSyxFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBQzVCLENBQUM7UUFDRCxPQUFPLEdBQUcsQ0FBQztJQUNiLENBQUM7SUFFRCw2REFBNkQ7SUFFN0QsNEVBQTRFO0lBQzVFLFlBQVksQ0FBQyxJQUF5QjtRQUNwQyxNQUFNLE1BQU0sR0FBUSxFQUFFLENBQUM7UUFDdkIsS0FBSyxNQUFNLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNyQyxJQUFJLElBQUksQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDO2dCQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDeEMsQ0FBQztRQUNELE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUM7SUFFRCw2REFBNkQ7SUFFN0QsbUVBQW1FO0lBQ25FLEtBQUs7UUFDSCxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ3BCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogS2V5ZWRSZWNvcmRlcjxUPiDigJQgYmFzZSBjbGFzcyBmb3IgTWFwLWJhc2VkIHJlY29yZGVycyBrZXllZCBieSBydW50aW1lU3RhZ2VJZC5cbiAqXG4gKiBQcm92aWRlcyB0eXBlZCBrZXktdmFsdWUgc3RvcmFnZSB3aXRoIE8oMSkgbG9va3VwLCBpbnNlcnRpb24tb3JkZXJlZCBpdGVyYXRpb24sXG4gKiBhbmQgdGhyZWUgc3RhbmRhcmQgb3BlcmF0aW9ucyBvbiBhdXRvLWNvbGxlY3RlZCB0cmF2ZXJzYWwgZGF0YTpcbiAqXG4gKiAgIC0gKipUcmFuc2xhdGUqKiAocmF3KTogYGdldEJ5S2V5KGlkKWAg4oCUIHBlci1zdGVwIHZhbHVlXG4gKiAgIC0gKipBY2N1bXVsYXRlKiogKHByb2dyZXNzaXZlKTogYGFjY3VtdWxhdGUoZm4sIGluaXRpYWwsIGtleXM/KWAg4oCUIHJ1bm5pbmcgdG90YWwgdXAgdG8gYSBwb2ludFxuICogICAtICoqQWdncmVnYXRlKiogKHN1bW1hcnkpOiBgYWdncmVnYXRlKGZuLCBpbml0aWFsKWAg4oCUIHJlZHVjZSBhbGwgZW50cmllc1xuICpcbiAqIERhdGEgaXMgYXV0b21hdGljYWxseSBjb2xsZWN0ZWQgZHVyaW5nIHRoZSBzaW5nbGUgREZTIHRyYXZlcnNhbC5cbiAqIFRoZSBjb25zdW1lciBjaG9vc2VzIHRoZSBvcGVyYXRpb24gYXQgcmVhZCB0aW1lLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBjbGFzcyBUb2tlblJlY29yZGVyIGV4dGVuZHMgS2V5ZWRSZWNvcmRlcjxMTE1DYWxsRW50cnk+IHtcbiAqICAgcmVhZG9ubHkgaWQgPSAndG9rZW5zJztcbiAqICAgb25MTE1DYWxsKGV2ZW50KSB7IHRoaXMuc3RvcmUoZXZlbnQucnVudGltZVN0YWdlSWQsIHsgdG9rZW5zOiBldmVudC51c2FnZSB9KTsgfVxuICpcbiAqICAgLy8gVHJhbnNsYXRlOiBwZXItc3RlcFxuICogICBnZXRGb3JTdGVwKGlkOiBzdHJpbmcpIHsgcmV0dXJuIHRoaXMuZ2V0QnlLZXkoaWQpOyB9XG4gKlxuICogICAvLyBBZ2dyZWdhdGU6IHRvdGFsXG4gKiAgIGdldFRvdGFsVG9rZW5zKCkgeyByZXR1cm4gdGhpcy5hZ2dyZWdhdGUoKHN1bSwgZSkgPT4gc3VtICsgZS50b2tlbnMsIDApOyB9XG4gKlxuICogICAvLyBBY2N1bXVsYXRlOiBwcm9ncmVzc2l2ZSB1cCB0byBzbGlkZXIgcG9zaXRpb25cbiAqICAgZ2V0VG9rZW5zVXBUbyhrZXlzOiBTZXQ8c3RyaW5nPikgeyByZXR1cm4gdGhpcy5hY2N1bXVsYXRlKChzdW0sIGUpID0+IHN1bSArIGUudG9rZW5zLCAwLCBrZXlzKTsgfVxuICogfVxuICogYGBgXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBLZXllZFJlY29yZGVyPFQ+IHtcbiAgYWJzdHJhY3QgcmVhZG9ubHkgaWQ6IHN0cmluZztcblxuICBwcml2YXRlIHJlYWRvbmx5IGRhdGEgPSBuZXcgTWFwPHN0cmluZywgVD4oKTtcblxuICAvKiogU3RvcmUgYW4gZW50cnkga2V5ZWQgYnkgcnVudGltZVN0YWdlSWQuICovXG4gIHByb3RlY3RlZCBzdG9yZShydW50aW1lU3RhZ2VJZDogc3RyaW5nLCBlbnRyeTogVCk6IHZvaWQge1xuICAgIHRoaXMuZGF0YS5zZXQocnVudGltZVN0YWdlSWQsIGVudHJ5KTtcbiAgfVxuXG4gIC8vIOKUgOKUgCBUcmFuc2xhdGUgKHJhdyBwZXItc3RlcCkg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbiAgLyoqIE8oMSkgbG9va3VwIGJ5IHJ1bnRpbWVTdGFnZUlkLiAqL1xuICBnZXRCeUtleShydW50aW1lU3RhZ2VJZDogc3RyaW5nKTogVCB8IHVuZGVmaW5lZCB7XG4gICAgcmV0dXJuIHRoaXMuZGF0YS5nZXQocnVudGltZVN0YWdlSWQpO1xuICB9XG5cbiAgLyoqIEFsbCBlbnRyaWVzIGFzIGEgcmVhZC1vbmx5IE1hcCAoaW5zZXJ0aW9uLW9yZGVyZWQpLiAqL1xuICBnZXRNYXAoKTogUmVhZG9ubHlNYXA8c3RyaW5nLCBUPiB7XG4gICAgcmV0dXJuIHRoaXMuZGF0YTtcbiAgfVxuXG4gIC8qKiBBbGwgZW50cmllcyBhcyBhbiBhcnJheSAoaW5zZXJ0aW9uLW9yZGVyZWQpLiAqL1xuICB2YWx1ZXMoKTogVFtdIHtcbiAgICByZXR1cm4gWy4uLnRoaXMuZGF0YS52YWx1ZXMoKV07XG4gIH1cblxuICAvKiogTnVtYmVyIG9mIGVudHJpZXMgc3RvcmVkLiAqL1xuICBnZXQgc2l6ZSgpOiBudW1iZXIge1xuICAgIHJldHVybiB0aGlzLmRhdGEuc2l6ZTtcbiAgfVxuXG4gIC8vIOKUgOKUgCBBZ2dyZWdhdGUgKHJlZHVjZSBhbGwgZW50cmllcykg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbiAgLyoqIFJlZHVjZSBBTEwgZW50cmllcyB0byBhIHNpbmdsZSB2YWx1ZS4gRm9yIGRhc2hib2FyZHMsIHRvdGFscywgc3VtbWFyaWVzLiAqL1xuICBhZ2dyZWdhdGU8Uj4oZm46IChhY2M6IFIsIGVudHJ5OiBULCBrZXk6IHN0cmluZykgPT4gUiwgaW5pdGlhbDogUik6IFIge1xuICAgIGxldCBhY2MgPSBpbml0aWFsO1xuICAgIGZvciAoY29uc3QgW2tleSwgZW50cnldIG9mIHRoaXMuZGF0YSkge1xuICAgICAgYWNjID0gZm4oYWNjLCBlbnRyeSwga2V5KTtcbiAgICB9XG4gICAgcmV0dXJuIGFjYztcbiAgfVxuXG4gIC8vIOKUgOKUgCBBY2N1bXVsYXRlIChwcm9ncmVzc2l2ZSByZWR1Y2UpIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4gIC8qKlxuICAgKiBSZWR1Y2UgZW50cmllcywgb3B0aW9uYWxseSBmaWx0ZXJlZCBieSBhIHNldCBvZiBrZXlzLlxuICAgKiBGb3IgdGltZS10cmF2ZWwgcHJvZ3Jlc3NpdmUgdmlldzogcGFzcyB0aGUgcnVudGltZVN0YWdlSWRzIHZpc2libGUgYXQgdGhlIGN1cnJlbnQgc2xpZGVyIHBvc2l0aW9uLlxuICAgKiBXaXRob3V0IGtleXMsIHJlZHVjZXMgYWxsIGVudHJpZXMgKHNhbWUgYXMgYWdncmVnYXRlKS5cbiAgICovXG4gIGFjY3VtdWxhdGU8Uj4oZm46IChhY2M6IFIsIGVudHJ5OiBULCBrZXk6IHN0cmluZykgPT4gUiwgaW5pdGlhbDogUiwga2V5cz86IFJlYWRvbmx5U2V0PHN0cmluZz4pOiBSIHtcbiAgICBsZXQgYWNjID0gaW5pdGlhbDtcbiAgICBmb3IgKGNvbnN0IFtrZXksIGVudHJ5XSBvZiB0aGlzLmRhdGEpIHtcbiAgICAgIGlmIChrZXlzICYmICFrZXlzLmhhcyhrZXkpKSBjb250aW51ZTtcbiAgICAgIGFjYyA9IGZuKGFjYywgZW50cnksIGtleSk7XG4gICAgfVxuICAgIHJldHVybiBhY2M7XG4gIH1cblxuICAvLyDilIDilIAgRmlsdGVyIChzdWJzZXQgYnkga2V5cykg4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSA4pSAXG5cbiAgLyoqIFJldHVybiBlbnRyaWVzIHdob3NlIGtleXMgYXJlIGluIHRoZSBzZXQsIHByZXNlcnZpbmcgaW5zZXJ0aW9uIG9yZGVyLiAqL1xuICBmaWx0ZXJCeUtleXMoa2V5czogUmVhZG9ubHlTZXQ8c3RyaW5nPik6IFRbXSB7XG4gICAgY29uc3QgcmVzdWx0OiBUW10gPSBbXTtcbiAgICBmb3IgKGNvbnN0IFtrZXksIGVudHJ5XSBvZiB0aGlzLmRhdGEpIHtcbiAgICAgIGlmIChrZXlzLmhhcyhrZXkpKSByZXN1bHQucHVzaChlbnRyeSk7XG4gICAgfVxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICAvLyDilIDilIAgTGlmZWN5Y2xlIOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgOKUgFxuXG4gIC8qKiBDbGVhciBhbGwgc3RvcmVkIGRhdGEuIENhbGxlZCBieSBleGVjdXRvciBiZWZvcmUgZWFjaCBydW4oKS4gKi9cbiAgY2xlYXIoKTogdm9pZCB7XG4gICAgdGhpcy5kYXRhLmNsZWFyKCk7XG4gIH1cbn1cbiJdfQ==
@@ -1,39 +1,32 @@
1
1
  /**
2
- * MetricRecorder — Production-focused recorder for timing and execution counts.
2
+ * MetricRecorder — per-step timing and execution counts, keyed by runtimeStageId.
3
3
  *
4
- * Tracks read/write/commit counts per stage and measures stage execution duration.
5
- *
6
- * Each instance gets a unique auto-increment ID (`metrics-1`, `metrics-2`, ...),
7
- * so multiple recorders with different configs coexist. Pass an explicit ID to
8
- * override a specific instance (e.g., a framework-attached recorder).
4
+ * Stores per-invocation data during traversal. Aggregated views computed on read.
5
+ * Extends KeyedRecorder<StepMetrics> for O(1) lookup and standard operations.
9
6
  *
10
7
  * @example
11
8
  * ```typescript
12
- * // Track all stages (default)
13
- * executor.attachRecorder(new MetricRecorder());
9
+ * const metric = new MetricRecorder();
10
+ * executor.attachRecorder(metric);
11
+ * await executor.run();
14
12
  *
15
- * // Track only LLM-related stages
16
- * executor.attachRecorder(new MetricRecorder({
17
- * stageFilter: (name) => ['CallLLM', 'ParseResponse'].includes(name),
18
- * }));
13
+ * // Per-step (time-travel):
14
+ * metric.getByKey('call-llm#5'); // { stageName, readCount, writeCount, duration }
19
15
  *
20
- * // Two recorders: one for LLM timing, one for everything else
21
- * executor.attachRecorder(new MetricRecorder({
22
- * stageFilter: (name) => name === 'CallLLM',
23
- * }));
24
- * executor.attachRecorder(new MetricRecorder({
25
- * stageFilter: (name) => name !== 'CallLLM',
26
- * }));
16
+ * // Aggregated (backward compat):
17
+ * metric.getMetrics(); // { totalDuration, totalReads, stageMetrics: Map<stageName, aggregated> }
27
18
  *
28
- * // Override a framework-attached recorder by passing its well-known ID
29
- * executor.attachRecorder(new MetricRecorder({ id: 'metrics' }));
19
+ * // Progressive (slider):
20
+ * metric.accumulate((sum, m) => sum + m.duration, 0, visibleKeys);
30
21
  * ```
31
22
  */
32
- export class MetricRecorder {
23
+ import { KeyedRecorder } from '../../recorder/KeyedRecorder.js';
24
+ export class MetricRecorder extends KeyedRecorder {
33
25
  constructor(idOrOptions) {
34
26
  var _a;
35
- this.metrics = new Map();
27
+ super();
36
28
  this.stageStartTimes = new Map();
29
+ this.currentRuntimeStageId = '';
37
30
  if (typeof idOrOptions === 'string') {
38
31
  this.id = idOrOptions;
39
32
  }
@@ -45,73 +38,96 @@ export class MetricRecorder {
45
38
  shouldRecord(stageName) {
46
39
  return !this.stageFilter || this.stageFilter(stageName);
47
40
  }
41
+ /** Get or create the StepMetrics for the current stage. */
42
+ current() {
43
+ const key = this.currentRuntimeStageId;
44
+ let m = this.getByKey(key);
45
+ if (!m) {
46
+ m = { stageName: '', readCount: 0, writeCount: 0, commitCount: 0, pauseCount: 0, duration: 0 };
47
+ this.store(key, m);
48
+ }
49
+ return m;
50
+ }
51
+ onStageStart(event) {
52
+ if (!this.shouldRecord(event.stageName))
53
+ return;
54
+ this.currentRuntimeStageId = event.runtimeStageId;
55
+ this.stageStartTimes.set(event.runtimeStageId, event.timestamp);
56
+ const m = this.current();
57
+ m.stageName = event.stageName;
58
+ }
48
59
  onRead(event) {
49
60
  if (!this.shouldRecord(event.stageName))
50
61
  return;
51
- this.getOrCreateStageMetrics(event.stageName).readCount++;
62
+ this.current().readCount++;
52
63
  }
53
64
  onWrite(event) {
54
65
  if (!this.shouldRecord(event.stageName))
55
66
  return;
56
- this.getOrCreateStageMetrics(event.stageName).writeCount++;
67
+ this.current().writeCount++;
57
68
  }
58
69
  onCommit(event) {
59
70
  if (!this.shouldRecord(event.stageName))
60
71
  return;
61
- this.getOrCreateStageMetrics(event.stageName).commitCount++;
72
+ this.current().commitCount++;
62
73
  }
63
74
  onPause(event) {
64
75
  if (!this.shouldRecord(event.stageName))
65
76
  return;
66
- this.getOrCreateStageMetrics(event.stageName).pauseCount++;
67
- }
68
- onStageStart(event) {
69
- if (!this.shouldRecord(event.stageName))
70
- return;
71
- this.stageStartTimes.set(event.stageName, event.timestamp);
72
- this.getOrCreateStageMetrics(event.stageName).invocationCount++;
77
+ this.current().pauseCount++;
73
78
  }
74
79
  onStageEnd(event) {
75
80
  if (!this.shouldRecord(event.stageName))
76
81
  return;
77
- const stageMetrics = this.getOrCreateStageMetrics(event.stageName);
78
- let duration;
82
+ const m = this.current();
79
83
  if (event.duration !== undefined) {
80
- duration = event.duration;
84
+ m.duration = event.duration;
81
85
  }
82
86
  else {
83
- const startTime = this.stageStartTimes.get(event.stageName);
84
- duration = startTime !== undefined ? event.timestamp - startTime : 0;
87
+ const startTime = this.stageStartTimes.get(event.runtimeStageId);
88
+ m.duration = startTime !== undefined ? event.timestamp - startTime : 0;
85
89
  }
86
- stageMetrics.totalDuration += duration;
87
- this.stageStartTimes.delete(event.stageName);
90
+ this.stageStartTimes.delete(event.runtimeStageId);
88
91
  }
92
+ /** Aggregated metrics — computes totals on the fly from per-step data (backward compatible). */
89
93
  getMetrics() {
90
- let totalDuration = 0;
91
- let totalReads = 0;
92
- let totalWrites = 0;
93
- let totalCommits = 0;
94
- let totalPauses = 0;
95
- for (const stageMetrics of this.metrics.values()) {
96
- totalDuration += stageMetrics.totalDuration;
97
- totalReads += stageMetrics.readCount;
98
- totalWrites += stageMetrics.writeCount;
99
- totalCommits += stageMetrics.commitCount;
100
- totalPauses += stageMetrics.pauseCount;
94
+ const byName = new Map();
95
+ const totalDuration = this.aggregate((sum, m) => sum + m.duration, 0);
96
+ const totalReads = this.aggregate((sum, m) => sum + m.readCount, 0);
97
+ const totalWrites = this.aggregate((sum, m) => sum + m.writeCount, 0);
98
+ const totalCommits = this.aggregate((sum, m) => sum + m.commitCount, 0);
99
+ const totalPauses = this.aggregate((sum, m) => sum + m.pauseCount, 0);
100
+ // Group by stageName for backward compat
101
+ for (const m of this.values()) {
102
+ const existing = byName.get(m.stageName);
103
+ if (existing) {
104
+ existing.readCount += m.readCount;
105
+ existing.writeCount += m.writeCount;
106
+ existing.commitCount += m.commitCount;
107
+ existing.pauseCount += m.pauseCount;
108
+ existing.totalDuration += m.duration;
109
+ existing.invocationCount++;
110
+ }
111
+ else {
112
+ byName.set(m.stageName, {
113
+ stageName: m.stageName,
114
+ readCount: m.readCount,
115
+ writeCount: m.writeCount,
116
+ commitCount: m.commitCount,
117
+ pauseCount: m.pauseCount,
118
+ totalDuration: m.duration,
119
+ invocationCount: 1,
120
+ });
121
+ }
101
122
  }
102
- return {
103
- totalDuration,
104
- totalReads,
105
- totalWrites,
106
- totalCommits,
107
- totalPauses,
108
- stageMetrics: new Map(this.metrics),
109
- };
123
+ return { totalDuration, totalReads, totalWrites, totalCommits, totalPauses, stageMetrics: byName };
110
124
  }
125
+ /** Get aggregated metrics for a specific stage name (backward compatible). */
111
126
  getStageMetrics(stageName) {
112
- const metrics = this.metrics.get(stageName);
113
- return metrics ? { ...metrics } : undefined;
127
+ const metrics = this.getMetrics();
128
+ return metrics.stageMetrics.get(stageName);
114
129
  }
130
+ /** Snapshot for serialization (backward compatible format). */
115
131
  toSnapshot() {
116
132
  const metrics = this.getMetrics();
117
133
  return {
@@ -125,29 +141,16 @@ export class MetricRecorder {
125
141
  },
126
142
  };
127
143
  }
128
- reset() {
129
- this.metrics.clear();
130
- this.stageStartTimes.clear();
131
- }
144
+ /** Clear all state — called by executor before each run(). */
132
145
  clear() {
133
- this.reset();
146
+ super.clear();
147
+ this.stageStartTimes.clear();
148
+ this.currentRuntimeStageId = '';
134
149
  }
135
- getOrCreateStageMetrics(stageName) {
136
- let stageMetrics = this.metrics.get(stageName);
137
- if (!stageMetrics) {
138
- stageMetrics = {
139
- stageName,
140
- readCount: 0,
141
- writeCount: 0,
142
- commitCount: 0,
143
- pauseCount: 0,
144
- totalDuration: 0,
145
- invocationCount: 0,
146
- };
147
- this.metrics.set(stageName, stageMetrics);
148
- }
149
- return stageMetrics;
150
+ /** Alias for clear() (backward compat). */
151
+ reset() {
152
+ this.clear();
150
153
  }
151
154
  }
152
155
  MetricRecorder._counter = 0;
153
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MetricRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/scope/recorders/MetricRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MetricRecorder — Production-focused recorder for timing and execution counts.\n *\n * Tracks read/write/commit counts per stage and measures stage execution duration.\n *\n * Each instance gets a unique auto-increment ID (`metrics-1`, `metrics-2`, ...),\n * so multiple recorders with different configs coexist. Pass an explicit ID to\n * override a specific instance (e.g., a framework-attached recorder).\n *\n * @example\n * ```typescript\n * // Track all stages (default)\n * executor.attachRecorder(new MetricRecorder());\n *\n * // Track only LLM-related stages\n * executor.attachRecorder(new MetricRecorder({\n *   stageFilter: (name) => ['CallLLM', 'ParseResponse'].includes(name),\n * }));\n *\n * // Two recorders: one for LLM timing, one for everything else\n * executor.attachRecorder(new MetricRecorder({\n *   stageFilter: (name) => name === 'CallLLM',\n * }));\n * executor.attachRecorder(new MetricRecorder({\n *   stageFilter: (name) => name !== 'CallLLM',\n * }));\n *\n * // Override a framework-attached recorder by passing its well-known ID\n * executor.attachRecorder(new MetricRecorder({ id: 'metrics' }));\n * ```\n */\n\nimport type { CommitEvent, PauseEvent, ReadEvent, Recorder, StageEvent, WriteEvent } from '../types.js';\n\nexport interface StageMetrics {\n  stageName: string;\n  readCount: number;\n  writeCount: number;\n  commitCount: number;\n  pauseCount: number;\n  totalDuration: number;\n  invocationCount: number;\n}\n\nexport interface AggregatedMetrics {\n  totalDuration: number;\n  totalReads: number;\n  totalWrites: number;\n  totalCommits: number;\n  totalPauses: number;\n  stageMetrics: Map<string, StageMetrics>;\n}\n\n/** Options for MetricRecorder. All fields are optional. */\nexport interface MetricRecorderOptions {\n  /** Recorder ID. Defaults to auto-increment (`metrics-1`, `metrics-2`, ...). */\n  id?: string;\n  /**\n   * Filter which stages are recorded. Return `true` to record, `false` to skip.\n   * When omitted, all stages are recorded.\n   *\n   * @example\n   * ```typescript\n   * // Only track stages that start with \"Call\"\n   * stageFilter: (name) => name.startsWith('Call')\n   * ```\n   */\n  stageFilter?: (stageName: string) => boolean;\n}\n\nexport class MetricRecorder implements Recorder {\n  private static _counter = 0;\n\n  readonly id: string;\n  private metrics: Map<string, StageMetrics> = new Map();\n  private stageStartTimes: Map<string, number> = new Map();\n  private stageFilter?: (stageName: string) => boolean;\n\n  constructor(idOrOptions?: string | MetricRecorderOptions) {\n    if (typeof idOrOptions === 'string') {\n      this.id = idOrOptions;\n    } else {\n      this.id = idOrOptions?.id ?? `metrics-${++MetricRecorder._counter}`;\n      this.stageFilter = idOrOptions?.stageFilter;\n    }\n  }\n\n  private shouldRecord(stageName: string): boolean {\n    return !this.stageFilter || this.stageFilter(stageName);\n  }\n\n  onRead(event: ReadEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.getOrCreateStageMetrics(event.stageName).readCount++;\n  }\n\n  onWrite(event: WriteEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.getOrCreateStageMetrics(event.stageName).writeCount++;\n  }\n\n  onCommit(event: CommitEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.getOrCreateStageMetrics(event.stageName).commitCount++;\n  }\n\n  onPause(event: PauseEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.getOrCreateStageMetrics(event.stageName).pauseCount++;\n  }\n\n  onStageStart(event: StageEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.stageStartTimes.set(event.stageName, event.timestamp);\n    this.getOrCreateStageMetrics(event.stageName).invocationCount++;\n  }\n\n  onStageEnd(event: StageEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    const stageMetrics = this.getOrCreateStageMetrics(event.stageName);\n    let duration: number;\n    if (event.duration !== undefined) {\n      duration = event.duration;\n    } else {\n      const startTime = this.stageStartTimes.get(event.stageName);\n      duration = startTime !== undefined ? event.timestamp - startTime : 0;\n    }\n    stageMetrics.totalDuration += duration;\n    this.stageStartTimes.delete(event.stageName);\n  }\n\n  getMetrics(): AggregatedMetrics {\n    let totalDuration = 0;\n    let totalReads = 0;\n    let totalWrites = 0;\n    let totalCommits = 0;\n    let totalPauses = 0;\n\n    for (const stageMetrics of this.metrics.values()) {\n      totalDuration += stageMetrics.totalDuration;\n      totalReads += stageMetrics.readCount;\n      totalWrites += stageMetrics.writeCount;\n      totalCommits += stageMetrics.commitCount;\n      totalPauses += stageMetrics.pauseCount;\n    }\n\n    return {\n      totalDuration,\n      totalReads,\n      totalWrites,\n      totalCommits,\n      totalPauses,\n      stageMetrics: new Map(this.metrics),\n    };\n  }\n\n  getStageMetrics(stageName: string): StageMetrics | undefined {\n    const metrics = this.metrics.get(stageName);\n    return metrics ? { ...metrics } : undefined;\n  }\n\n  toSnapshot(): { name: string; data: unknown } {\n    const metrics = this.getMetrics();\n    return {\n      name: 'Metrics',\n      data: {\n        totalDuration: metrics.totalDuration,\n        totalReads: metrics.totalReads,\n        totalWrites: metrics.totalWrites,\n        totalCommits: metrics.totalCommits,\n        stages: Object.fromEntries(metrics.stageMetrics),\n      },\n    };\n  }\n\n  reset(): void {\n    this.metrics.clear();\n    this.stageStartTimes.clear();\n  }\n\n  clear(): void {\n    this.reset();\n  }\n\n  private getOrCreateStageMetrics(stageName: string): StageMetrics {\n    let stageMetrics = this.metrics.get(stageName);\n    if (!stageMetrics) {\n      stageMetrics = {\n        stageName,\n        readCount: 0,\n        writeCount: 0,\n        commitCount: 0,\n        pauseCount: 0,\n        totalDuration: 0,\n        invocationCount: 0,\n      };\n      this.metrics.set(stageName, stageMetrics);\n    }\n    return stageMetrics;\n  }\n}\n"]}
156
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MetricRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/scope/recorders/MetricRecorder.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;GAqBG;AAEH,OAAO,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAC;AAiDhE,MAAM,OAAO,cAAe,SAAQ,aAA0B;IAQ5D,YAAY,WAA4C;;QACtD,KAAK,EAAE,CAAC;QALF,oBAAe,GAAG,IAAI,GAAG,EAAkB,CAAC;QAC5C,0BAAqB,GAAG,EAAE,CAAC;QAKjC,IAAI,OAAO,WAAW,KAAK,QAAQ,EAAE,CAAC;YACpC,IAAI,CAAC,EAAE,GAAG,WAAW,CAAC;QACxB,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,EAAE,GAAG,MAAA,WAAW,aAAX,WAAW,uBAAX,WAAW,CAAE,EAAE,mCAAI,WAAW,EAAE,cAAc,CAAC,QAAQ,EAAE,CAAC;YACpE,IAAI,CAAC,WAAW,GAAG,WAAW,aAAX,WAAW,uBAAX,WAAW,CAAE,WAAW,CAAC;QAC9C,CAAC;IACH,CAAC;IAEO,YAAY,CAAC,SAAiB;QACpC,OAAO,CAAC,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;IAC1D,CAAC;IAED,2DAA2D;IACnD,OAAO;QACb,MAAM,GAAG,GAAG,IAAI,CAAC,qBAAqB,CAAC;QACvC,IAAI,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;QAC3B,IAAI,CAAC,CAAC,EAAE,CAAC;YACP,CAAC,GAAG,EAAE,SAAS,EAAE,EAAE,EAAE,SAAS,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,WAAW,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC;YAC/F,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;QACrB,CAAC;QACD,OAAO,CAAC,CAAC;IACX,CAAC;IAED,YAAY,CAAC,KAAiB;QAC5B,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC;YAAE,OAAO;QAChD,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC,cAAc,CAAC;QAClD,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC,cAAc,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;QAChE,MAAM,CAAC,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;QACzB,CAAC,CAAC,SAAS,GAAG,KAAK,CAAC,SAAS,CAAC;IAChC,CAAC;IAED,MAAM,CAAC,KAAgB;QACrB,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC;YAAE,OAAO;QAChD,IAAI,CAAC,OAAO,EAAE,CAAC,SAAS,EAAE,CAAC;IAC7B,CAAC;IAED,OAAO,CAAC,KAAiB;QACvB,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC;YAAE,OAAO;QAChD,IAAI,CAAC,OAAO,EAAE,CAAC,UAAU,EAAE,CAAC;IAC9B,CAAC;IAED,QAAQ,CAAC,KAAkB;QACzB,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC;YAAE,OAAO;QAChD,IAAI,CAAC,OAAO,EAAE,CAAC,WAAW,EAAE,CAAC;IAC/B,CAAC;IAED,OAAO,CAAC,KAAiB;QACvB,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC;YAAE,OAAO;QAChD,IAAI,CAAC,OAAO,EAAE,CAAC,UAAU,EAAE,CAAC;IAC9B,CAAC;IAED,UAAU,CAAC,KAAiB;QAC1B,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,CAAC;YAAE,OAAO;QAChD,MAAM,CAAC,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;QACzB,IAAI,KAAK,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YACjC,CAAC,CAAC,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;QAC9B,CAAC;aAAM,CAAC;YACN,MAAM,SAAS,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;YACjE,CAAC,CAAC,QAAQ,GAAG,SAAS,KAAK,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;QACzE,CAAC;QACD,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;IACpD,CAAC;IAED,gGAAgG;IAChG,UAAU;QACR,MAAM,MAAM,GAAG,IAAI,GAAG,EAAwB,CAAC;QAE/C,MAAM,aAAa,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;QACtE,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QACpE,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC;QACtE,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;QACxE,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC;QAEtE,yCAAyC;QACzC,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;YACzC,IAAI,QAAQ,EAAE,CAAC;gBACb,QAAQ,CAAC,SAAS,IAAI,CAAC,CAAC,SAAS,CAAC;gBAClC,QAAQ,CAAC,UAAU,IAAI,CAAC,CAAC,UAAU,CAAC;gBACpC,QAAQ,CAAC,WAAW,IAAI,CAAC,CAAC,WAAW,CAAC;gBACtC,QAAQ,CAAC,UAAU,IAAI,CAAC,CAAC,UAAU,CAAC;gBACpC,QAAQ,CAAC,aAAa,IAAI,CAAC,CAAC,QAAQ,CAAC;gBACrC,QAAQ,CAAC,eAAe,EAAE,CAAC;YAC7B,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,EAAE;oBACtB,SAAS,EAAE,CAAC,CAAC,SAAS;oBACtB,SAAS,EAAE,CAAC,CAAC,SAAS;oBACtB,UAAU,EAAE,CAAC,CAAC,UAAU;oBACxB,WAAW,EAAE,CAAC,CAAC,WAAW;oBAC1B,UAAU,EAAE,CAAC,CAAC,UAAU;oBACxB,aAAa,EAAE,CAAC,CAAC,QAAQ;oBACzB,eAAe,EAAE,CAAC;iBACnB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,OAAO,EAAE,aAAa,EAAE,UAAU,EAAE,WAAW,EAAE,YAAY,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,EAAE,CAAC;IACrG,CAAC;IAED,8EAA8E;IAC9E,eAAe,CAAC,SAAiB;QAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAClC,OAAO,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAC7C,CAAC;IAED,+DAA+D;IAC/D,UAAU;QACR,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAClC,OAAO;YACL,IAAI,EAAE,SAAS;YACf,IAAI,EAAE;gBACJ,aAAa,EAAE,OAAO,CAAC,aAAa;gBACpC,UAAU,EAAE,OAAO,CAAC,UAAU;gBAC9B,WAAW,EAAE,OAAO,CAAC,WAAW;gBAChC,YAAY,EAAE,OAAO,CAAC,YAAY;gBAClC,MAAM,EAAE,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,YAAY,CAAC;aACjD;SACF,CAAC;IACJ,CAAC;IAED,8DAA8D;IACrD,KAAK;QACZ,KAAK,CAAC,KAAK,EAAE,CAAC;QACd,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;QAC7B,IAAI,CAAC,qBAAqB,GAAG,EAAE,CAAC;IAClC,CAAC;IAED,2CAA2C;IAC3C,KAAK;QACH,IAAI,CAAC,KAAK,EAAE,CAAC;IACf,CAAC;;AA3Ic,uBAAQ,GAAG,CAAC,AAAJ,CAAK","sourcesContent":["/**\n * MetricRecorder — per-step timing and execution counts, keyed by runtimeStageId.\n *\n * Stores per-invocation data during traversal. Aggregated views computed on read.\n * Extends KeyedRecorder<StepMetrics> for O(1) lookup and standard operations.\n *\n * @example\n * ```typescript\n * const metric = new MetricRecorder();\n * executor.attachRecorder(metric);\n * await executor.run();\n *\n * // Per-step (time-travel):\n * metric.getByKey('call-llm#5');  // { stageName, readCount, writeCount, duration }\n *\n * // Aggregated (backward compat):\n * metric.getMetrics();  // { totalDuration, totalReads, stageMetrics: Map<stageName, aggregated> }\n *\n * // Progressive (slider):\n * metric.accumulate((sum, m) => sum + m.duration, 0, visibleKeys);\n * ```\n */\n\nimport { KeyedRecorder } from '../../recorder/KeyedRecorder.js';\nimport type { CommitEvent, PauseEvent, ReadEvent, Recorder, StageEvent, WriteEvent } from '../types.js';\n\n/** Per-invocation metrics for a single execution step. */\nexport interface StepMetrics {\n  /** Human-readable stage name. */\n  stageName: string;\n  /** Number of scope reads during this invocation. */\n  readCount: number;\n  /** Number of scope writes during this invocation. */\n  writeCount: number;\n  /** Number of commits during this invocation. */\n  commitCount: number;\n  /** Number of pauses during this invocation. */\n  pauseCount: number;\n  /** Duration in ms for this invocation. */\n  duration: number;\n}\n\n/** Aggregated metrics across all invocations (backward compatible). */\nexport interface AggregatedMetrics {\n  totalDuration: number;\n  totalReads: number;\n  totalWrites: number;\n  totalCommits: number;\n  totalPauses: number;\n  /** Aggregated by stageName — sums across loop invocations. */\n  stageMetrics: Map<string, StageMetrics>;\n}\n\n/** Aggregated per-stageName (backward compatible with pre-runtimeStageId API). */\nexport interface StageMetrics {\n  stageName: string;\n  readCount: number;\n  writeCount: number;\n  commitCount: number;\n  pauseCount: number;\n  totalDuration: number;\n  invocationCount: number;\n}\n\n/** Options for MetricRecorder. */\nexport interface MetricRecorderOptions {\n  /** Recorder ID. Defaults to auto-increment (`metrics-1`, `metrics-2`, ...). */\n  id?: string;\n  /** Filter which stages are recorded. Return `true` to record, `false` to skip. */\n  stageFilter?: (stageName: string) => boolean;\n}\n\nexport class MetricRecorder extends KeyedRecorder<StepMetrics> implements Recorder {\n  private static _counter = 0;\n\n  readonly id: string;\n  private stageStartTimes = new Map<string, number>();\n  private currentRuntimeStageId = '';\n  private stageFilter?: (stageName: string) => boolean;\n\n  constructor(idOrOptions?: string | MetricRecorderOptions) {\n    super();\n    if (typeof idOrOptions === 'string') {\n      this.id = idOrOptions;\n    } else {\n      this.id = idOrOptions?.id ?? `metrics-${++MetricRecorder._counter}`;\n      this.stageFilter = idOrOptions?.stageFilter;\n    }\n  }\n\n  private shouldRecord(stageName: string): boolean {\n    return !this.stageFilter || this.stageFilter(stageName);\n  }\n\n  /** Get or create the StepMetrics for the current stage. */\n  private current(): StepMetrics {\n    const key = this.currentRuntimeStageId;\n    let m = this.getByKey(key);\n    if (!m) {\n      m = { stageName: '', readCount: 0, writeCount: 0, commitCount: 0, pauseCount: 0, duration: 0 };\n      this.store(key, m);\n    }\n    return m;\n  }\n\n  onStageStart(event: StageEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.currentRuntimeStageId = event.runtimeStageId;\n    this.stageStartTimes.set(event.runtimeStageId, event.timestamp);\n    const m = this.current();\n    m.stageName = event.stageName;\n  }\n\n  onRead(event: ReadEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.current().readCount++;\n  }\n\n  onWrite(event: WriteEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.current().writeCount++;\n  }\n\n  onCommit(event: CommitEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.current().commitCount++;\n  }\n\n  onPause(event: PauseEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    this.current().pauseCount++;\n  }\n\n  onStageEnd(event: StageEvent): void {\n    if (!this.shouldRecord(event.stageName)) return;\n    const m = this.current();\n    if (event.duration !== undefined) {\n      m.duration = event.duration;\n    } else {\n      const startTime = this.stageStartTimes.get(event.runtimeStageId);\n      m.duration = startTime !== undefined ? event.timestamp - startTime : 0;\n    }\n    this.stageStartTimes.delete(event.runtimeStageId);\n  }\n\n  /** Aggregated metrics — computes totals on the fly from per-step data (backward compatible). */\n  getMetrics(): AggregatedMetrics {\n    const byName = new Map<string, StageMetrics>();\n\n    const totalDuration = this.aggregate((sum, m) => sum + m.duration, 0);\n    const totalReads = this.aggregate((sum, m) => sum + m.readCount, 0);\n    const totalWrites = this.aggregate((sum, m) => sum + m.writeCount, 0);\n    const totalCommits = this.aggregate((sum, m) => sum + m.commitCount, 0);\n    const totalPauses = this.aggregate((sum, m) => sum + m.pauseCount, 0);\n\n    // Group by stageName for backward compat\n    for (const m of this.values()) {\n      const existing = byName.get(m.stageName);\n      if (existing) {\n        existing.readCount += m.readCount;\n        existing.writeCount += m.writeCount;\n        existing.commitCount += m.commitCount;\n        existing.pauseCount += m.pauseCount;\n        existing.totalDuration += m.duration;\n        existing.invocationCount++;\n      } else {\n        byName.set(m.stageName, {\n          stageName: m.stageName,\n          readCount: m.readCount,\n          writeCount: m.writeCount,\n          commitCount: m.commitCount,\n          pauseCount: m.pauseCount,\n          totalDuration: m.duration,\n          invocationCount: 1,\n        });\n      }\n    }\n\n    return { totalDuration, totalReads, totalWrites, totalCommits, totalPauses, stageMetrics: byName };\n  }\n\n  /** Get aggregated metrics for a specific stage name (backward compatible). */\n  getStageMetrics(stageName: string): StageMetrics | undefined {\n    const metrics = this.getMetrics();\n    return metrics.stageMetrics.get(stageName);\n  }\n\n  /** Snapshot for serialization (backward compatible format). */\n  toSnapshot(): { name: string; data: unknown } {\n    const metrics = this.getMetrics();\n    return {\n      name: 'Metrics',\n      data: {\n        totalDuration: metrics.totalDuration,\n        totalReads: metrics.totalReads,\n        totalWrites: metrics.totalWrites,\n        totalCommits: metrics.totalCommits,\n        stages: Object.fromEntries(metrics.stageMetrics),\n      },\n    };\n  }\n\n  /** Clear all state — called by executor before each run(). */\n  override clear(): void {\n    super.clear();\n    this.stageStartTimes.clear();\n    this.currentRuntimeStageId = '';\n  }\n\n  /** Alias for clear() (backward compat). */\n  reset(): void {\n    this.clear();\n  }\n}\n"]}
@@ -1,3 +1,3 @@
1
1
  export { DebugRecorder } from './DebugRecorder.js';
2
2
  export { MetricRecorder } from './MetricRecorder.js';
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Njb3BlL3JlY29yZGVycy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFXQSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFbkQsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyogaXN0YW5idWwgaWdub3JlIGZpbGUgKi9cbmV4cG9ydCB0eXBlIHtcbiAgQ29tbWl0RXZlbnQsXG4gIEVycm9yRXZlbnQsXG4gIFJlYWRFdmVudCxcbiAgUmVjb3JkZXIsXG4gIFJlY29yZGVyQ29udGV4dCxcbiAgU3RhZ2VFdmVudCxcbiAgV3JpdGVFdmVudCxcbn0gZnJvbSAnLi4vdHlwZXMuanMnO1xuZXhwb3J0IHR5cGUgeyBEZWJ1Z0VudHJ5LCBEZWJ1Z1JlY29yZGVyT3B0aW9ucywgRGVidWdWZXJib3NpdHkgfSBmcm9tICcuL0RlYnVnUmVjb3JkZXIuanMnO1xuZXhwb3J0IHsgRGVidWdSZWNvcmRlciB9IGZyb20gJy4vRGVidWdSZWNvcmRlci5qcyc7XG5leHBvcnQgdHlwZSB7IEFnZ3JlZ2F0ZWRNZXRyaWNzLCBTdGFnZU1ldHJpY3MgfSBmcm9tICcuL01ldHJpY1JlY29yZGVyLmpzJztcbmV4cG9ydCB7IE1ldHJpY1JlY29yZGVyIH0gZnJvbSAnLi9NZXRyaWNSZWNvcmRlci5qcyc7XG4iXX0=
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Njb3BlL3JlY29yZGVycy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFXQSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFbkQsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyogaXN0YW5idWwgaWdub3JlIGZpbGUgKi9cbmV4cG9ydCB0eXBlIHtcbiAgQ29tbWl0RXZlbnQsXG4gIEVycm9yRXZlbnQsXG4gIFJlYWRFdmVudCxcbiAgUmVjb3JkZXIsXG4gIFJlY29yZGVyQ29udGV4dCxcbiAgU3RhZ2VFdmVudCxcbiAgV3JpdGVFdmVudCxcbn0gZnJvbSAnLi4vdHlwZXMuanMnO1xuZXhwb3J0IHR5cGUgeyBEZWJ1Z0VudHJ5LCBEZWJ1Z1JlY29yZGVyT3B0aW9ucywgRGVidWdWZXJib3NpdHkgfSBmcm9tICcuL0RlYnVnUmVjb3JkZXIuanMnO1xuZXhwb3J0IHsgRGVidWdSZWNvcmRlciB9IGZyb20gJy4vRGVidWdSZWNvcmRlci5qcyc7XG5leHBvcnQgdHlwZSB7IEFnZ3JlZ2F0ZWRNZXRyaWNzLCBTdGFnZU1ldHJpY3MsIFN0ZXBNZXRyaWNzIH0gZnJvbSAnLi9NZXRyaWNSZWNvcmRlci5qcyc7XG5leHBvcnQgeyBNZXRyaWNSZWNvcmRlciB9IGZyb20gJy4vTWV0cmljUmVjb3JkZXIuanMnO1xuIl19