footprintjs 4.5.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/engine/narrative/types.js +1 -1
- package/dist/esm/lib/engine/runtimeStageId.js +57 -0
- package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +12 -3
- package/dist/esm/lib/memory/StageContext.js +9 -2
- package/dist/esm/lib/memory/commitLogUtils.js +25 -0
- package/dist/esm/lib/memory/types.js +1 -1
- package/dist/esm/lib/recorder/KeyedRecorder.js +47 -0
- package/dist/esm/lib/recorder/index.js +2 -1
- package/dist/esm/lib/runner/FlowChartExecutor.js +7 -1
- package/dist/esm/lib/scope/ScopeFacade.js +11 -1
- package/dist/esm/lib/scope/types.js +1 -1
- package/dist/esm/trace.js +25 -0
- package/dist/lib/engine/narrative/types.js +1 -1
- package/dist/lib/engine/runtimeStageId.js +63 -0
- package/dist/lib/engine/traversal/FlowchartTraverser.js +12 -3
- package/dist/lib/memory/StageContext.js +9 -2
- package/dist/lib/memory/commitLogUtils.js +31 -0
- package/dist/lib/memory/types.js +1 -1
- package/dist/lib/recorder/KeyedRecorder.js +51 -0
- package/dist/lib/recorder/index.js +4 -2
- package/dist/lib/runner/FlowChartExecutor.js +7 -1
- package/dist/lib/scope/ScopeFacade.js +11 -1
- package/dist/lib/scope/types.js +1 -1
- package/dist/trace.js +35 -0
- package/dist/types/advanced.d.ts +4 -0
- package/dist/types/lib/engine/narrative/types.d.ts +2 -0
- package/dist/types/lib/engine/runtimeStageId.d.ts +47 -0
- package/dist/types/lib/engine/traversal/FlowchartTraverser.d.ts +9 -0
- package/dist/types/lib/memory/StageContext.d.ts +2 -0
- package/dist/types/lib/memory/commitLogUtils.d.ts +13 -0
- package/dist/types/lib/memory/types.d.ts +2 -0
- package/dist/types/lib/recorder/KeyedRecorder.d.ts +33 -0
- package/dist/types/lib/recorder/index.d.ts +1 -0
- package/dist/types/lib/runner/FlowChartExecutor.d.ts +2 -0
- package/dist/types/lib/scope/types.d.ts +2 -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"]}
|
|
@@ -7,4 +7,4 @@
|
|
|
7
7
|
* @module
|
|
8
8
|
*/
|
|
9
9
|
export {};
|
|
10
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/types.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG","sourcesContent":["/**\n * IControlFlowNarrative — Interface for control flow narrative generation.\n *\n * Captures FLOW events during traversal: decisions, forks, loops, subflows.\n * Complementary to scope/recorders/NarrativeRecorder which captures DATA events.\n *\n * @module\n */\n\nimport type { DecisionEvidence, SelectionEvidence } from '../../decide/types.js';\nimport type { StructuredErrorInfo } from '../errors/errorInfo.js';\n\n/**\n *\n * Uses Null Object pattern: NullControlFlowNarrativeGenerator satisfies this\n * interface with empty methods for zero-cost disabled path.\n */\n\nexport interface IControlFlowNarrative {\n  /** Called when a stage executes. First stage uses distinct opening pattern. */\n  onStageExecuted(stageName: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called on linear continuation from one stage to the next. */\n  onNext(fromStage: string, toStage: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a decider selects a branch. Most valuable for LLM context. */\n  onDecision(\n    deciderName: string,\n    chosenBranch: string,\n    rationale?: string,\n    deciderDescription?: string,\n    traversalContext?: TraversalContext,\n    evidence?: DecisionEvidence,\n  ): void;\n\n  /** Called when a fork executes all children in parallel. */\n  onFork(parentStage: string, childNames: string[], traversalContext?: TraversalContext): void;\n\n  /** Called when a selector picks a subset of children. */\n  onSelected(\n    parentStage: string,\n    selectedNames: string[],\n    totalCount: number,\n    traversalContext?: TraversalContext,\n    evidence?: SelectionEvidence,\n  ): void;\n\n  /** Called when entering a subflow (nested context boundary). */\n  onSubflowEntry(\n    subflowName: string,\n    subflowId?: string,\n    description?: string,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when exiting a subflow. */\n  onSubflowExit(subflowName: string, subflowId?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered(subflowId: string, name: string, description?: string, specStructure?: unknown): void;\n\n  /** Called on loop iteration (back-edge traversal). */\n  onLoop(targetStage: string, iteration: number, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage triggers break (early termination). */\n  onBreak(stageName: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage throws an error. Raw error is extracted into structured details. */\n  onError(stageName: string, errorMessage: string, error: unknown, traversalContext?: TraversalContext): void;\n\n  /** Called when a pausable stage pauses execution. */\n  onPause(\n    stageName: string,\n    stageId: string,\n    pauseData: unknown,\n    subflowPath: readonly string[],\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when a paused stage is resumed. */\n  onResume(stageName: string, stageId: string, hasInput: boolean, traversalContext?: TraversalContext): void;\n\n  /** Returns accumulated narrative sentences in execution order. */\n  getSentences(): string[];\n}\n\n// ============================================================================\n// TraversalContext — read-only execution context from the traverser.\n// ============================================================================\n\n/**\n * Traversal context attached to every FlowRecorder event.\n * Created by the traverser during DFS, passed to recorders as read-only data.\n * Enables recorders to build trees, group by subflow, and correlate events\n * without maintaining their own stacks or post-processing.\n *\n * Like OpenTelemetry's span context: stageId + parentStageId form a tree.\n */\nexport interface TraversalContext {\n  /** Stable stage identifier from the builder (matches spec node id). */\n  readonly stageId: string;\n  /** Human-readable stage name. */\n  readonly stageName: string;\n  /** Parent stage ID — walk up to reconstruct the tree. Undefined at root. */\n  readonly parentStageId?: string;\n  /** Subflow ID when inside a subflow. Undefined at root level. */\n  readonly subflowId?: string;\n  /** Full subflow path for nested subflows (e.g., \"sf-outer/sf-inner\"). */\n  readonly subflowPath?: string;\n  /** Nesting depth (0 = root, 1 = inside first subflow, etc.). */\n  readonly depth: number;\n  /** Loop iteration number when revisiting a node via loopTo. */\n  readonly loopIteration?: number;\n  /** Fork branch ID when inside a parallel or decider branch. */\n  readonly forkBranch?: string;\n}\n\n// ============================================================================\n// FlowRecorder — Pluggable observer for control flow events.\n// ============================================================================\n\n/** Event passed to FlowRecorder.onStageExecuted. */\nexport interface FlowStageEvent {\n  stageName: string;\n  description?: string;\n  /** Traversal context from the engine — read-only, set by traverser. */\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onNext. */\nexport interface FlowNextEvent {\n  from: string;\n  to: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onDecision. */\nexport interface FlowDecisionEvent {\n  decider: string;\n  chosen: string;\n  rationale?: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n  /** Structured decision evidence from decide() helper. */\n  evidence?: DecisionEvidence;\n}\n\n/** Event passed to FlowRecorder.onFork. */\nexport interface FlowForkEvent {\n  parent: string;\n  children: string[];\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSelected. */\nexport interface FlowSelectedEvent {\n  parent: string;\n  selected: string[];\n  total: number;\n  traversalContext?: TraversalContext;\n  /** Structured selection evidence from select() helper. */\n  evidence?: SelectionEvidence;\n}\n\n/** Event passed to FlowRecorder.onSubflow. */\nexport interface FlowSubflowEvent {\n  name: string;\n  /** Subflow identifier — use this to look up the full spec via the manifest. */\n  subflowId?: string;\n  /** Build-time description of what this subflow does. */\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSubflowRegistered (dynamic subflow attachment). */\nexport interface FlowSubflowRegisteredEvent {\n  /** Subflow identifier. */\n  subflowId: string;\n  /** Human-readable name. */\n  name: string;\n  /** Build-time description. */\n  description?: string;\n  /** Full spec structure (when available from buildTimeStructure). */\n  specStructure?: unknown;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onLoop. */\nexport interface FlowLoopEvent {\n  target: string;\n  iteration: number;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onBreak. */\nexport interface FlowBreakEvent {\n  stageName: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onError. */\nexport interface FlowErrorEvent {\n  stageName: string;\n  message: string;\n  /** Structured error details — preserves field-level issues, error codes, etc. */\n  structuredError: StructuredErrorInfo;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onPause. */\nexport interface FlowPauseEvent {\n  stageName: string;\n  stageId: string;\n  /** Data from the pause signal (question, reason, metadata). */\n  pauseData?: unknown;\n  /** Path through subflows to the paused stage. Empty at root level. */\n  subflowPath: readonly string[];\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onResume. */\nexport interface FlowResumeEvent {\n  stageName: string;\n  stageId: string;\n  /** Whether resume input was provided. */\n  hasInput: boolean;\n  traversalContext?: TraversalContext;\n}\n\n/**\n * FlowRecorder — Pluggable observer for control flow events.\n *\n * Mirrors the scope-level Recorder pattern for the engine layer.\n * All methods are optional — implement only the hooks you need.\n * Recorders are invoked synchronously in attachment order.\n * If a recorder throws, the error is caught and swallowed; execution continues.\n *\n * @example\n * ```typescript\n * const metricsRecorder: FlowRecorder = {\n *   id: 'metrics',\n *   onLoop: (event) => recordMetric('loop.iteration', event.iteration),\n *   onDecision: (event) => recordMetric('decision', event.chosen),\n * };\n * executor.attachFlowRecorder(metricsRecorder);\n * ```\n */\nexport interface FlowRecorder {\n  readonly id: string;\n  onStageExecuted?(event: FlowStageEvent): void;\n  onNext?(event: FlowNextEvent): void;\n  onDecision?(event: FlowDecisionEvent): void;\n  onFork?(event: FlowForkEvent): void;\n  onSelected?(event: FlowSelectedEvent): void;\n  onSubflowEntry?(event: FlowSubflowEvent): void;\n  onSubflowExit?(event: FlowSubflowEvent): void;\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered?(event: FlowSubflowRegisteredEvent): void;\n  onLoop?(event: FlowLoopEvent): void;\n  onBreak?(event: FlowBreakEvent): void;\n  onError?(event: FlowErrorEvent): void;\n  onPause?(event: FlowPauseEvent): void;\n  onResume?(event: FlowResumeEvent): void;\n  /** Called before each run to reset per-run state. Implement for stateful recorders. */\n  clear?(): void;\n  /** Optional: expose collected data for inclusion in snapshots. */\n  toSnapshot?(): { name: string; data: unknown };\n}\n"]}
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/types.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG","sourcesContent":["/**\n * IControlFlowNarrative — Interface for control flow narrative generation.\n *\n * Captures FLOW events during traversal: decisions, forks, loops, subflows.\n * Complementary to scope/recorders/NarrativeRecorder which captures DATA events.\n *\n * @module\n */\n\nimport type { DecisionEvidence, SelectionEvidence } from '../../decide/types.js';\nimport type { StructuredErrorInfo } from '../errors/errorInfo.js';\n\n/**\n *\n * Uses Null Object pattern: NullControlFlowNarrativeGenerator satisfies this\n * interface with empty methods for zero-cost disabled path.\n */\n\nexport interface IControlFlowNarrative {\n  /** Called when a stage executes. First stage uses distinct opening pattern. */\n  onStageExecuted(stageName: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called on linear continuation from one stage to the next. */\n  onNext(fromStage: string, toStage: string, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a decider selects a branch. Most valuable for LLM context. */\n  onDecision(\n    deciderName: string,\n    chosenBranch: string,\n    rationale?: string,\n    deciderDescription?: string,\n    traversalContext?: TraversalContext,\n    evidence?: DecisionEvidence,\n  ): void;\n\n  /** Called when a fork executes all children in parallel. */\n  onFork(parentStage: string, childNames: string[], traversalContext?: TraversalContext): void;\n\n  /** Called when a selector picks a subset of children. */\n  onSelected(\n    parentStage: string,\n    selectedNames: string[],\n    totalCount: number,\n    traversalContext?: TraversalContext,\n    evidence?: SelectionEvidence,\n  ): void;\n\n  /** Called when entering a subflow (nested context boundary). */\n  onSubflowEntry(\n    subflowName: string,\n    subflowId?: string,\n    description?: string,\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when exiting a subflow. */\n  onSubflowExit(subflowName: string, subflowId?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered(subflowId: string, name: string, description?: string, specStructure?: unknown): void;\n\n  /** Called on loop iteration (back-edge traversal). */\n  onLoop(targetStage: string, iteration: number, description?: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage triggers break (early termination). */\n  onBreak(stageName: string, traversalContext?: TraversalContext): void;\n\n  /** Called when a stage throws an error. Raw error is extracted into structured details. */\n  onError(stageName: string, errorMessage: string, error: unknown, traversalContext?: TraversalContext): void;\n\n  /** Called when a pausable stage pauses execution. */\n  onPause(\n    stageName: string,\n    stageId: string,\n    pauseData: unknown,\n    subflowPath: readonly string[],\n    traversalContext?: TraversalContext,\n  ): void;\n\n  /** Called when a paused stage is resumed. */\n  onResume(stageName: string, stageId: string, hasInput: boolean, traversalContext?: TraversalContext): void;\n\n  /** Returns accumulated narrative sentences in execution order. */\n  getSentences(): string[];\n}\n\n// ============================================================================\n// TraversalContext — read-only execution context from the traverser.\n// ============================================================================\n\n/**\n * Traversal context attached to every FlowRecorder event.\n * Created by the traverser during DFS, passed to recorders as read-only data.\n * Enables recorders to build trees, group by subflow, and correlate events\n * without maintaining their own stacks or post-processing.\n *\n * Like OpenTelemetry's span context: stageId + parentStageId form a tree.\n */\nexport interface TraversalContext {\n  /** Stable stage identifier from the builder (matches spec node id). */\n  readonly stageId: string;\n  /** Unique per-execution-step identifier. Format: [subflowPath/]stageId#executionIndex */\n  readonly runtimeStageId: string;\n  /** Human-readable stage name. */\n  readonly stageName: string;\n  /** Parent stage ID — walk up to reconstruct the tree. Undefined at root. */\n  readonly parentStageId?: string;\n  /** Subflow ID when inside a subflow. Undefined at root level. */\n  readonly subflowId?: string;\n  /** Full subflow path for nested subflows (e.g., \"sf-outer/sf-inner\"). */\n  readonly subflowPath?: string;\n  /** Nesting depth (0 = root, 1 = inside first subflow, etc.). */\n  readonly depth: number;\n  /** Loop iteration number when revisiting a node via loopTo. */\n  readonly loopIteration?: number;\n  /** Fork branch ID when inside a parallel or decider branch. */\n  readonly forkBranch?: string;\n}\n\n// ============================================================================\n// FlowRecorder — Pluggable observer for control flow events.\n// ============================================================================\n\n/** Event passed to FlowRecorder.onStageExecuted. */\nexport interface FlowStageEvent {\n  stageName: string;\n  description?: string;\n  /** Traversal context from the engine — read-only, set by traverser. */\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onNext. */\nexport interface FlowNextEvent {\n  from: string;\n  to: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onDecision. */\nexport interface FlowDecisionEvent {\n  decider: string;\n  chosen: string;\n  rationale?: string;\n  description?: string;\n  traversalContext?: TraversalContext;\n  /** Structured decision evidence from decide() helper. */\n  evidence?: DecisionEvidence;\n}\n\n/** Event passed to FlowRecorder.onFork. */\nexport interface FlowForkEvent {\n  parent: string;\n  children: string[];\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSelected. */\nexport interface FlowSelectedEvent {\n  parent: string;\n  selected: string[];\n  total: number;\n  traversalContext?: TraversalContext;\n  /** Structured selection evidence from select() helper. */\n  evidence?: SelectionEvidence;\n}\n\n/** Event passed to FlowRecorder.onSubflow. */\nexport interface FlowSubflowEvent {\n  name: string;\n  /** Subflow identifier — use this to look up the full spec via the manifest. */\n  subflowId?: string;\n  /** Build-time description of what this subflow does. */\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onSubflowRegistered (dynamic subflow attachment). */\nexport interface FlowSubflowRegisteredEvent {\n  /** Subflow identifier. */\n  subflowId: string;\n  /** Human-readable name. */\n  name: string;\n  /** Build-time description. */\n  description?: string;\n  /** Full spec structure (when available from buildTimeStructure). */\n  specStructure?: unknown;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onLoop. */\nexport interface FlowLoopEvent {\n  target: string;\n  iteration: number;\n  description?: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onBreak. */\nexport interface FlowBreakEvent {\n  stageName: string;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onError. */\nexport interface FlowErrorEvent {\n  stageName: string;\n  message: string;\n  /** Structured error details — preserves field-level issues, error codes, etc. */\n  structuredError: StructuredErrorInfo;\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onPause. */\nexport interface FlowPauseEvent {\n  stageName: string;\n  stageId: string;\n  /** Data from the pause signal (question, reason, metadata). */\n  pauseData?: unknown;\n  /** Path through subflows to the paused stage. Empty at root level. */\n  subflowPath: readonly string[];\n  traversalContext?: TraversalContext;\n}\n\n/** Event passed to FlowRecorder.onResume. */\nexport interface FlowResumeEvent {\n  stageName: string;\n  stageId: string;\n  /** Whether resume input was provided. */\n  hasInput: boolean;\n  traversalContext?: TraversalContext;\n}\n\n/**\n * FlowRecorder — Pluggable observer for control flow events.\n *\n * Mirrors the scope-level Recorder pattern for the engine layer.\n * All methods are optional — implement only the hooks you need.\n * Recorders are invoked synchronously in attachment order.\n * If a recorder throws, the error is caught and swallowed; execution continues.\n *\n * @example\n * ```typescript\n * const metricsRecorder: FlowRecorder = {\n *   id: 'metrics',\n *   onLoop: (event) => recordMetric('loop.iteration', event.iteration),\n *   onDecision: (event) => recordMetric('decision', event.chosen),\n * };\n * executor.attachFlowRecorder(metricsRecorder);\n * ```\n */\nexport interface FlowRecorder {\n  readonly id: string;\n  onStageExecuted?(event: FlowStageEvent): void;\n  onNext?(event: FlowNextEvent): void;\n  onDecision?(event: FlowDecisionEvent): void;\n  onFork?(event: FlowForkEvent): void;\n  onSelected?(event: FlowSelectedEvent): void;\n  onSubflowEntry?(event: FlowSubflowEvent): void;\n  onSubflowExit?(event: FlowSubflowEvent): void;\n  /** Called when a dynamic subflow is registered during traversal. */\n  onSubflowRegistered?(event: FlowSubflowRegisteredEvent): void;\n  onLoop?(event: FlowLoopEvent): void;\n  onBreak?(event: FlowBreakEvent): void;\n  onError?(event: FlowErrorEvent): void;\n  onPause?(event: FlowPauseEvent): void;\n  onResume?(event: FlowResumeEvent): void;\n  /** Called before each run to reset per-run state. Implement for stateful recorders. */\n  clear?(): void;\n  /** Optional: expose collected data for inclusion in snapshots. */\n  toSnapshot?(): { name: string; data: unknown };\n}\n"]}
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* runtimeStageId — unique identifier for each execution step during traversal.
|
|
3
|
+
*
|
|
4
|
+
* Format: [subflowPath/]stageId#executionIndex
|
|
5
|
+
*
|
|
6
|
+
* Components:
|
|
7
|
+
* stageId — stable node ID from the builder ('call-llm', 'seed')
|
|
8
|
+
* executionIndex — monotonic counter incremented per stage execution (0, 1, 2...)
|
|
9
|
+
* subflowPath — optional path for subflow stages ('sf-tools', 'sf-outer/sf-inner')
|
|
10
|
+
*
|
|
11
|
+
* Properties:
|
|
12
|
+
* - Unique within a run (executionIndex never repeats)
|
|
13
|
+
* - Execution-ordered (sort by executionIndex = execution order)
|
|
14
|
+
* - Human-readable ('sf-tools/execute-tool-calls#8')
|
|
15
|
+
* - Parseable (split on '#' for stageId and index, split stageId on '/' for subflow path)
|
|
16
|
+
*
|
|
17
|
+
* @example
|
|
18
|
+
* ```
|
|
19
|
+
* buildRuntimeStageId('call-llm', 5) // 'call-llm#5'
|
|
20
|
+
* buildRuntimeStageId('execute-tool-calls', 8, 'sf-tools') // 'sf-tools/execute-tool-calls#8'
|
|
21
|
+
* buildRuntimeStageId('validate', 3, 'sf-outer/sf-inner') // 'sf-outer/sf-inner/validate#3'
|
|
22
|
+
* ```
|
|
23
|
+
*/
|
|
24
|
+
/**
|
|
25
|
+
* Build a runtimeStageId from its components.
|
|
26
|
+
*
|
|
27
|
+
* Note: The traverser does NOT use the subflowPath parameter — node.id already
|
|
28
|
+
* includes the subflow prefix from the builder. This parameter exists for external
|
|
29
|
+
* consumers constructing IDs from parsed components (round-trip via parseRuntimeStageId).
|
|
30
|
+
*/
|
|
31
|
+
export function buildRuntimeStageId(stageId, executionIndex, subflowPath) {
|
|
32
|
+
const prefix = subflowPath ? `${subflowPath}/` : '';
|
|
33
|
+
return `${prefix}${stageId}#${executionIndex}`;
|
|
34
|
+
}
|
|
35
|
+
/** Parse a runtimeStageId into its components. */
|
|
36
|
+
export function parseRuntimeStageId(runtimeStageId) {
|
|
37
|
+
const hashIdx = runtimeStageId.lastIndexOf('#');
|
|
38
|
+
if (hashIdx === -1) {
|
|
39
|
+
return { stageId: runtimeStageId, executionIndex: 0, subflowPath: undefined };
|
|
40
|
+
}
|
|
41
|
+
const beforeHash = runtimeStageId.slice(0, hashIdx);
|
|
42
|
+
const executionIndex = parseInt(runtimeStageId.slice(hashIdx + 1), 10);
|
|
43
|
+
const lastSlash = beforeHash.lastIndexOf('/');
|
|
44
|
+
if (lastSlash === -1) {
|
|
45
|
+
return { stageId: beforeHash, executionIndex, subflowPath: undefined };
|
|
46
|
+
}
|
|
47
|
+
return {
|
|
48
|
+
stageId: beforeHash.slice(lastSlash + 1),
|
|
49
|
+
executionIndex,
|
|
50
|
+
subflowPath: beforeHash.slice(0, lastSlash),
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
/** Create a new execution counter starting at 0. */
|
|
54
|
+
export function createExecutionCounter() {
|
|
55
|
+
return { value: 0 };
|
|
56
|
+
}
|
|
57
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVudGltZVN0YWdlSWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL2VuZ2luZS9ydW50aW1lU3RhZ2VJZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXNCRztBQUVIOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSxtQkFBbUIsQ0FBQyxPQUFlLEVBQUUsY0FBc0IsRUFBRSxXQUFvQjtJQUMvRixNQUFNLE1BQU0sR0FBRyxXQUFXLENBQUMsQ0FBQyxDQUFDLEdBQUcsV0FBVyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztJQUNwRCxPQUFPLEdBQUcsTUFBTSxHQUFHLE9BQU8sSUFBSSxjQUFjLEVBQUUsQ0FBQztBQUNqRCxDQUFDO0FBRUQsa0RBQWtEO0FBQ2xELE1BQU0sVUFBVSxtQkFBbUIsQ0FBQyxjQUFzQjtJQUt4RCxNQUFNLE9BQU8sR0FBRyxjQUFjLENBQUMsV0FBVyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ2hELElBQUksT0FBTyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFDbkIsT0FBTyxFQUFFLE9BQU8sRUFBRSxjQUFjLEVBQUUsY0FBYyxFQUFFLENBQUMsRUFBRSxXQUFXLEVBQUUsU0FBUyxFQUFFLENBQUM7SUFDaEYsQ0FBQztJQUVELE1BQU0sVUFBVSxHQUFHLGNBQWMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0lBQ3BELE1BQU0sY0FBYyxHQUFHLFFBQVEsQ0FBQyxjQUFjLENBQUMsS0FBSyxDQUFDLE9BQU8sR0FBRyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztJQUV2RSxNQUFNLFNBQVMsR0FBRyxVQUFVLENBQUMsV0FBVyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQzlDLElBQUksU0FBUyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFDckIsT0FBTyxFQUFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsY0FBYyxFQUFFLFdBQVcsRUFBRSxTQUFTLEVBQUUsQ0FBQztJQUN6RSxDQUFDO0lBRUQsT0FBTztRQUNMLE9BQU8sRUFBRSxVQUFVLENBQUMsS0FBSyxDQUFDLFNBQVMsR0FBRyxDQUFDLENBQUM7UUFDeEMsY0FBYztRQUNkLFdBQVcsRUFBRSxVQUFVLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxTQUFTLENBQUM7S0FDNUMsQ0FBQztBQUNKLENBQUM7QUFXRCxvREFBb0Q7QUFDcEQsTUFBTSxVQUFVLHNCQUFzQjtJQUNwQyxPQUFPLEVBQUUsS0FBSyxFQUFFLENBQUMsRUFBRSxDQUFDO0FBQ3RCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIHJ1bnRpbWVTdGFnZUlkIOKAlCB1bmlxdWUgaWRlbnRpZmllciBmb3IgZWFjaCBleGVjdXRpb24gc3RlcCBkdXJpbmcgdHJhdmVyc2FsLlxuICpcbiAqIEZvcm1hdDogW3N1YmZsb3dQYXRoL11zdGFnZUlkI2V4ZWN1dGlvbkluZGV4XG4gKlxuICogQ29tcG9uZW50czpcbiAqICAgc3RhZ2VJZCAgICAgICAg4oCUIHN0YWJsZSBub2RlIElEIGZyb20gdGhlIGJ1aWxkZXIgKCdjYWxsLWxsbScsICdzZWVkJylcbiAqICAgZXhlY3V0aW9uSW5kZXgg4oCUIG1vbm90b25pYyBjb3VudGVyIGluY3JlbWVudGVkIHBlciBzdGFnZSBleGVjdXRpb24gKDAsIDEsIDIuLi4pXG4gKiAgIHN1YmZsb3dQYXRoICAgIOKAlCBvcHRpb25hbCBwYXRoIGZvciBzdWJmbG93IHN0YWdlcyAoJ3NmLXRvb2xzJywgJ3NmLW91dGVyL3NmLWlubmVyJylcbiAqXG4gKiBQcm9wZXJ0aWVzOlxuICogICAtIFVuaXF1ZSB3aXRoaW4gYSBydW4gKGV4ZWN1dGlvbkluZGV4IG5ldmVyIHJlcGVhdHMpXG4gKiAgIC0gRXhlY3V0aW9uLW9yZGVyZWQgKHNvcnQgYnkgZXhlY3V0aW9uSW5kZXggPSBleGVjdXRpb24gb3JkZXIpXG4gKiAgIC0gSHVtYW4tcmVhZGFibGUgKCdzZi10b29scy9leGVjdXRlLXRvb2wtY2FsbHMjOCcpXG4gKiAgIC0gUGFyc2VhYmxlIChzcGxpdCBvbiAnIycgZm9yIHN0YWdlSWQgYW5kIGluZGV4LCBzcGxpdCBzdGFnZUlkIG9uICcvJyBmb3Igc3ViZmxvdyBwYXRoKVxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGBcbiAqIGJ1aWxkUnVudGltZVN0YWdlSWQoJ2NhbGwtbGxtJywgNSkgICAgICAgICAgICAgICAgICAgIC8vICdjYWxsLWxsbSM1J1xuICogYnVpbGRSdW50aW1lU3RhZ2VJZCgnZXhlY3V0ZS10b29sLWNhbGxzJywgOCwgJ3NmLXRvb2xzJykgLy8gJ3NmLXRvb2xzL2V4ZWN1dGUtdG9vbC1jYWxscyM4J1xuICogYnVpbGRSdW50aW1lU3RhZ2VJZCgndmFsaWRhdGUnLCAzLCAnc2Ytb3V0ZXIvc2YtaW5uZXInKSAgLy8gJ3NmLW91dGVyL3NmLWlubmVyL3ZhbGlkYXRlIzMnXG4gKiBgYGBcbiAqL1xuXG4vKipcbiAqIEJ1aWxkIGEgcnVudGltZVN0YWdlSWQgZnJvbSBpdHMgY29tcG9uZW50cy5cbiAqXG4gKiBOb3RlOiBUaGUgdHJhdmVyc2VyIGRvZXMgTk9UIHVzZSB0aGUgc3ViZmxvd1BhdGggcGFyYW1ldGVyIOKAlCBub2RlLmlkIGFscmVhZHlcbiAqIGluY2x1ZGVzIHRoZSBzdWJmbG93IHByZWZpeCBmcm9tIHRoZSBidWlsZGVyLiBUaGlzIHBhcmFtZXRlciBleGlzdHMgZm9yIGV4dGVybmFsXG4gKiBjb25zdW1lcnMgY29uc3RydWN0aW5nIElEcyBmcm9tIHBhcnNlZCBjb21wb25lbnRzIChyb3VuZC10cmlwIHZpYSBwYXJzZVJ1bnRpbWVTdGFnZUlkKS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGJ1aWxkUnVudGltZVN0YWdlSWQoc3RhZ2VJZDogc3RyaW5nLCBleGVjdXRpb25JbmRleDogbnVtYmVyLCBzdWJmbG93UGF0aD86IHN0cmluZyk6IHN0cmluZyB7XG4gIGNvbnN0IHByZWZpeCA9IHN1YmZsb3dQYXRoID8gYCR7c3ViZmxvd1BhdGh9L2AgOiAnJztcbiAgcmV0dXJuIGAke3ByZWZpeH0ke3N0YWdlSWR9IyR7ZXhlY3V0aW9uSW5kZXh9YDtcbn1cblxuLyoqIFBhcnNlIGEgcnVudGltZVN0YWdlSWQgaW50byBpdHMgY29tcG9uZW50cy4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwYXJzZVJ1bnRpbWVTdGFnZUlkKHJ1bnRpbWVTdGFnZUlkOiBzdHJpbmcpOiB7XG4gIHN0YWdlSWQ6IHN0cmluZztcbiAgZXhlY3V0aW9uSW5kZXg6IG51bWJlcjtcbiAgc3ViZmxvd1BhdGg6IHN0cmluZyB8IHVuZGVmaW5lZDtcbn0ge1xuICBjb25zdCBoYXNoSWR4ID0gcnVudGltZVN0YWdlSWQubGFzdEluZGV4T2YoJyMnKTtcbiAgaWYgKGhhc2hJZHggPT09IC0xKSB7XG4gICAgcmV0dXJuIHsgc3RhZ2VJZDogcnVudGltZVN0YWdlSWQsIGV4ZWN1dGlvbkluZGV4OiAwLCBzdWJmbG93UGF0aDogdW5kZWZpbmVkIH07XG4gIH1cblxuICBjb25zdCBiZWZvcmVIYXNoID0gcnVudGltZVN0YWdlSWQuc2xpY2UoMCwgaGFzaElkeCk7XG4gIGNvbnN0IGV4ZWN1dGlvbkluZGV4ID0gcGFyc2VJbnQocnVudGltZVN0YWdlSWQuc2xpY2UoaGFzaElkeCArIDEpLCAxMCk7XG5cbiAgY29uc3QgbGFzdFNsYXNoID0gYmVmb3JlSGFzaC5sYXN0SW5kZXhPZignLycpO1xuICBpZiAobGFzdFNsYXNoID09PSAtMSkge1xuICAgIHJldHVybiB7IHN0YWdlSWQ6IGJlZm9yZUhhc2gsIGV4ZWN1dGlvbkluZGV4LCBzdWJmbG93UGF0aDogdW5kZWZpbmVkIH07XG4gIH1cblxuICByZXR1cm4ge1xuICAgIHN0YWdlSWQ6IGJlZm9yZUhhc2guc2xpY2UobGFzdFNsYXNoICsgMSksXG4gICAgZXhlY3V0aW9uSW5kZXgsXG4gICAgc3ViZmxvd1BhdGg6IGJlZm9yZUhhc2guc2xpY2UoMCwgbGFzdFNsYXNoKSxcbiAgfTtcbn1cblxuLyoqXG4gKiBTaGFyZWQgbXV0YWJsZSBjb3VudGVyIGZvciBleGVjdXRpb24gaW5kZXguXG4gKiBQYXNzZWQgYnkgcmVmZXJlbmNlIHRvIGNoaWxkIHRyYXZlcnNlcnMgKHN1YmZsb3dzKSBzbyB0aGV5XG4gKiBjb250aW51ZSB0aGUgZ2xvYmFsIG51bWJlcmluZyBpbnN0ZWFkIG9mIHJlc3RhcnRpbmcgYXQgMC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFeGVjdXRpb25Db3VudGVyIHtcbiAgdmFsdWU6IG51bWJlcjtcbn1cblxuLyoqIENyZWF0ZSBhIG5ldyBleGVjdXRpb24gY291bnRlciBzdGFydGluZyBhdCAwLiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUV4ZWN1dGlvbkNvdW50ZXIoKTogRXhlY3V0aW9uQ291bnRlciB7XG4gIHJldHVybiB7IHZhbHVlOiAwIH07XG59XG4iXX0=
|