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