footprintjs 4.14.0 → 4.15.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CLAUDE.md CHANGED
@@ -268,6 +268,83 @@ const llmCommit = findCommit(commitLog, 'call-llm', 'adapterRawResponse');
268
268
  | `findLastWriter(commitLog, key, beforeIdx?)` | `CommitBundle \| undefined` | Search backwards for who wrote a key |
269
269
  | `KeyedRecorder<T>` | abstract class | Base for 1:1 Map-based recorders |
270
270
  | `SequenceRecorder<T>` | abstract class | Base for 1:N ordered sequence recorders (has `getEntryRanges()` for O(1) time-travel) |
271
+ | `topologyRecorder()` / `TopologyRecorder` | factory / class | Live composition graph for streaming consumers (subflow nodes + control-flow edges) |
272
+
273
+ ### TopologyRecorder — Composition Graph for Streaming Consumers
274
+
275
+ **One-liner:** reconstructs a live, queryable mini-flowchart of what your run actually traced, built from the 3 primitive recorder channels during traversal.
276
+
277
+ **Mental model:**
278
+
279
+ ```
280
+ flowChart() builder → STATIC flowchart (design-time definition)
281
+
282
+ ▼ executor runs it
283
+ Traversal emits events on 3 channels:
284
+ Recorder · FlowRecorder · EmitRecorder
285
+
286
+ ▼ TopologyRecorder listens
287
+ DYNAMIC flowchart (runtime shape):
288
+ Nodes = composition points
289
+ (subflow / fork-branch / decision-branch)
290
+ Edges = transitions
291
+ (next / fork / decision / loop)
292
+ Queryable any moment — during or after run
293
+ ```
294
+
295
+ **What it IS:**
296
+ - Live composition graph derived from 3 primitive channels
297
+ - Each node = one composition-significant moment (subflow entered, fork child, decision chosen)
298
+ - Each edge = a control-flow transition, timestamped with `runtimeStageId`
299
+ - Works identically during or after a run
300
+
301
+ **What it ISN'T:**
302
+ - Not a full execution tree — that's `StageContext` / `executor.getSnapshot()`
303
+ - Not per-stage data — that's `MetricRecorder` / custom `KeyedRecorder<T>`
304
+ - Not agent-specific — agentfootprint composes it; footprintjs owns it
305
+
306
+ **Why live consumers need it:** The executor already has the topology internally (execution tree in `StageContext`). But streaming consumers can't access that tree mid-run — they only see events. `TopologyRecorder` = "the tree, reconstructed from events, live-queryable."
307
+
308
+ Fills the gap between "post-run snapshot (full tree available)" and "live event stream (only point observations)." Attach once; query `getTopology()` anytime during or after a run.
309
+
310
+ ```typescript
311
+ import { topologyRecorder } from 'footprintjs/trace';
312
+
313
+ const topo = topologyRecorder();
314
+ executor.attachCombinedRecorder(topo); // auto-routes to FlowRecorder channel
315
+
316
+ await executor.run({ input });
317
+
318
+ const { nodes, edges, activeNodeId, rootId } = topo.getTopology();
319
+ topo.getSubflowNodes(); // agent-centric view
320
+ topo.getByKind('fork-branch'); // all parallel branches
321
+ topo.getParallelSiblings(id); // siblings of a parallel branch
322
+ ```
323
+
324
+ **Three node kinds — complete composition coverage:**
325
+
326
+ | Kind | Fires on | Represents |
327
+ |---|---|---|
328
+ | `subflow` | `onSubflowEntry` | Mounted subflow boundary (with stable `subflowId`) |
329
+ | `fork-branch` | `onFork` (synthesized one per child) | One branch of a parallel split — works for plain stages AND subflows |
330
+ | `decision-branch` | `onDecision` (synthesized for chosen) | The chosen branch of a conditional |
331
+
332
+ When a fork-branch or decision-branch target is also a subflow, the subsequent `onSubflowEntry` creates a subflow CHILD of the synthetic node. Layered shape preserves both "who branched" and "what the branch ran."
333
+
334
+ **Edges:** one per control-flow transition. `edge.kind ∈ 'next' | 'fork-branch' | 'decision-branch' | 'loop-iteration'`. Each carries `at: runtimeStageId` for time correlation.
335
+
336
+ **Correlation rules:**
337
+ - `onFork({ parent, children })` → N `fork-branch` nodes synthesized up-front; subsequent matching `onSubflowEntry` nests under the right fork-branch
338
+ - `onDecision({ chosen })` → `decision-branch` node synthesized up-front; matching `onSubflowEntry` nests under it
339
+ - Pending correlation clears on `onSubflowExit` so state doesn't leak across scopes
340
+ - `onLoop` → self-edge on the currently-active subflow (synthetic nodes don't participate)
341
+ - Re-entry of same `subflowId` (loop body) disambiguates via `id#n` suffix
342
+
343
+ **What it does NOT track:** plain sequential stages. Use `MetricRecorder` / `StageContext` for per-stage data. Topology is a graph of control-flow branching, not a full execution tree.
344
+
345
+ **For downstream libraries:** compose, don't duplicate. An agent-shaped recorder should wrap a `topologyRecorder()` internally and translate topology nodes into agent semantics — not re-implement subflow-stack + fork + decision tracking.
346
+
347
+ Example: [examples/flow-recorders/06-topology-recorder.ts](examples/flow-recorders/06-topology-recorder.ts)
271
348
 
272
349
  **Two recorder base classes** — choose based on data shape:
273
350
 
@@ -0,0 +1,272 @@
1
+ /**
2
+ * TopologyRecorder — composition graph built during traversal.
3
+ *
4
+ * The gap this fills:
5
+ * footprintjs fires atomic flow events (onSubflowEntry, onFork, onDecision,
6
+ * onLoop) but the accumulated *shape* of a run — who nests inside whom,
7
+ * which nodes are parallel siblings vs branches of a decision — is only
8
+ * visible post-run via `executor.getSnapshot()` tree-walking.
9
+ *
10
+ * Streaming consumers (live UIs, in-flight debuggers) see only the event
11
+ * stream. Every such consumer has to rebuild subflow-stack + fork-map +
12
+ * decision-tracker from scratch, usually slightly wrong in different ways.
13
+ *
14
+ * TopologyRecorder is the standard accumulator: one subscription to the
15
+ * three primitive channels, one live graph, queryable at any moment during
16
+ * or after a run.
17
+ *
18
+ * What it records — THREE node kinds for complete composition coverage:
19
+ * 1. 'subflow' — via onSubflowEntry (a mounted subflow boundary)
20
+ * 2. 'fork-branch' — via onFork (one node per child, synthesized)
21
+ * 3. 'decision-branch' — via onDecision (the chosen branch, synthesized)
22
+ *
23
+ * When a fork-branch or decision-branch target IS ALSO a subflow, the
24
+ * subsequent onSubflowEntry creates a subflow CHILD of the synthetic node.
25
+ * The layered shape preserves both "who branched" and "what the branch ran."
26
+ *
27
+ * Plain sequential stages are NOT nodes — that's StageContext's job.
28
+ * Topology is a graph of control-flow branching, not a full execution tree.
29
+ *
30
+ * Edges:
31
+ * One edge per traversal transition — `kind` matches the child's
32
+ * `incomingKind`. A consumer rendering "parallel columns" filters edges
33
+ * where `kind === 'fork-branch'` sharing the same `from`.
34
+ *
35
+ * @example
36
+ * ```typescript
37
+ * import { topologyRecorder } from 'footprintjs/trace';
38
+ *
39
+ * const topo = topologyRecorder();
40
+ * executor.attachCombinedRecorder(topo); // auto-routes to FlowRecorder channel
41
+ *
42
+ * await executor.run();
43
+ *
44
+ * const { nodes, edges, activeNodeId, rootId } = topo.getTopology();
45
+ * // Consumer queries:
46
+ * topo.getChildren('sf-parent'); // direct children (any kind)
47
+ * topo.getByKind('fork-branch'); // all parallel branches
48
+ * topo.getSubflowNodes(); // only mounted subflows
49
+ * ```
50
+ */
51
+ let _counter = 0;
52
+ /**
53
+ * Factory — matches the `narrative()` / `metrics()` style.
54
+ */
55
+ export function topologyRecorder(options = {}) {
56
+ return new TopologyRecorder(options);
57
+ }
58
+ /**
59
+ * Stateful accumulator that watches FlowRecorder events and maintains a live
60
+ * composition graph. Attach via `executor.attachCombinedRecorder(recorder)` —
61
+ * footprintjs detects the `FlowRecorder` method shape and routes events.
62
+ */
63
+ export class TopologyRecorder {
64
+ constructor(options = {}) {
65
+ var _a;
66
+ this.nodesById = new Map();
67
+ this.nodeOrder = [];
68
+ this.edges = [];
69
+ /** Stack of active SUBFLOW node ids. Fork/decision-branch nodes never push. */
70
+ this.subflowStack = [];
71
+ /** Map of childName → pending fork-branch synthetic node, consumed by
72
+ * the next matching `onSubflowEntry`. */
73
+ this.pendingForkByName = new Map();
74
+ this.id = (_a = options.id) !== null && _a !== void 0 ? _a : `topology-${++_counter}`;
75
+ }
76
+ // ── FlowRecorder hooks ────────────────────────────────────────────────
77
+ onSubflowEntry(event) {
78
+ var _a, _b;
79
+ const subflowId = event.subflowId;
80
+ if (!subflowId)
81
+ return; // Need a stable id to track.
82
+ const enteredAt = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
83
+ // Determine the parent: prefer a pending fork/decision match by name,
84
+ // otherwise the current top-of-subflow-stack.
85
+ let parentId;
86
+ let incomingKind;
87
+ const pendingFork = this.pendingForkByName.get(event.name);
88
+ if (pendingFork) {
89
+ parentId = pendingFork.nodeId;
90
+ incomingKind = 'next'; // Child OF a fork-branch node; the fork semantic
91
+ // is captured by the fork-branch's own incomingKind.
92
+ this.pendingForkByName.delete(event.name);
93
+ }
94
+ else if (this.pendingDecision && this.pendingDecision.name === event.name) {
95
+ parentId = this.pendingDecision.nodeId;
96
+ incomingKind = 'next';
97
+ this.pendingDecision = undefined;
98
+ }
99
+ else {
100
+ parentId = this.subflowStack[this.subflowStack.length - 1];
101
+ incomingKind = parentId ? 'next' : 'root';
102
+ }
103
+ // Disambiguate re-entry (e.g., loop body re-enters the same subflow).
104
+ let nodeId = subflowId;
105
+ if (this.nodesById.has(nodeId)) {
106
+ let n = 1;
107
+ while (this.nodesById.has(`${subflowId}#${n}`))
108
+ n++;
109
+ nodeId = `${subflowId}#${n}`;
110
+ }
111
+ const depth = parentId ? this.nodesById.get(parentId).depth + 1 : 0;
112
+ const metadata = event.description ? { description: event.description } : undefined;
113
+ const node = {
114
+ id: nodeId,
115
+ kind: 'subflow',
116
+ name: event.name,
117
+ parentId,
118
+ depth,
119
+ incomingKind,
120
+ enteredAt,
121
+ metadata,
122
+ };
123
+ this.nodesById.set(nodeId, node);
124
+ this.nodeOrder.push(nodeId);
125
+ if (parentId && incomingKind !== 'root') {
126
+ this.edges.push({
127
+ from: parentId,
128
+ to: nodeId,
129
+ kind: incomingKind,
130
+ at: enteredAt,
131
+ });
132
+ }
133
+ this.subflowStack.push(nodeId);
134
+ }
135
+ onSubflowExit(event) {
136
+ var _a, _b;
137
+ const nodeId = this.subflowStack.pop();
138
+ if (!nodeId)
139
+ return;
140
+ const node = this.nodesById.get(nodeId);
141
+ if (node) {
142
+ node.exitedAt = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
143
+ }
144
+ // A subflow exit implies no more children are pending for this scope —
145
+ // clear stale pending state that belonged to the scope we just left.
146
+ this.pendingForkByName.clear();
147
+ this.pendingDecision = undefined;
148
+ }
149
+ onFork(event) {
150
+ var _a, _b;
151
+ const activeId = this.subflowStack[this.subflowStack.length - 1];
152
+ const at = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
153
+ const depth = activeId ? this.nodesById.get(activeId).depth + 1 : 0;
154
+ // Reset any prior pending fork state — a new fork starts fresh.
155
+ this.pendingForkByName.clear();
156
+ event.children.forEach((childName, i) => {
157
+ const nodeId = `fork-${at || event.parent}-${i}-${childName}`;
158
+ const node = {
159
+ id: nodeId,
160
+ kind: 'fork-branch',
161
+ name: childName,
162
+ parentId: activeId,
163
+ depth,
164
+ incomingKind: 'fork-branch',
165
+ enteredAt: at,
166
+ metadata: { forkParent: event.parent },
167
+ };
168
+ this.nodesById.set(nodeId, node);
169
+ this.nodeOrder.push(nodeId);
170
+ if (activeId) {
171
+ this.edges.push({ from: activeId, to: nodeId, kind: 'fork-branch', at });
172
+ }
173
+ this.pendingForkByName.set(childName, { nodeId, at });
174
+ });
175
+ }
176
+ onDecision(event) {
177
+ var _a, _b;
178
+ const activeId = this.subflowStack[this.subflowStack.length - 1];
179
+ const at = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
180
+ const depth = activeId ? this.nodesById.get(activeId).depth + 1 : 0;
181
+ // A new decision supersedes any prior unresolved pending one.
182
+ this.pendingDecision = undefined;
183
+ const nodeId = `decision-${at || event.decider}-${event.chosen}`;
184
+ const metadata = { decider: event.decider };
185
+ if (event.rationale)
186
+ metadata.rationale = event.rationale;
187
+ if (event.description)
188
+ metadata.description = event.description;
189
+ const node = {
190
+ id: nodeId,
191
+ kind: 'decision-branch',
192
+ name: event.chosen,
193
+ parentId: activeId,
194
+ depth,
195
+ incomingKind: 'decision-branch',
196
+ enteredAt: at,
197
+ metadata,
198
+ };
199
+ this.nodesById.set(nodeId, node);
200
+ this.nodeOrder.push(nodeId);
201
+ if (activeId) {
202
+ this.edges.push({ from: activeId, to: nodeId, kind: 'decision-branch', at });
203
+ }
204
+ this.pendingDecision = { name: event.chosen, nodeId, at };
205
+ }
206
+ onLoop(event) {
207
+ var _a, _b;
208
+ // loopTo jumps back inside the CURRENT subflow. Record a self-edge on the
209
+ // active subflow — synthetic fork/decision nodes don't participate in loops.
210
+ const activeId = this.subflowStack[this.subflowStack.length - 1];
211
+ if (!activeId)
212
+ return;
213
+ this.edges.push({
214
+ from: activeId,
215
+ to: activeId,
216
+ kind: 'loop-iteration',
217
+ at: (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '',
218
+ });
219
+ }
220
+ /** Called by the executor before each `run()` — resets all state. */
221
+ clear() {
222
+ this.nodesById.clear();
223
+ this.nodeOrder.length = 0;
224
+ this.edges.length = 0;
225
+ this.subflowStack.length = 0;
226
+ this.pendingForkByName.clear();
227
+ this.pendingDecision = undefined;
228
+ }
229
+ // ── Query API ─────────────────────────────────────────────────────────
230
+ /** Live snapshot of the composition graph. Safe during or after a run. */
231
+ getTopology() {
232
+ var _a, _b;
233
+ const nodes = this.nodeOrder.map((id) => this.nodesById.get(id));
234
+ return {
235
+ nodes,
236
+ edges: [...this.edges],
237
+ activeNodeId: (_a = this.subflowStack[this.subflowStack.length - 1]) !== null && _a !== void 0 ? _a : null,
238
+ rootId: (_b = this.nodeOrder[0]) !== null && _b !== void 0 ? _b : null,
239
+ };
240
+ }
241
+ /** Direct children of a node — insertion-ordered. */
242
+ getChildren(nodeId) {
243
+ return this.nodeOrder.map((id) => this.nodesById.get(id)).filter((n) => n.parentId === nodeId);
244
+ }
245
+ /** All nodes of a given kind. */
246
+ getByKind(kind) {
247
+ return this.nodeOrder.map((id) => this.nodesById.get(id)).filter((n) => n.kind === kind);
248
+ }
249
+ /** All mounted subflow nodes. Convenience for agent-centric views. */
250
+ getSubflowNodes() {
251
+ return this.getByKind('subflow');
252
+ }
253
+ /** All fork-branch nodes sharing the same parent as `nodeId` — i.e.,
254
+ * parallel siblings of a parallel branch. Empty if `nodeId` isn't a
255
+ * fork-branch or has no parent. */
256
+ getParallelSiblings(nodeId) {
257
+ const node = this.nodesById.get(nodeId);
258
+ if (!node || node.kind !== 'fork-branch' || !node.parentId)
259
+ return [];
260
+ return this.getChildren(node.parentId).filter((n) => n.kind === 'fork-branch');
261
+ }
262
+ /** Emit a snapshot bundle for inclusion in `executor.getSnapshot()`. */
263
+ toSnapshot() {
264
+ return {
265
+ name: 'Topology',
266
+ description: 'Composition graph: subflow boundaries, fork branches, decision branches',
267
+ preferredOperation: 'translate',
268
+ data: this.getTopology(),
269
+ };
270
+ }
271
+ }
272
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TopologyRecorder.js","sourceRoot":"","sources":["../../../../src/lib/recorder/TopologyRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * TopologyRecorder — composition graph built during traversal.\n *\n * The gap this fills:\n *   footprintjs fires atomic flow events (onSubflowEntry, onFork, onDecision,\n *   onLoop) but the accumulated *shape* of a run — who nests inside whom,\n *   which nodes are parallel siblings vs branches of a decision — is only\n *   visible post-run via `executor.getSnapshot()` tree-walking.\n *\n *   Streaming consumers (live UIs, in-flight debuggers) see only the event\n *   stream. Every such consumer has to rebuild subflow-stack + fork-map +\n *   decision-tracker from scratch, usually slightly wrong in different ways.\n *\n *   TopologyRecorder is the standard accumulator: one subscription to the\n *   three primitive channels, one live graph, queryable at any moment during\n *   or after a run.\n *\n * What it records — THREE node kinds for complete composition coverage:\n *   1. 'subflow'          — via onSubflowEntry (a mounted subflow boundary)\n *   2. 'fork-branch'      — via onFork (one node per child, synthesized)\n *   3. 'decision-branch'  — via onDecision (the chosen branch, synthesized)\n *\n *   When a fork-branch or decision-branch target IS ALSO a subflow, the\n *   subsequent onSubflowEntry creates a subflow CHILD of the synthetic node.\n *   The layered shape preserves both \"who branched\" and \"what the branch ran.\"\n *\n *   Plain sequential stages are NOT nodes — that's StageContext's job.\n *   Topology is a graph of control-flow branching, not a full execution tree.\n *\n * Edges:\n *   One edge per traversal transition — `kind` matches the child's\n *   `incomingKind`. A consumer rendering \"parallel columns\" filters edges\n *   where `kind === 'fork-branch'` sharing the same `from`.\n *\n * @example\n * ```typescript\n * import { topologyRecorder } from 'footprintjs/trace';\n *\n * const topo = topologyRecorder();\n * executor.attachCombinedRecorder(topo);  // auto-routes to FlowRecorder channel\n *\n * await executor.run();\n *\n * const { nodes, edges, activeNodeId, rootId } = topo.getTopology();\n * // Consumer queries:\n * topo.getChildren('sf-parent');              // direct children (any kind)\n * topo.getByKind('fork-branch');              // all parallel branches\n * topo.getSubflowNodes();                     // only mounted subflows\n * ```\n */\n\nimport type {\n  FlowDecisionEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSubflowEvent,\n} from '../engine/narrative/types.js';\n\n/** The kind of composition unit a node represents. */\nexport type TopologyNodeKind = 'subflow' | 'fork-branch' | 'decision-branch';\n\n/** How the traversal reached this node — drives consumer layout decisions. */\nexport type TopologyIncomingKind = 'root' | 'next' | 'fork-branch' | 'decision-branch' | 'loop-iteration';\n\n/** A composition-significant point in the graph. */\nexport interface TopologyNode {\n  /** Unique id. Subflows use their subflowId (with `#n` suffix on re-entry).\n   *  Synthetic nodes (fork-branch / decision-branch) use\n   *  `fork-${runtimeStageId}-${i}` / `decision-${runtimeStageId}` form. */\n  readonly id: string;\n  /** What this node represents. */\n  readonly kind: TopologyNodeKind;\n  /** Display name. For subflows: `FlowSubflowEvent.name`. For fork-branches:\n   *  the child name from `FlowForkEvent.children`. For decision-branches:\n   *  the chosen name from `FlowDecisionEvent.chosen`. */\n  readonly name: string;\n  /** Parent node id. Undefined when this node sits at the run's top level. */\n  readonly parentId?: string;\n  /** Depth in the topology tree (0 = top-level). */\n  readonly depth: number;\n  /** How the traversal reached this node. */\n  readonly incomingKind: TopologyIncomingKind;\n  /** runtimeStageId at the moment the node was created. */\n  readonly enteredAt: string;\n  /** runtimeStageId when the corresponding subflow exited. Only meaningful\n   *  for kind='subflow'; fork/decision-branch nodes are instantaneous. */\n  exitedAt?: string;\n  /** Kind-specific extras: forkParent, decider, rationale, description. */\n  readonly metadata?: Readonly<Record<string, unknown>>;\n}\n\n/** A traversal transition between two nodes. */\nexport interface TopologyEdge {\n  readonly from: string;\n  readonly to: string;\n  readonly kind: Exclude<TopologyIncomingKind, 'root'>;\n  readonly at: string;\n}\n\n/** Snapshot of the composition graph. */\nexport interface Topology {\n  readonly nodes: ReadonlyArray<TopologyNode>;\n  readonly edges: ReadonlyArray<TopologyEdge>;\n  /** Currently-active subflow (top of the subflow stack). Fork-branch and\n   *  decision-branch nodes are instantaneous — they don't affect activeNodeId. */\n  readonly activeNodeId: string | null;\n  /** First node inserted. null before any composition event fires. */\n  readonly rootId: string | null;\n}\n\nexport interface TopologyRecorderOptions {\n  /** Recorder id. Defaults to `topology-N` (auto-incremented). */\n  id?: string;\n}\n\n// Correlation state: maps a pending fork/decision child name to its synthetic\n// node id, so a subsequent onSubflowEntry matching that name can be nested\n// under the synthetic node (rather than creating a peer).\ninterface PendingChild {\n  nodeId: string;\n  at: string;\n}\n\nlet _counter = 0;\n\n/**\n * Factory — matches the `narrative()` / `metrics()` style.\n */\nexport function topologyRecorder(options: TopologyRecorderOptions = {}): TopologyRecorder {\n  return new TopologyRecorder(options);\n}\n\n/**\n * Stateful accumulator that watches FlowRecorder events and maintains a live\n * composition graph. Attach via `executor.attachCombinedRecorder(recorder)` —\n * footprintjs detects the `FlowRecorder` method shape and routes events.\n */\nexport class TopologyRecorder implements FlowRecorder {\n  readonly id: string;\n\n  private readonly nodesById = new Map<string, TopologyNode>();\n  private readonly nodeOrder: string[] = [];\n  private readonly edges: TopologyEdge[] = [];\n  /** Stack of active SUBFLOW node ids. Fork/decision-branch nodes never push. */\n  private readonly subflowStack: string[] = [];\n\n  /** Map of childName → pending fork-branch synthetic node, consumed by\n   *  the next matching `onSubflowEntry`. */\n  private readonly pendingForkByName = new Map<string, PendingChild>();\n  /** Pending decision-branch synthetic node, consumed by a matching entry. */\n  private pendingDecision?: { name: string } & PendingChild;\n\n  constructor(options: TopologyRecorderOptions = {}) {\n    this.id = options.id ?? `topology-${++_counter}`;\n  }\n\n  // ── FlowRecorder hooks ────────────────────────────────────────────────\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    const subflowId = event.subflowId;\n    if (!subflowId) return; // Need a stable id to track.\n\n    const enteredAt = event.traversalContext?.runtimeStageId ?? '';\n\n    // Determine the parent: prefer a pending fork/decision match by name,\n    // otherwise the current top-of-subflow-stack.\n    let parentId: string | undefined;\n    let incomingKind: TopologyIncomingKind;\n\n    const pendingFork = this.pendingForkByName.get(event.name);\n    if (pendingFork) {\n      parentId = pendingFork.nodeId;\n      incomingKind = 'next'; // Child OF a fork-branch node; the fork semantic\n      // is captured by the fork-branch's own incomingKind.\n      this.pendingForkByName.delete(event.name);\n    } else if (this.pendingDecision && this.pendingDecision.name === event.name) {\n      parentId = this.pendingDecision.nodeId;\n      incomingKind = 'next';\n      this.pendingDecision = undefined;\n    } else {\n      parentId = this.subflowStack[this.subflowStack.length - 1];\n      incomingKind = parentId ? 'next' : 'root';\n    }\n\n    // Disambiguate re-entry (e.g., loop body re-enters the same subflow).\n    let nodeId = subflowId;\n    if (this.nodesById.has(nodeId)) {\n      let n = 1;\n      while (this.nodesById.has(`${subflowId}#${n}`)) n++;\n      nodeId = `${subflowId}#${n}`;\n    }\n\n    const depth = parentId ? this.nodesById.get(parentId)!.depth + 1 : 0;\n    const metadata = event.description ? { description: event.description } : undefined;\n\n    const node: TopologyNode = {\n      id: nodeId,\n      kind: 'subflow',\n      name: event.name,\n      parentId,\n      depth,\n      incomingKind,\n      enteredAt,\n      metadata,\n    };\n    this.nodesById.set(nodeId, node);\n    this.nodeOrder.push(nodeId);\n\n    if (parentId && incomingKind !== 'root') {\n      this.edges.push({\n        from: parentId,\n        to: nodeId,\n        kind: incomingKind,\n        at: enteredAt,\n      });\n    }\n\n    this.subflowStack.push(nodeId);\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    const nodeId = this.subflowStack.pop();\n    if (!nodeId) return;\n    const node = this.nodesById.get(nodeId);\n    if (node) {\n      node.exitedAt = event.traversalContext?.runtimeStageId ?? '';\n    }\n    // A subflow exit implies no more children are pending for this scope —\n    // clear stale pending state that belonged to the scope we just left.\n    this.pendingForkByName.clear();\n    this.pendingDecision = undefined;\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const activeId = this.subflowStack[this.subflowStack.length - 1];\n    const at = event.traversalContext?.runtimeStageId ?? '';\n    const depth = activeId ? this.nodesById.get(activeId)!.depth + 1 : 0;\n\n    // Reset any prior pending fork state — a new fork starts fresh.\n    this.pendingForkByName.clear();\n\n    event.children.forEach((childName, i) => {\n      const nodeId = `fork-${at || event.parent}-${i}-${childName}`;\n      const node: TopologyNode = {\n        id: nodeId,\n        kind: 'fork-branch',\n        name: childName,\n        parentId: activeId,\n        depth,\n        incomingKind: 'fork-branch',\n        enteredAt: at,\n        metadata: { forkParent: event.parent },\n      };\n      this.nodesById.set(nodeId, node);\n      this.nodeOrder.push(nodeId);\n      if (activeId) {\n        this.edges.push({ from: activeId, to: nodeId, kind: 'fork-branch', at });\n      }\n      this.pendingForkByName.set(childName, { nodeId, at });\n    });\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const activeId = this.subflowStack[this.subflowStack.length - 1];\n    const at = event.traversalContext?.runtimeStageId ?? '';\n    const depth = activeId ? this.nodesById.get(activeId)!.depth + 1 : 0;\n\n    // A new decision supersedes any prior unresolved pending one.\n    this.pendingDecision = undefined;\n\n    const nodeId = `decision-${at || event.decider}-${event.chosen}`;\n    const metadata: Record<string, unknown> = { decider: event.decider };\n    if (event.rationale) metadata.rationale = event.rationale;\n    if (event.description) metadata.description = event.description;\n\n    const node: TopologyNode = {\n      id: nodeId,\n      kind: 'decision-branch',\n      name: event.chosen,\n      parentId: activeId,\n      depth,\n      incomingKind: 'decision-branch',\n      enteredAt: at,\n      metadata,\n    };\n    this.nodesById.set(nodeId, node);\n    this.nodeOrder.push(nodeId);\n    if (activeId) {\n      this.edges.push({ from: activeId, to: nodeId, kind: 'decision-branch', at });\n    }\n    this.pendingDecision = { name: event.chosen, nodeId, at };\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    // loopTo jumps back inside the CURRENT subflow. Record a self-edge on the\n    // active subflow — synthetic fork/decision nodes don't participate in loops.\n    const activeId = this.subflowStack[this.subflowStack.length - 1];\n    if (!activeId) return;\n    this.edges.push({\n      from: activeId,\n      to: activeId,\n      kind: 'loop-iteration',\n      at: event.traversalContext?.runtimeStageId ?? '',\n    });\n  }\n\n  /** Called by the executor before each `run()` — resets all state. */\n  clear(): void {\n    this.nodesById.clear();\n    this.nodeOrder.length = 0;\n    this.edges.length = 0;\n    this.subflowStack.length = 0;\n    this.pendingForkByName.clear();\n    this.pendingDecision = undefined;\n  }\n\n  // ── Query API ─────────────────────────────────────────────────────────\n\n  /** Live snapshot of the composition graph. Safe during or after a run. */\n  getTopology(): Topology {\n    const nodes = this.nodeOrder.map((id) => this.nodesById.get(id)!);\n    return {\n      nodes,\n      edges: [...this.edges],\n      activeNodeId: this.subflowStack[this.subflowStack.length - 1] ?? null,\n      rootId: this.nodeOrder[0] ?? null,\n    };\n  }\n\n  /** Direct children of a node — insertion-ordered. */\n  getChildren(nodeId: string): TopologyNode[] {\n    return this.nodeOrder.map((id) => this.nodesById.get(id)!).filter((n) => n.parentId === nodeId);\n  }\n\n  /** All nodes of a given kind. */\n  getByKind(kind: TopologyNodeKind): TopologyNode[] {\n    return this.nodeOrder.map((id) => this.nodesById.get(id)!).filter((n) => n.kind === kind);\n  }\n\n  /** All mounted subflow nodes. Convenience for agent-centric views. */\n  getSubflowNodes(): TopologyNode[] {\n    return this.getByKind('subflow');\n  }\n\n  /** All fork-branch nodes sharing the same parent as `nodeId` — i.e.,\n   *  parallel siblings of a parallel branch. Empty if `nodeId` isn't a\n   *  fork-branch or has no parent. */\n  getParallelSiblings(nodeId: string): TopologyNode[] {\n    const node = this.nodesById.get(nodeId);\n    if (!node || node.kind !== 'fork-branch' || !node.parentId) return [];\n    return this.getChildren(node.parentId).filter((n) => n.kind === 'fork-branch');\n  }\n\n  /** Emit a snapshot bundle for inclusion in `executor.getSnapshot()`. */\n  toSnapshot() {\n    return {\n      name: 'Topology',\n      description: 'Composition graph: subflow boundaries, fork branches, decision branches',\n      preferredOperation: 'translate' as const,\n      data: this.getTopology(),\n    };\n  }\n}\n"]}
package/dist/esm/trace.js CHANGED
@@ -28,6 +28,7 @@ export { causalChain, flattenCausalDAG, formatCausalChain } from './lib/memory/b
28
28
  export { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';
29
29
  // SequenceRecorder — base class for 1:N ordered sequence recorders with keyed index
30
30
  export { SequenceRecorder } from './lib/recorder/SequenceRecorder.js';
31
+ export { TopologyRecorder, topologyRecorder } from './lib/recorder/TopologyRecorder.js';
31
32
  export { QualityRecorder } from './lib/recorder/QualityRecorder.js';
32
33
  export { formatQualityTrace, qualityTrace } from './lib/recorder/qualityTrace.js';
33
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHJhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXFCRztBQUlILE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxzQkFBc0IsRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBRWxILHdEQUF3RDtBQUN4RCxPQUFPLEVBQUUsVUFBVSxFQUFFLFdBQVcsRUFBRSxjQUFjLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUl6RixPQUFPLEVBQUUsV0FBVyxFQUFFLGdCQUFnQixFQUFFLGlCQUFpQixFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFFN0YseURBQXlEO0FBQ3pELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUVoRSxvRkFBb0Y7QUFDcEYsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFJdEUsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBSXBFLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxZQUFZLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogZm9vdHByaW50anMvdHJhY2Ug4oCUIEV4ZWN1dGlvbiB0cmFjaW5nLCBkZWJ1Z2dpbmcsIGFuZCBiYWNrdHJhY2tpbmcgdXRpbGl0aWVzLlxuICpcbiAqIFJ1bnRpbWUgc3RhZ2UgSURzLCBjb21taXQgbG9nIHF1ZXJpZXMsIGFuZCByZWNvcmRlciBiYXNlIGNsYXNzZXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IHBhcnNlUnVudGltZVN0YWdlSWQsIGZpbmRMYXN0V3JpdGVyLCBLZXllZFJlY29yZGVyLCBTZXF1ZW5jZVJlY29yZGVyIH0gZnJvbSAnZm9vdHByaW50anMvdHJhY2UnO1xuICpcbiAqIC8vIFBhcnNlIGEgcnVudGltZVN0YWdlSWRcbiAqIGNvbnN0IHsgc3RhZ2VJZCwgZXhlY3V0aW9uSW5kZXggfSA9IHBhcnNlUnVudGltZVN0YWdlSWQoJ2NhbGwtbGxtIzUnKTtcbiAqXG4gKiAvLyBCYWNrdHJhY2s6IHdobyB3cm90ZSAnc3lzdGVtUHJvbXB0JyBiZWZvcmUgc3RhZ2UgYXQgaWR4IDg/XG4gKiBjb25zdCB3cml0ZXIgPSBmaW5kTGFzdFdyaXRlcihjb21taXRMb2csICdzeXN0ZW1Qcm9tcHQnLCA4KTtcbiAqXG4gKiAvLyBCdWlsZCBhIGtleWVkIHJlY29yZGVyICgxOjEg4oCUIG9uZSBlbnRyeSBwZXIgc3RlcClcbiAqIGNsYXNzIE15UmVjb3JkZXIgZXh0ZW5kcyBLZXllZFJlY29yZGVyPE15RW50cnk+IHsgLi4uIH1cbiAqXG4gKiAvLyBCdWlsZCBhIHNlcXVlbmNlIHJlY29yZGVyICgxOk4g4oCUIG11bHRpcGxlIGVudHJpZXMgcGVyIHN0ZXAsIG9yZGVyaW5nIG1hdHRlcnMpXG4gKiBjbGFzcyBBdWRpdFJlY29yZGVyIGV4dGVuZHMgU2VxdWVuY2VSZWNvcmRlcjxBdWRpdEVudHJ5PiB7IC4uLiB9XG4gKiBgYGBcbiAqL1xuXG4vLyBSdW50aW1lIHN0YWdlIElEIOKAlCB1bmlxdWUgZXhlY3V0aW9uIHN0ZXAgaWRlbnRpZmllcnNcbmV4cG9ydCB0eXBlIHsgRXhlY3V0aW9uQ291bnRlciB9IGZyb20gJy4vbGliL2VuZ2luZS9ydW50aW1lU3RhZ2VJZC5qcyc7XG5leHBvcnQgeyBidWlsZFJ1bnRpbWVTdGFnZUlkLCBjcmVhdGVFeGVjdXRpb25Db3VudGVyLCBwYXJzZVJ1bnRpbWVTdGFnZUlkIH0gZnJvbSAnLi9saWIvZW5naW5lL3J1bnRpbWVTdGFnZUlkLmpzJztcblxuLy8gQ29tbWl0IGxvZyBxdWVyaWVzIOKAlCB0eXBlZCB1dGlsaXRpZXMgZm9yIGJhY2t0cmFja2luZ1xuZXhwb3J0IHsgZmluZENvbW1pdCwgZmluZENvbW1pdHMsIGZpbmRMYXN0V3JpdGVyIH0gZnJvbSAnLi9saWIvbWVtb3J5L2NvbW1pdExvZ1V0aWxzLmpzJztcblxuLy8gQ2F1c2FsIGNoYWluIOKAlCBiYWNrd2FyZCBwcm9ncmFtIHNsaWNpbmcgb24gY29tbWl0IGxvZyAoREFHKVxuZXhwb3J0IHR5cGUgeyBDYXVzYWxDaGFpbk9wdGlvbnMsIENhdXNhbE5vZGUsIEtleXNSZWFkTG9va3VwIH0gZnJvbSAnLi9saWIvbWVtb3J5L2JhY2t0cmFjay5qcyc7XG5leHBvcnQgeyBjYXVzYWxDaGFpbiwgZmxhdHRlbkNhdXNhbERBRywgZm9ybWF0Q2F1c2FsQ2hhaW4gfSBmcm9tICcuL2xpYi9tZW1vcnkvYmFja3RyYWNrLmpzJztcblxuLy8gS2V5ZWRSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMToxIE1hcC1iYXNlZCByZWNvcmRlcnNcbmV4cG9ydCB7IEtleWVkUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9LZXllZFJlY29yZGVyLmpzJztcblxuLy8gU2VxdWVuY2VSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMTpOIG9yZGVyZWQgc2VxdWVuY2UgcmVjb3JkZXJzIHdpdGgga2V5ZWQgaW5kZXhcbmV4cG9ydCB7IFNlcXVlbmNlUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9TZXF1ZW5jZVJlY29yZGVyLmpzJztcblxuLy8gUXVhbGl0eVJlY29yZGVyIOKAlCBwZXItc3RlcCBxdWFsaXR5IHNjb3Jpbmcgd2l0aCBiYWNrdHJhY2tpbmdcbmV4cG9ydCB0eXBlIHsgUXVhbGl0eUVudHJ5LCBRdWFsaXR5UmVjb3JkZXJPcHRpb25zLCBRdWFsaXR5U2NvcmluZ0ZuIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvUXVhbGl0eVJlY29yZGVyLmpzJztcbmV4cG9ydCB7IFF1YWxpdHlSZWNvcmRlciB9IGZyb20gJy4vbGliL3JlY29yZGVyL1F1YWxpdHlSZWNvcmRlci5qcyc7XG5cbi8vIHF1YWxpdHlUcmFjZSDigJQgUXVhbGl0eSBTdGFjayBUcmFjZSAoYmFja3RyYWNrIGZyb20gbG93LXNjb3Jpbmcgc3RlcHMpXG5leHBvcnQgdHlwZSB7IFF1YWxpdHlGcmFtZSwgUXVhbGl0eVN0YWNrVHJhY2UgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9xdWFsaXR5VHJhY2UuanMnO1xuZXhwb3J0IHsgZm9ybWF0UXVhbGl0eVRyYWNlLCBxdWFsaXR5VHJhY2UgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9xdWFsaXR5VHJhY2UuanMnO1xuIl19
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHJhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXFCRztBQUlILE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxzQkFBc0IsRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBRWxILHdEQUF3RDtBQUN4RCxPQUFPLEVBQUUsVUFBVSxFQUFFLFdBQVcsRUFBRSxjQUFjLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUl6RixPQUFPLEVBQUUsV0FBVyxFQUFFLGdCQUFnQixFQUFFLGlCQUFpQixFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFFN0YseURBQXlEO0FBQ3pELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUVoRSxvRkFBb0Y7QUFDcEYsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFVdEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFJeEYsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBSXBFLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxZQUFZLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogZm9vdHByaW50anMvdHJhY2Ug4oCUIEV4ZWN1dGlvbiB0cmFjaW5nLCBkZWJ1Z2dpbmcsIGFuZCBiYWNrdHJhY2tpbmcgdXRpbGl0aWVzLlxuICpcbiAqIFJ1bnRpbWUgc3RhZ2UgSURzLCBjb21taXQgbG9nIHF1ZXJpZXMsIGFuZCByZWNvcmRlciBiYXNlIGNsYXNzZXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IHBhcnNlUnVudGltZVN0YWdlSWQsIGZpbmRMYXN0V3JpdGVyLCBLZXllZFJlY29yZGVyLCBTZXF1ZW5jZVJlY29yZGVyIH0gZnJvbSAnZm9vdHByaW50anMvdHJhY2UnO1xuICpcbiAqIC8vIFBhcnNlIGEgcnVudGltZVN0YWdlSWRcbiAqIGNvbnN0IHsgc3RhZ2VJZCwgZXhlY3V0aW9uSW5kZXggfSA9IHBhcnNlUnVudGltZVN0YWdlSWQoJ2NhbGwtbGxtIzUnKTtcbiAqXG4gKiAvLyBCYWNrdHJhY2s6IHdobyB3cm90ZSAnc3lzdGVtUHJvbXB0JyBiZWZvcmUgc3RhZ2UgYXQgaWR4IDg/XG4gKiBjb25zdCB3cml0ZXIgPSBmaW5kTGFzdFdyaXRlcihjb21taXRMb2csICdzeXN0ZW1Qcm9tcHQnLCA4KTtcbiAqXG4gKiAvLyBCdWlsZCBhIGtleWVkIHJlY29yZGVyICgxOjEg4oCUIG9uZSBlbnRyeSBwZXIgc3RlcClcbiAqIGNsYXNzIE15UmVjb3JkZXIgZXh0ZW5kcyBLZXllZFJlY29yZGVyPE15RW50cnk+IHsgLi4uIH1cbiAqXG4gKiAvLyBCdWlsZCBhIHNlcXVlbmNlIHJlY29yZGVyICgxOk4g4oCUIG11bHRpcGxlIGVudHJpZXMgcGVyIHN0ZXAsIG9yZGVyaW5nIG1hdHRlcnMpXG4gKiBjbGFzcyBBdWRpdFJlY29yZGVyIGV4dGVuZHMgU2VxdWVuY2VSZWNvcmRlcjxBdWRpdEVudHJ5PiB7IC4uLiB9XG4gKiBgYGBcbiAqL1xuXG4vLyBSdW50aW1lIHN0YWdlIElEIOKAlCB1bmlxdWUgZXhlY3V0aW9uIHN0ZXAgaWRlbnRpZmllcnNcbmV4cG9ydCB0eXBlIHsgRXhlY3V0aW9uQ291bnRlciB9IGZyb20gJy4vbGliL2VuZ2luZS9ydW50aW1lU3RhZ2VJZC5qcyc7XG5leHBvcnQgeyBidWlsZFJ1bnRpbWVTdGFnZUlkLCBjcmVhdGVFeGVjdXRpb25Db3VudGVyLCBwYXJzZVJ1bnRpbWVTdGFnZUlkIH0gZnJvbSAnLi9saWIvZW5naW5lL3J1bnRpbWVTdGFnZUlkLmpzJztcblxuLy8gQ29tbWl0IGxvZyBxdWVyaWVzIOKAlCB0eXBlZCB1dGlsaXRpZXMgZm9yIGJhY2t0cmFja2luZ1xuZXhwb3J0IHsgZmluZENvbW1pdCwgZmluZENvbW1pdHMsIGZpbmRMYXN0V3JpdGVyIH0gZnJvbSAnLi9saWIvbWVtb3J5L2NvbW1pdExvZ1V0aWxzLmpzJztcblxuLy8gQ2F1c2FsIGNoYWluIOKAlCBiYWNrd2FyZCBwcm9ncmFtIHNsaWNpbmcgb24gY29tbWl0IGxvZyAoREFHKVxuZXhwb3J0IHR5cGUgeyBDYXVzYWxDaGFpbk9wdGlvbnMsIENhdXNhbE5vZGUsIEtleXNSZWFkTG9va3VwIH0gZnJvbSAnLi9saWIvbWVtb3J5L2JhY2t0cmFjay5qcyc7XG5leHBvcnQgeyBjYXVzYWxDaGFpbiwgZmxhdHRlbkNhdXNhbERBRywgZm9ybWF0Q2F1c2FsQ2hhaW4gfSBmcm9tICcuL2xpYi9tZW1vcnkvYmFja3RyYWNrLmpzJztcblxuLy8gS2V5ZWRSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMToxIE1hcC1iYXNlZCByZWNvcmRlcnNcbmV4cG9ydCB7IEtleWVkUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9LZXllZFJlY29yZGVyLmpzJztcblxuLy8gU2VxdWVuY2VSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMTpOIG9yZGVyZWQgc2VxdWVuY2UgcmVjb3JkZXJzIHdpdGgga2V5ZWQgaW5kZXhcbmV4cG9ydCB7IFNlcXVlbmNlUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9TZXF1ZW5jZVJlY29yZGVyLmpzJztcblxuLy8gVG9wb2xvZ3lSZWNvcmRlciDigJQgY29tcG9zaXRpb24gZ3JhcGggYWNjdW11bGF0b3IgKHN1YmZsb3dzICsgY29udHJvbC1mbG93IGVkZ2VzKVxuZXhwb3J0IHR5cGUge1xuICBUb3BvbG9neSxcbiAgVG9wb2xvZ3lFZGdlLFxuICBUb3BvbG9neUluY29taW5nS2luZCxcbiAgVG9wb2xvZ3lOb2RlLFxuICBUb3BvbG9neVJlY29yZGVyT3B0aW9ucyxcbn0gZnJvbSAnLi9saWIvcmVjb3JkZXIvVG9wb2xvZ3lSZWNvcmRlci5qcyc7XG5leHBvcnQgeyBUb3BvbG9neVJlY29yZGVyLCB0b3BvbG9neVJlY29yZGVyIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvVG9wb2xvZ3lSZWNvcmRlci5qcyc7XG5cbi8vIFF1YWxpdHlSZWNvcmRlciDigJQgcGVyLXN0ZXAgcXVhbGl0eSBzY29yaW5nIHdpdGggYmFja3RyYWNraW5nXG5leHBvcnQgdHlwZSB7IFF1YWxpdHlFbnRyeSwgUXVhbGl0eVJlY29yZGVyT3B0aW9ucywgUXVhbGl0eVNjb3JpbmdGbiB9IGZyb20gJy4vbGliL3JlY29yZGVyL1F1YWxpdHlSZWNvcmRlci5qcyc7XG5leHBvcnQgeyBRdWFsaXR5UmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9RdWFsaXR5UmVjb3JkZXIuanMnO1xuXG4vLyBxdWFsaXR5VHJhY2Ug4oCUIFF1YWxpdHkgU3RhY2sgVHJhY2UgKGJhY2t0cmFjayBmcm9tIGxvdy1zY29yaW5nIHN0ZXBzKVxuZXhwb3J0IHR5cGUgeyBRdWFsaXR5RnJhbWUsIFF1YWxpdHlTdGFja1RyYWNlIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvcXVhbGl0eVRyYWNlLmpzJztcbmV4cG9ydCB7IGZvcm1hdFF1YWxpdHlUcmFjZSwgcXVhbGl0eVRyYWNlIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvcXVhbGl0eVRyYWNlLmpzJztcbiJdfQ==
@@ -0,0 +1,277 @@
1
+ "use strict";
2
+ /**
3
+ * TopologyRecorder — composition graph built during traversal.
4
+ *
5
+ * The gap this fills:
6
+ * footprintjs fires atomic flow events (onSubflowEntry, onFork, onDecision,
7
+ * onLoop) but the accumulated *shape* of a run — who nests inside whom,
8
+ * which nodes are parallel siblings vs branches of a decision — is only
9
+ * visible post-run via `executor.getSnapshot()` tree-walking.
10
+ *
11
+ * Streaming consumers (live UIs, in-flight debuggers) see only the event
12
+ * stream. Every such consumer has to rebuild subflow-stack + fork-map +
13
+ * decision-tracker from scratch, usually slightly wrong in different ways.
14
+ *
15
+ * TopologyRecorder is the standard accumulator: one subscription to the
16
+ * three primitive channels, one live graph, queryable at any moment during
17
+ * or after a run.
18
+ *
19
+ * What it records — THREE node kinds for complete composition coverage:
20
+ * 1. 'subflow' — via onSubflowEntry (a mounted subflow boundary)
21
+ * 2. 'fork-branch' — via onFork (one node per child, synthesized)
22
+ * 3. 'decision-branch' — via onDecision (the chosen branch, synthesized)
23
+ *
24
+ * When a fork-branch or decision-branch target IS ALSO a subflow, the
25
+ * subsequent onSubflowEntry creates a subflow CHILD of the synthetic node.
26
+ * The layered shape preserves both "who branched" and "what the branch ran."
27
+ *
28
+ * Plain sequential stages are NOT nodes — that's StageContext's job.
29
+ * Topology is a graph of control-flow branching, not a full execution tree.
30
+ *
31
+ * Edges:
32
+ * One edge per traversal transition — `kind` matches the child's
33
+ * `incomingKind`. A consumer rendering "parallel columns" filters edges
34
+ * where `kind === 'fork-branch'` sharing the same `from`.
35
+ *
36
+ * @example
37
+ * ```typescript
38
+ * import { topologyRecorder } from 'footprintjs/trace';
39
+ *
40
+ * const topo = topologyRecorder();
41
+ * executor.attachCombinedRecorder(topo); // auto-routes to FlowRecorder channel
42
+ *
43
+ * await executor.run();
44
+ *
45
+ * const { nodes, edges, activeNodeId, rootId } = topo.getTopology();
46
+ * // Consumer queries:
47
+ * topo.getChildren('sf-parent'); // direct children (any kind)
48
+ * topo.getByKind('fork-branch'); // all parallel branches
49
+ * topo.getSubflowNodes(); // only mounted subflows
50
+ * ```
51
+ */
52
+ Object.defineProperty(exports, "__esModule", { value: true });
53
+ exports.TopologyRecorder = exports.topologyRecorder = void 0;
54
+ let _counter = 0;
55
+ /**
56
+ * Factory — matches the `narrative()` / `metrics()` style.
57
+ */
58
+ function topologyRecorder(options = {}) {
59
+ return new TopologyRecorder(options);
60
+ }
61
+ exports.topologyRecorder = topologyRecorder;
62
+ /**
63
+ * Stateful accumulator that watches FlowRecorder events and maintains a live
64
+ * composition graph. Attach via `executor.attachCombinedRecorder(recorder)` —
65
+ * footprintjs detects the `FlowRecorder` method shape and routes events.
66
+ */
67
+ class TopologyRecorder {
68
+ constructor(options = {}) {
69
+ var _a;
70
+ this.nodesById = new Map();
71
+ this.nodeOrder = [];
72
+ this.edges = [];
73
+ /** Stack of active SUBFLOW node ids. Fork/decision-branch nodes never push. */
74
+ this.subflowStack = [];
75
+ /** Map of childName → pending fork-branch synthetic node, consumed by
76
+ * the next matching `onSubflowEntry`. */
77
+ this.pendingForkByName = new Map();
78
+ this.id = (_a = options.id) !== null && _a !== void 0 ? _a : `topology-${++_counter}`;
79
+ }
80
+ // ── FlowRecorder hooks ────────────────────────────────────────────────
81
+ onSubflowEntry(event) {
82
+ var _a, _b;
83
+ const subflowId = event.subflowId;
84
+ if (!subflowId)
85
+ return; // Need a stable id to track.
86
+ const enteredAt = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
87
+ // Determine the parent: prefer a pending fork/decision match by name,
88
+ // otherwise the current top-of-subflow-stack.
89
+ let parentId;
90
+ let incomingKind;
91
+ const pendingFork = this.pendingForkByName.get(event.name);
92
+ if (pendingFork) {
93
+ parentId = pendingFork.nodeId;
94
+ incomingKind = 'next'; // Child OF a fork-branch node; the fork semantic
95
+ // is captured by the fork-branch's own incomingKind.
96
+ this.pendingForkByName.delete(event.name);
97
+ }
98
+ else if (this.pendingDecision && this.pendingDecision.name === event.name) {
99
+ parentId = this.pendingDecision.nodeId;
100
+ incomingKind = 'next';
101
+ this.pendingDecision = undefined;
102
+ }
103
+ else {
104
+ parentId = this.subflowStack[this.subflowStack.length - 1];
105
+ incomingKind = parentId ? 'next' : 'root';
106
+ }
107
+ // Disambiguate re-entry (e.g., loop body re-enters the same subflow).
108
+ let nodeId = subflowId;
109
+ if (this.nodesById.has(nodeId)) {
110
+ let n = 1;
111
+ while (this.nodesById.has(`${subflowId}#${n}`))
112
+ n++;
113
+ nodeId = `${subflowId}#${n}`;
114
+ }
115
+ const depth = parentId ? this.nodesById.get(parentId).depth + 1 : 0;
116
+ const metadata = event.description ? { description: event.description } : undefined;
117
+ const node = {
118
+ id: nodeId,
119
+ kind: 'subflow',
120
+ name: event.name,
121
+ parentId,
122
+ depth,
123
+ incomingKind,
124
+ enteredAt,
125
+ metadata,
126
+ };
127
+ this.nodesById.set(nodeId, node);
128
+ this.nodeOrder.push(nodeId);
129
+ if (parentId && incomingKind !== 'root') {
130
+ this.edges.push({
131
+ from: parentId,
132
+ to: nodeId,
133
+ kind: incomingKind,
134
+ at: enteredAt,
135
+ });
136
+ }
137
+ this.subflowStack.push(nodeId);
138
+ }
139
+ onSubflowExit(event) {
140
+ var _a, _b;
141
+ const nodeId = this.subflowStack.pop();
142
+ if (!nodeId)
143
+ return;
144
+ const node = this.nodesById.get(nodeId);
145
+ if (node) {
146
+ node.exitedAt = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
147
+ }
148
+ // A subflow exit implies no more children are pending for this scope —
149
+ // clear stale pending state that belonged to the scope we just left.
150
+ this.pendingForkByName.clear();
151
+ this.pendingDecision = undefined;
152
+ }
153
+ onFork(event) {
154
+ var _a, _b;
155
+ const activeId = this.subflowStack[this.subflowStack.length - 1];
156
+ const at = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
157
+ const depth = activeId ? this.nodesById.get(activeId).depth + 1 : 0;
158
+ // Reset any prior pending fork state — a new fork starts fresh.
159
+ this.pendingForkByName.clear();
160
+ event.children.forEach((childName, i) => {
161
+ const nodeId = `fork-${at || event.parent}-${i}-${childName}`;
162
+ const node = {
163
+ id: nodeId,
164
+ kind: 'fork-branch',
165
+ name: childName,
166
+ parentId: activeId,
167
+ depth,
168
+ incomingKind: 'fork-branch',
169
+ enteredAt: at,
170
+ metadata: { forkParent: event.parent },
171
+ };
172
+ this.nodesById.set(nodeId, node);
173
+ this.nodeOrder.push(nodeId);
174
+ if (activeId) {
175
+ this.edges.push({ from: activeId, to: nodeId, kind: 'fork-branch', at });
176
+ }
177
+ this.pendingForkByName.set(childName, { nodeId, at });
178
+ });
179
+ }
180
+ onDecision(event) {
181
+ var _a, _b;
182
+ const activeId = this.subflowStack[this.subflowStack.length - 1];
183
+ const at = (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '';
184
+ const depth = activeId ? this.nodesById.get(activeId).depth + 1 : 0;
185
+ // A new decision supersedes any prior unresolved pending one.
186
+ this.pendingDecision = undefined;
187
+ const nodeId = `decision-${at || event.decider}-${event.chosen}`;
188
+ const metadata = { decider: event.decider };
189
+ if (event.rationale)
190
+ metadata.rationale = event.rationale;
191
+ if (event.description)
192
+ metadata.description = event.description;
193
+ const node = {
194
+ id: nodeId,
195
+ kind: 'decision-branch',
196
+ name: event.chosen,
197
+ parentId: activeId,
198
+ depth,
199
+ incomingKind: 'decision-branch',
200
+ enteredAt: at,
201
+ metadata,
202
+ };
203
+ this.nodesById.set(nodeId, node);
204
+ this.nodeOrder.push(nodeId);
205
+ if (activeId) {
206
+ this.edges.push({ from: activeId, to: nodeId, kind: 'decision-branch', at });
207
+ }
208
+ this.pendingDecision = { name: event.chosen, nodeId, at };
209
+ }
210
+ onLoop(event) {
211
+ var _a, _b;
212
+ // loopTo jumps back inside the CURRENT subflow. Record a self-edge on the
213
+ // active subflow — synthetic fork/decision nodes don't participate in loops.
214
+ const activeId = this.subflowStack[this.subflowStack.length - 1];
215
+ if (!activeId)
216
+ return;
217
+ this.edges.push({
218
+ from: activeId,
219
+ to: activeId,
220
+ kind: 'loop-iteration',
221
+ at: (_b = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.runtimeStageId) !== null && _b !== void 0 ? _b : '',
222
+ });
223
+ }
224
+ /** Called by the executor before each `run()` — resets all state. */
225
+ clear() {
226
+ this.nodesById.clear();
227
+ this.nodeOrder.length = 0;
228
+ this.edges.length = 0;
229
+ this.subflowStack.length = 0;
230
+ this.pendingForkByName.clear();
231
+ this.pendingDecision = undefined;
232
+ }
233
+ // ── Query API ─────────────────────────────────────────────────────────
234
+ /** Live snapshot of the composition graph. Safe during or after a run. */
235
+ getTopology() {
236
+ var _a, _b;
237
+ const nodes = this.nodeOrder.map((id) => this.nodesById.get(id));
238
+ return {
239
+ nodes,
240
+ edges: [...this.edges],
241
+ activeNodeId: (_a = this.subflowStack[this.subflowStack.length - 1]) !== null && _a !== void 0 ? _a : null,
242
+ rootId: (_b = this.nodeOrder[0]) !== null && _b !== void 0 ? _b : null,
243
+ };
244
+ }
245
+ /** Direct children of a node — insertion-ordered. */
246
+ getChildren(nodeId) {
247
+ return this.nodeOrder.map((id) => this.nodesById.get(id)).filter((n) => n.parentId === nodeId);
248
+ }
249
+ /** All nodes of a given kind. */
250
+ getByKind(kind) {
251
+ return this.nodeOrder.map((id) => this.nodesById.get(id)).filter((n) => n.kind === kind);
252
+ }
253
+ /** All mounted subflow nodes. Convenience for agent-centric views. */
254
+ getSubflowNodes() {
255
+ return this.getByKind('subflow');
256
+ }
257
+ /** All fork-branch nodes sharing the same parent as `nodeId` — i.e.,
258
+ * parallel siblings of a parallel branch. Empty if `nodeId` isn't a
259
+ * fork-branch or has no parent. */
260
+ getParallelSiblings(nodeId) {
261
+ const node = this.nodesById.get(nodeId);
262
+ if (!node || node.kind !== 'fork-branch' || !node.parentId)
263
+ return [];
264
+ return this.getChildren(node.parentId).filter((n) => n.kind === 'fork-branch');
265
+ }
266
+ /** Emit a snapshot bundle for inclusion in `executor.getSnapshot()`. */
267
+ toSnapshot() {
268
+ return {
269
+ name: 'Topology',
270
+ description: 'Composition graph: subflow boundaries, fork branches, decision branches',
271
+ preferredOperation: 'translate',
272
+ data: this.getTopology(),
273
+ };
274
+ }
275
+ }
276
+ exports.TopologyRecorder = TopologyRecorder;
277
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TopologyRecorder.js","sourceRoot":"","sources":["../../../src/lib/recorder/TopologyRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * TopologyRecorder — composition graph built during traversal.\n *\n * The gap this fills:\n *   footprintjs fires atomic flow events (onSubflowEntry, onFork, onDecision,\n *   onLoop) but the accumulated *shape* of a run — who nests inside whom,\n *   which nodes are parallel siblings vs branches of a decision — is only\n *   visible post-run via `executor.getSnapshot()` tree-walking.\n *\n *   Streaming consumers (live UIs, in-flight debuggers) see only the event\n *   stream. Every such consumer has to rebuild subflow-stack + fork-map +\n *   decision-tracker from scratch, usually slightly wrong in different ways.\n *\n *   TopologyRecorder is the standard accumulator: one subscription to the\n *   three primitive channels, one live graph, queryable at any moment during\n *   or after a run.\n *\n * What it records — THREE node kinds for complete composition coverage:\n *   1. 'subflow'          — via onSubflowEntry (a mounted subflow boundary)\n *   2. 'fork-branch'      — via onFork (one node per child, synthesized)\n *   3. 'decision-branch'  — via onDecision (the chosen branch, synthesized)\n *\n *   When a fork-branch or decision-branch target IS ALSO a subflow, the\n *   subsequent onSubflowEntry creates a subflow CHILD of the synthetic node.\n *   The layered shape preserves both \"who branched\" and \"what the branch ran.\"\n *\n *   Plain sequential stages are NOT nodes — that's StageContext's job.\n *   Topology is a graph of control-flow branching, not a full execution tree.\n *\n * Edges:\n *   One edge per traversal transition — `kind` matches the child's\n *   `incomingKind`. A consumer rendering \"parallel columns\" filters edges\n *   where `kind === 'fork-branch'` sharing the same `from`.\n *\n * @example\n * ```typescript\n * import { topologyRecorder } from 'footprintjs/trace';\n *\n * const topo = topologyRecorder();\n * executor.attachCombinedRecorder(topo);  // auto-routes to FlowRecorder channel\n *\n * await executor.run();\n *\n * const { nodes, edges, activeNodeId, rootId } = topo.getTopology();\n * // Consumer queries:\n * topo.getChildren('sf-parent');              // direct children (any kind)\n * topo.getByKind('fork-branch');              // all parallel branches\n * topo.getSubflowNodes();                     // only mounted subflows\n * ```\n */\n\nimport type {\n  FlowDecisionEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSubflowEvent,\n} from '../engine/narrative/types.js';\n\n/** The kind of composition unit a node represents. */\nexport type TopologyNodeKind = 'subflow' | 'fork-branch' | 'decision-branch';\n\n/** How the traversal reached this node — drives consumer layout decisions. */\nexport type TopologyIncomingKind = 'root' | 'next' | 'fork-branch' | 'decision-branch' | 'loop-iteration';\n\n/** A composition-significant point in the graph. */\nexport interface TopologyNode {\n  /** Unique id. Subflows use their subflowId (with `#n` suffix on re-entry).\n   *  Synthetic nodes (fork-branch / decision-branch) use\n   *  `fork-${runtimeStageId}-${i}` / `decision-${runtimeStageId}` form. */\n  readonly id: string;\n  /** What this node represents. */\n  readonly kind: TopologyNodeKind;\n  /** Display name. For subflows: `FlowSubflowEvent.name`. For fork-branches:\n   *  the child name from `FlowForkEvent.children`. For decision-branches:\n   *  the chosen name from `FlowDecisionEvent.chosen`. */\n  readonly name: string;\n  /** Parent node id. Undefined when this node sits at the run's top level. */\n  readonly parentId?: string;\n  /** Depth in the topology tree (0 = top-level). */\n  readonly depth: number;\n  /** How the traversal reached this node. */\n  readonly incomingKind: TopologyIncomingKind;\n  /** runtimeStageId at the moment the node was created. */\n  readonly enteredAt: string;\n  /** runtimeStageId when the corresponding subflow exited. Only meaningful\n   *  for kind='subflow'; fork/decision-branch nodes are instantaneous. */\n  exitedAt?: string;\n  /** Kind-specific extras: forkParent, decider, rationale, description. */\n  readonly metadata?: Readonly<Record<string, unknown>>;\n}\n\n/** A traversal transition between two nodes. */\nexport interface TopologyEdge {\n  readonly from: string;\n  readonly to: string;\n  readonly kind: Exclude<TopologyIncomingKind, 'root'>;\n  readonly at: string;\n}\n\n/** Snapshot of the composition graph. */\nexport interface Topology {\n  readonly nodes: ReadonlyArray<TopologyNode>;\n  readonly edges: ReadonlyArray<TopologyEdge>;\n  /** Currently-active subflow (top of the subflow stack). Fork-branch and\n   *  decision-branch nodes are instantaneous — they don't affect activeNodeId. */\n  readonly activeNodeId: string | null;\n  /** First node inserted. null before any composition event fires. */\n  readonly rootId: string | null;\n}\n\nexport interface TopologyRecorderOptions {\n  /** Recorder id. Defaults to `topology-N` (auto-incremented). */\n  id?: string;\n}\n\n// Correlation state: maps a pending fork/decision child name to its synthetic\n// node id, so a subsequent onSubflowEntry matching that name can be nested\n// under the synthetic node (rather than creating a peer).\ninterface PendingChild {\n  nodeId: string;\n  at: string;\n}\n\nlet _counter = 0;\n\n/**\n * Factory — matches the `narrative()` / `metrics()` style.\n */\nexport function topologyRecorder(options: TopologyRecorderOptions = {}): TopologyRecorder {\n  return new TopologyRecorder(options);\n}\n\n/**\n * Stateful accumulator that watches FlowRecorder events and maintains a live\n * composition graph. Attach via `executor.attachCombinedRecorder(recorder)` —\n * footprintjs detects the `FlowRecorder` method shape and routes events.\n */\nexport class TopologyRecorder implements FlowRecorder {\n  readonly id: string;\n\n  private readonly nodesById = new Map<string, TopologyNode>();\n  private readonly nodeOrder: string[] = [];\n  private readonly edges: TopologyEdge[] = [];\n  /** Stack of active SUBFLOW node ids. Fork/decision-branch nodes never push. */\n  private readonly subflowStack: string[] = [];\n\n  /** Map of childName → pending fork-branch synthetic node, consumed by\n   *  the next matching `onSubflowEntry`. */\n  private readonly pendingForkByName = new Map<string, PendingChild>();\n  /** Pending decision-branch synthetic node, consumed by a matching entry. */\n  private pendingDecision?: { name: string } & PendingChild;\n\n  constructor(options: TopologyRecorderOptions = {}) {\n    this.id = options.id ?? `topology-${++_counter}`;\n  }\n\n  // ── FlowRecorder hooks ────────────────────────────────────────────────\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    const subflowId = event.subflowId;\n    if (!subflowId) return; // Need a stable id to track.\n\n    const enteredAt = event.traversalContext?.runtimeStageId ?? '';\n\n    // Determine the parent: prefer a pending fork/decision match by name,\n    // otherwise the current top-of-subflow-stack.\n    let parentId: string | undefined;\n    let incomingKind: TopologyIncomingKind;\n\n    const pendingFork = this.pendingForkByName.get(event.name);\n    if (pendingFork) {\n      parentId = pendingFork.nodeId;\n      incomingKind = 'next'; // Child OF a fork-branch node; the fork semantic\n      // is captured by the fork-branch's own incomingKind.\n      this.pendingForkByName.delete(event.name);\n    } else if (this.pendingDecision && this.pendingDecision.name === event.name) {\n      parentId = this.pendingDecision.nodeId;\n      incomingKind = 'next';\n      this.pendingDecision = undefined;\n    } else {\n      parentId = this.subflowStack[this.subflowStack.length - 1];\n      incomingKind = parentId ? 'next' : 'root';\n    }\n\n    // Disambiguate re-entry (e.g., loop body re-enters the same subflow).\n    let nodeId = subflowId;\n    if (this.nodesById.has(nodeId)) {\n      let n = 1;\n      while (this.nodesById.has(`${subflowId}#${n}`)) n++;\n      nodeId = `${subflowId}#${n}`;\n    }\n\n    const depth = parentId ? this.nodesById.get(parentId)!.depth + 1 : 0;\n    const metadata = event.description ? { description: event.description } : undefined;\n\n    const node: TopologyNode = {\n      id: nodeId,\n      kind: 'subflow',\n      name: event.name,\n      parentId,\n      depth,\n      incomingKind,\n      enteredAt,\n      metadata,\n    };\n    this.nodesById.set(nodeId, node);\n    this.nodeOrder.push(nodeId);\n\n    if (parentId && incomingKind !== 'root') {\n      this.edges.push({\n        from: parentId,\n        to: nodeId,\n        kind: incomingKind,\n        at: enteredAt,\n      });\n    }\n\n    this.subflowStack.push(nodeId);\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    const nodeId = this.subflowStack.pop();\n    if (!nodeId) return;\n    const node = this.nodesById.get(nodeId);\n    if (node) {\n      node.exitedAt = event.traversalContext?.runtimeStageId ?? '';\n    }\n    // A subflow exit implies no more children are pending for this scope —\n    // clear stale pending state that belonged to the scope we just left.\n    this.pendingForkByName.clear();\n    this.pendingDecision = undefined;\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const activeId = this.subflowStack[this.subflowStack.length - 1];\n    const at = event.traversalContext?.runtimeStageId ?? '';\n    const depth = activeId ? this.nodesById.get(activeId)!.depth + 1 : 0;\n\n    // Reset any prior pending fork state — a new fork starts fresh.\n    this.pendingForkByName.clear();\n\n    event.children.forEach((childName, i) => {\n      const nodeId = `fork-${at || event.parent}-${i}-${childName}`;\n      const node: TopologyNode = {\n        id: nodeId,\n        kind: 'fork-branch',\n        name: childName,\n        parentId: activeId,\n        depth,\n        incomingKind: 'fork-branch',\n        enteredAt: at,\n        metadata: { forkParent: event.parent },\n      };\n      this.nodesById.set(nodeId, node);\n      this.nodeOrder.push(nodeId);\n      if (activeId) {\n        this.edges.push({ from: activeId, to: nodeId, kind: 'fork-branch', at });\n      }\n      this.pendingForkByName.set(childName, { nodeId, at });\n    });\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const activeId = this.subflowStack[this.subflowStack.length - 1];\n    const at = event.traversalContext?.runtimeStageId ?? '';\n    const depth = activeId ? this.nodesById.get(activeId)!.depth + 1 : 0;\n\n    // A new decision supersedes any prior unresolved pending one.\n    this.pendingDecision = undefined;\n\n    const nodeId = `decision-${at || event.decider}-${event.chosen}`;\n    const metadata: Record<string, unknown> = { decider: event.decider };\n    if (event.rationale) metadata.rationale = event.rationale;\n    if (event.description) metadata.description = event.description;\n\n    const node: TopologyNode = {\n      id: nodeId,\n      kind: 'decision-branch',\n      name: event.chosen,\n      parentId: activeId,\n      depth,\n      incomingKind: 'decision-branch',\n      enteredAt: at,\n      metadata,\n    };\n    this.nodesById.set(nodeId, node);\n    this.nodeOrder.push(nodeId);\n    if (activeId) {\n      this.edges.push({ from: activeId, to: nodeId, kind: 'decision-branch', at });\n    }\n    this.pendingDecision = { name: event.chosen, nodeId, at };\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    // loopTo jumps back inside the CURRENT subflow. Record a self-edge on the\n    // active subflow — synthetic fork/decision nodes don't participate in loops.\n    const activeId = this.subflowStack[this.subflowStack.length - 1];\n    if (!activeId) return;\n    this.edges.push({\n      from: activeId,\n      to: activeId,\n      kind: 'loop-iteration',\n      at: event.traversalContext?.runtimeStageId ?? '',\n    });\n  }\n\n  /** Called by the executor before each `run()` — resets all state. */\n  clear(): void {\n    this.nodesById.clear();\n    this.nodeOrder.length = 0;\n    this.edges.length = 0;\n    this.subflowStack.length = 0;\n    this.pendingForkByName.clear();\n    this.pendingDecision = undefined;\n  }\n\n  // ── Query API ─────────────────────────────────────────────────────────\n\n  /** Live snapshot of the composition graph. Safe during or after a run. */\n  getTopology(): Topology {\n    const nodes = this.nodeOrder.map((id) => this.nodesById.get(id)!);\n    return {\n      nodes,\n      edges: [...this.edges],\n      activeNodeId: this.subflowStack[this.subflowStack.length - 1] ?? null,\n      rootId: this.nodeOrder[0] ?? null,\n    };\n  }\n\n  /** Direct children of a node — insertion-ordered. */\n  getChildren(nodeId: string): TopologyNode[] {\n    return this.nodeOrder.map((id) => this.nodesById.get(id)!).filter((n) => n.parentId === nodeId);\n  }\n\n  /** All nodes of a given kind. */\n  getByKind(kind: TopologyNodeKind): TopologyNode[] {\n    return this.nodeOrder.map((id) => this.nodesById.get(id)!).filter((n) => n.kind === kind);\n  }\n\n  /** All mounted subflow nodes. Convenience for agent-centric views. */\n  getSubflowNodes(): TopologyNode[] {\n    return this.getByKind('subflow');\n  }\n\n  /** All fork-branch nodes sharing the same parent as `nodeId` — i.e.,\n   *  parallel siblings of a parallel branch. Empty if `nodeId` isn't a\n   *  fork-branch or has no parent. */\n  getParallelSiblings(nodeId: string): TopologyNode[] {\n    const node = this.nodesById.get(nodeId);\n    if (!node || node.kind !== 'fork-branch' || !node.parentId) return [];\n    return this.getChildren(node.parentId).filter((n) => n.kind === 'fork-branch');\n  }\n\n  /** Emit a snapshot bundle for inclusion in `executor.getSnapshot()`. */\n  toSnapshot() {\n    return {\n      name: 'Topology',\n      description: 'Composition graph: subflow boundaries, fork branches, decision branches',\n      preferredOperation: 'translate' as const,\n      data: this.getTopology(),\n    };\n  }\n}\n"]}
package/dist/trace.js CHANGED
@@ -22,7 +22,7 @@
22
22
  * ```
23
23
  */
24
24
  Object.defineProperty(exports, "__esModule", { value: true });
25
- exports.qualityTrace = exports.formatQualityTrace = exports.QualityRecorder = exports.SequenceRecorder = exports.KeyedRecorder = exports.formatCausalChain = exports.flattenCausalDAG = exports.causalChain = exports.findLastWriter = exports.findCommits = exports.findCommit = exports.parseRuntimeStageId = exports.createExecutionCounter = exports.buildRuntimeStageId = void 0;
25
+ exports.qualityTrace = exports.formatQualityTrace = exports.QualityRecorder = exports.topologyRecorder = exports.TopologyRecorder = exports.SequenceRecorder = exports.KeyedRecorder = exports.formatCausalChain = exports.flattenCausalDAG = exports.causalChain = exports.findLastWriter = exports.findCommits = exports.findCommit = exports.parseRuntimeStageId = exports.createExecutionCounter = exports.buildRuntimeStageId = void 0;
26
26
  var runtimeStageId_js_1 = require("./lib/engine/runtimeStageId.js");
27
27
  Object.defineProperty(exports, "buildRuntimeStageId", { enumerable: true, get: function () { return runtimeStageId_js_1.buildRuntimeStageId; } });
28
28
  Object.defineProperty(exports, "createExecutionCounter", { enumerable: true, get: function () { return runtimeStageId_js_1.createExecutionCounter; } });
@@ -42,9 +42,12 @@ Object.defineProperty(exports, "KeyedRecorder", { enumerable: true, get: functio
42
42
  // SequenceRecorder — base class for 1:N ordered sequence recorders with keyed index
43
43
  var SequenceRecorder_js_1 = require("./lib/recorder/SequenceRecorder.js");
44
44
  Object.defineProperty(exports, "SequenceRecorder", { enumerable: true, get: function () { return SequenceRecorder_js_1.SequenceRecorder; } });
45
+ var TopologyRecorder_js_1 = require("./lib/recorder/TopologyRecorder.js");
46
+ Object.defineProperty(exports, "TopologyRecorder", { enumerable: true, get: function () { return TopologyRecorder_js_1.TopologyRecorder; } });
47
+ Object.defineProperty(exports, "topologyRecorder", { enumerable: true, get: function () { return TopologyRecorder_js_1.topologyRecorder; } });
45
48
  var QualityRecorder_js_1 = require("./lib/recorder/QualityRecorder.js");
46
49
  Object.defineProperty(exports, "QualityRecorder", { enumerable: true, get: function () { return QualityRecorder_js_1.QualityRecorder; } });
47
50
  var qualityTrace_js_1 = require("./lib/recorder/qualityTrace.js");
48
51
  Object.defineProperty(exports, "formatQualityTrace", { enumerable: true, get: function () { return qualityTrace_js_1.formatQualityTrace; } });
49
52
  Object.defineProperty(exports, "qualityTrace", { enumerable: true, get: function () { return qualityTrace_js_1.qualityTrace; } });
50
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHJhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FxQkc7OztBQUlILG9FQUFrSDtBQUF6Ryx3SEFBQSxtQkFBbUIsT0FBQTtBQUFFLDJIQUFBLHNCQUFzQixPQUFBO0FBQUUsd0hBQUEsbUJBQW1CLE9BQUE7QUFFekUsd0RBQXdEO0FBQ3hELG9FQUF5RjtBQUFoRiwrR0FBQSxVQUFVLE9BQUE7QUFBRSxnSEFBQSxXQUFXLE9BQUE7QUFBRSxtSEFBQSxjQUFjLE9BQUE7QUFJaEQsMERBQTZGO0FBQXBGLDJHQUFBLFdBQVcsT0FBQTtBQUFFLGdIQUFBLGdCQUFnQixPQUFBO0FBQUUsaUhBQUEsaUJBQWlCLE9BQUE7QUFFekQseURBQXlEO0FBQ3pELG9FQUFnRTtBQUF2RCxpSEFBQSxhQUFhLE9BQUE7QUFFdEIsb0ZBQW9GO0FBQ3BGLDBFQUFzRTtBQUE3RCx1SEFBQSxnQkFBZ0IsT0FBQTtBQUl6Qix3RUFBb0U7QUFBM0QscUhBQUEsZUFBZSxPQUFBO0FBSXhCLGtFQUFrRjtBQUF6RSxxSEFBQSxrQkFBa0IsT0FBQTtBQUFFLCtHQUFBLFlBQVksT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogZm9vdHByaW50anMvdHJhY2Ug4oCUIEV4ZWN1dGlvbiB0cmFjaW5nLCBkZWJ1Z2dpbmcsIGFuZCBiYWNrdHJhY2tpbmcgdXRpbGl0aWVzLlxuICpcbiAqIFJ1bnRpbWUgc3RhZ2UgSURzLCBjb21taXQgbG9nIHF1ZXJpZXMsIGFuZCByZWNvcmRlciBiYXNlIGNsYXNzZXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IHBhcnNlUnVudGltZVN0YWdlSWQsIGZpbmRMYXN0V3JpdGVyLCBLZXllZFJlY29yZGVyLCBTZXF1ZW5jZVJlY29yZGVyIH0gZnJvbSAnZm9vdHByaW50anMvdHJhY2UnO1xuICpcbiAqIC8vIFBhcnNlIGEgcnVudGltZVN0YWdlSWRcbiAqIGNvbnN0IHsgc3RhZ2VJZCwgZXhlY3V0aW9uSW5kZXggfSA9IHBhcnNlUnVudGltZVN0YWdlSWQoJ2NhbGwtbGxtIzUnKTtcbiAqXG4gKiAvLyBCYWNrdHJhY2s6IHdobyB3cm90ZSAnc3lzdGVtUHJvbXB0JyBiZWZvcmUgc3RhZ2UgYXQgaWR4IDg/XG4gKiBjb25zdCB3cml0ZXIgPSBmaW5kTGFzdFdyaXRlcihjb21taXRMb2csICdzeXN0ZW1Qcm9tcHQnLCA4KTtcbiAqXG4gKiAvLyBCdWlsZCBhIGtleWVkIHJlY29yZGVyICgxOjEg4oCUIG9uZSBlbnRyeSBwZXIgc3RlcClcbiAqIGNsYXNzIE15UmVjb3JkZXIgZXh0ZW5kcyBLZXllZFJlY29yZGVyPE15RW50cnk+IHsgLi4uIH1cbiAqXG4gKiAvLyBCdWlsZCBhIHNlcXVlbmNlIHJlY29yZGVyICgxOk4g4oCUIG11bHRpcGxlIGVudHJpZXMgcGVyIHN0ZXAsIG9yZGVyaW5nIG1hdHRlcnMpXG4gKiBjbGFzcyBBdWRpdFJlY29yZGVyIGV4dGVuZHMgU2VxdWVuY2VSZWNvcmRlcjxBdWRpdEVudHJ5PiB7IC4uLiB9XG4gKiBgYGBcbiAqL1xuXG4vLyBSdW50aW1lIHN0YWdlIElEIOKAlCB1bmlxdWUgZXhlY3V0aW9uIHN0ZXAgaWRlbnRpZmllcnNcbmV4cG9ydCB0eXBlIHsgRXhlY3V0aW9uQ291bnRlciB9IGZyb20gJy4vbGliL2VuZ2luZS9ydW50aW1lU3RhZ2VJZC5qcyc7XG5leHBvcnQgeyBidWlsZFJ1bnRpbWVTdGFnZUlkLCBjcmVhdGVFeGVjdXRpb25Db3VudGVyLCBwYXJzZVJ1bnRpbWVTdGFnZUlkIH0gZnJvbSAnLi9saWIvZW5naW5lL3J1bnRpbWVTdGFnZUlkLmpzJztcblxuLy8gQ29tbWl0IGxvZyBxdWVyaWVzIOKAlCB0eXBlZCB1dGlsaXRpZXMgZm9yIGJhY2t0cmFja2luZ1xuZXhwb3J0IHsgZmluZENvbW1pdCwgZmluZENvbW1pdHMsIGZpbmRMYXN0V3JpdGVyIH0gZnJvbSAnLi9saWIvbWVtb3J5L2NvbW1pdExvZ1V0aWxzLmpzJztcblxuLy8gQ2F1c2FsIGNoYWluIOKAlCBiYWNrd2FyZCBwcm9ncmFtIHNsaWNpbmcgb24gY29tbWl0IGxvZyAoREFHKVxuZXhwb3J0IHR5cGUgeyBDYXVzYWxDaGFpbk9wdGlvbnMsIENhdXNhbE5vZGUsIEtleXNSZWFkTG9va3VwIH0gZnJvbSAnLi9saWIvbWVtb3J5L2JhY2t0cmFjay5qcyc7XG5leHBvcnQgeyBjYXVzYWxDaGFpbiwgZmxhdHRlbkNhdXNhbERBRywgZm9ybWF0Q2F1c2FsQ2hhaW4gfSBmcm9tICcuL2xpYi9tZW1vcnkvYmFja3RyYWNrLmpzJztcblxuLy8gS2V5ZWRSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMToxIE1hcC1iYXNlZCByZWNvcmRlcnNcbmV4cG9ydCB7IEtleWVkUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9LZXllZFJlY29yZGVyLmpzJztcblxuLy8gU2VxdWVuY2VSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMTpOIG9yZGVyZWQgc2VxdWVuY2UgcmVjb3JkZXJzIHdpdGgga2V5ZWQgaW5kZXhcbmV4cG9ydCB7IFNlcXVlbmNlUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9TZXF1ZW5jZVJlY29yZGVyLmpzJztcblxuLy8gUXVhbGl0eVJlY29yZGVyIOKAlCBwZXItc3RlcCBxdWFsaXR5IHNjb3Jpbmcgd2l0aCBiYWNrdHJhY2tpbmdcbmV4cG9ydCB0eXBlIHsgUXVhbGl0eUVudHJ5LCBRdWFsaXR5UmVjb3JkZXJPcHRpb25zLCBRdWFsaXR5U2NvcmluZ0ZuIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvUXVhbGl0eVJlY29yZGVyLmpzJztcbmV4cG9ydCB7IFF1YWxpdHlSZWNvcmRlciB9IGZyb20gJy4vbGliL3JlY29yZGVyL1F1YWxpdHlSZWNvcmRlci5qcyc7XG5cbi8vIHF1YWxpdHlUcmFjZSDigJQgUXVhbGl0eSBTdGFjayBUcmFjZSAoYmFja3RyYWNrIGZyb20gbG93LXNjb3Jpbmcgc3RlcHMpXG5leHBvcnQgdHlwZSB7IFF1YWxpdHlGcmFtZSwgUXVhbGl0eVN0YWNrVHJhY2UgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9xdWFsaXR5VHJhY2UuanMnO1xuZXhwb3J0IHsgZm9ybWF0UXVhbGl0eVRyYWNlLCBxdWFsaXR5VHJhY2UgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9xdWFsaXR5VHJhY2UuanMnO1xuIl19
53
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHJhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FxQkc7OztBQUlILG9FQUFrSDtBQUF6Ryx3SEFBQSxtQkFBbUIsT0FBQTtBQUFFLDJIQUFBLHNCQUFzQixPQUFBO0FBQUUsd0hBQUEsbUJBQW1CLE9BQUE7QUFFekUsd0RBQXdEO0FBQ3hELG9FQUF5RjtBQUFoRiwrR0FBQSxVQUFVLE9BQUE7QUFBRSxnSEFBQSxXQUFXLE9BQUE7QUFBRSxtSEFBQSxjQUFjLE9BQUE7QUFJaEQsMERBQTZGO0FBQXBGLDJHQUFBLFdBQVcsT0FBQTtBQUFFLGdIQUFBLGdCQUFnQixPQUFBO0FBQUUsaUhBQUEsaUJBQWlCLE9BQUE7QUFFekQseURBQXlEO0FBQ3pELG9FQUFnRTtBQUF2RCxpSEFBQSxhQUFhLE9BQUE7QUFFdEIsb0ZBQW9GO0FBQ3BGLDBFQUFzRTtBQUE3RCx1SEFBQSxnQkFBZ0IsT0FBQTtBQVV6QiwwRUFBd0Y7QUFBL0UsdUhBQUEsZ0JBQWdCLE9BQUE7QUFBRSx1SEFBQSxnQkFBZ0IsT0FBQTtBQUkzQyx3RUFBb0U7QUFBM0QscUhBQUEsZUFBZSxPQUFBO0FBSXhCLGtFQUFrRjtBQUF6RSxxSEFBQSxrQkFBa0IsT0FBQTtBQUFFLCtHQUFBLFlBQVksT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogZm9vdHByaW50anMvdHJhY2Ug4oCUIEV4ZWN1dGlvbiB0cmFjaW5nLCBkZWJ1Z2dpbmcsIGFuZCBiYWNrdHJhY2tpbmcgdXRpbGl0aWVzLlxuICpcbiAqIFJ1bnRpbWUgc3RhZ2UgSURzLCBjb21taXQgbG9nIHF1ZXJpZXMsIGFuZCByZWNvcmRlciBiYXNlIGNsYXNzZXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IHBhcnNlUnVudGltZVN0YWdlSWQsIGZpbmRMYXN0V3JpdGVyLCBLZXllZFJlY29yZGVyLCBTZXF1ZW5jZVJlY29yZGVyIH0gZnJvbSAnZm9vdHByaW50anMvdHJhY2UnO1xuICpcbiAqIC8vIFBhcnNlIGEgcnVudGltZVN0YWdlSWRcbiAqIGNvbnN0IHsgc3RhZ2VJZCwgZXhlY3V0aW9uSW5kZXggfSA9IHBhcnNlUnVudGltZVN0YWdlSWQoJ2NhbGwtbGxtIzUnKTtcbiAqXG4gKiAvLyBCYWNrdHJhY2s6IHdobyB3cm90ZSAnc3lzdGVtUHJvbXB0JyBiZWZvcmUgc3RhZ2UgYXQgaWR4IDg/XG4gKiBjb25zdCB3cml0ZXIgPSBmaW5kTGFzdFdyaXRlcihjb21taXRMb2csICdzeXN0ZW1Qcm9tcHQnLCA4KTtcbiAqXG4gKiAvLyBCdWlsZCBhIGtleWVkIHJlY29yZGVyICgxOjEg4oCUIG9uZSBlbnRyeSBwZXIgc3RlcClcbiAqIGNsYXNzIE15UmVjb3JkZXIgZXh0ZW5kcyBLZXllZFJlY29yZGVyPE15RW50cnk+IHsgLi4uIH1cbiAqXG4gKiAvLyBCdWlsZCBhIHNlcXVlbmNlIHJlY29yZGVyICgxOk4g4oCUIG11bHRpcGxlIGVudHJpZXMgcGVyIHN0ZXAsIG9yZGVyaW5nIG1hdHRlcnMpXG4gKiBjbGFzcyBBdWRpdFJlY29yZGVyIGV4dGVuZHMgU2VxdWVuY2VSZWNvcmRlcjxBdWRpdEVudHJ5PiB7IC4uLiB9XG4gKiBgYGBcbiAqL1xuXG4vLyBSdW50aW1lIHN0YWdlIElEIOKAlCB1bmlxdWUgZXhlY3V0aW9uIHN0ZXAgaWRlbnRpZmllcnNcbmV4cG9ydCB0eXBlIHsgRXhlY3V0aW9uQ291bnRlciB9IGZyb20gJy4vbGliL2VuZ2luZS9ydW50aW1lU3RhZ2VJZC5qcyc7XG5leHBvcnQgeyBidWlsZFJ1bnRpbWVTdGFnZUlkLCBjcmVhdGVFeGVjdXRpb25Db3VudGVyLCBwYXJzZVJ1bnRpbWVTdGFnZUlkIH0gZnJvbSAnLi9saWIvZW5naW5lL3J1bnRpbWVTdGFnZUlkLmpzJztcblxuLy8gQ29tbWl0IGxvZyBxdWVyaWVzIOKAlCB0eXBlZCB1dGlsaXRpZXMgZm9yIGJhY2t0cmFja2luZ1xuZXhwb3J0IHsgZmluZENvbW1pdCwgZmluZENvbW1pdHMsIGZpbmRMYXN0V3JpdGVyIH0gZnJvbSAnLi9saWIvbWVtb3J5L2NvbW1pdExvZ1V0aWxzLmpzJztcblxuLy8gQ2F1c2FsIGNoYWluIOKAlCBiYWNrd2FyZCBwcm9ncmFtIHNsaWNpbmcgb24gY29tbWl0IGxvZyAoREFHKVxuZXhwb3J0IHR5cGUgeyBDYXVzYWxDaGFpbk9wdGlvbnMsIENhdXNhbE5vZGUsIEtleXNSZWFkTG9va3VwIH0gZnJvbSAnLi9saWIvbWVtb3J5L2JhY2t0cmFjay5qcyc7XG5leHBvcnQgeyBjYXVzYWxDaGFpbiwgZmxhdHRlbkNhdXNhbERBRywgZm9ybWF0Q2F1c2FsQ2hhaW4gfSBmcm9tICcuL2xpYi9tZW1vcnkvYmFja3RyYWNrLmpzJztcblxuLy8gS2V5ZWRSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMToxIE1hcC1iYXNlZCByZWNvcmRlcnNcbmV4cG9ydCB7IEtleWVkUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9LZXllZFJlY29yZGVyLmpzJztcblxuLy8gU2VxdWVuY2VSZWNvcmRlciDigJQgYmFzZSBjbGFzcyBmb3IgMTpOIG9yZGVyZWQgc2VxdWVuY2UgcmVjb3JkZXJzIHdpdGgga2V5ZWQgaW5kZXhcbmV4cG9ydCB7IFNlcXVlbmNlUmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9TZXF1ZW5jZVJlY29yZGVyLmpzJztcblxuLy8gVG9wb2xvZ3lSZWNvcmRlciDigJQgY29tcG9zaXRpb24gZ3JhcGggYWNjdW11bGF0b3IgKHN1YmZsb3dzICsgY29udHJvbC1mbG93IGVkZ2VzKVxuZXhwb3J0IHR5cGUge1xuICBUb3BvbG9neSxcbiAgVG9wb2xvZ3lFZGdlLFxuICBUb3BvbG9neUluY29taW5nS2luZCxcbiAgVG9wb2xvZ3lOb2RlLFxuICBUb3BvbG9neVJlY29yZGVyT3B0aW9ucyxcbn0gZnJvbSAnLi9saWIvcmVjb3JkZXIvVG9wb2xvZ3lSZWNvcmRlci5qcyc7XG5leHBvcnQgeyBUb3BvbG9neVJlY29yZGVyLCB0b3BvbG9neVJlY29yZGVyIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvVG9wb2xvZ3lSZWNvcmRlci5qcyc7XG5cbi8vIFF1YWxpdHlSZWNvcmRlciDigJQgcGVyLXN0ZXAgcXVhbGl0eSBzY29yaW5nIHdpdGggYmFja3RyYWNraW5nXG5leHBvcnQgdHlwZSB7IFF1YWxpdHlFbnRyeSwgUXVhbGl0eVJlY29yZGVyT3B0aW9ucywgUXVhbGl0eVNjb3JpbmdGbiB9IGZyb20gJy4vbGliL3JlY29yZGVyL1F1YWxpdHlSZWNvcmRlci5qcyc7XG5leHBvcnQgeyBRdWFsaXR5UmVjb3JkZXIgfSBmcm9tICcuL2xpYi9yZWNvcmRlci9RdWFsaXR5UmVjb3JkZXIuanMnO1xuXG4vLyBxdWFsaXR5VHJhY2Ug4oCUIFF1YWxpdHkgU3RhY2sgVHJhY2UgKGJhY2t0cmFjayBmcm9tIGxvdy1zY29yaW5nIHN0ZXBzKVxuZXhwb3J0IHR5cGUgeyBRdWFsaXR5RnJhbWUsIFF1YWxpdHlTdGFja1RyYWNlIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvcXVhbGl0eVRyYWNlLmpzJztcbmV4cG9ydCB7IGZvcm1hdFF1YWxpdHlUcmFjZSwgcXVhbGl0eVRyYWNlIH0gZnJvbSAnLi9saWIvcmVjb3JkZXIvcXVhbGl0eVRyYWNlLmpzJztcbiJdfQ==
@@ -0,0 +1,151 @@
1
+ /**
2
+ * TopologyRecorder — composition graph built during traversal.
3
+ *
4
+ * The gap this fills:
5
+ * footprintjs fires atomic flow events (onSubflowEntry, onFork, onDecision,
6
+ * onLoop) but the accumulated *shape* of a run — who nests inside whom,
7
+ * which nodes are parallel siblings vs branches of a decision — is only
8
+ * visible post-run via `executor.getSnapshot()` tree-walking.
9
+ *
10
+ * Streaming consumers (live UIs, in-flight debuggers) see only the event
11
+ * stream. Every such consumer has to rebuild subflow-stack + fork-map +
12
+ * decision-tracker from scratch, usually slightly wrong in different ways.
13
+ *
14
+ * TopologyRecorder is the standard accumulator: one subscription to the
15
+ * three primitive channels, one live graph, queryable at any moment during
16
+ * or after a run.
17
+ *
18
+ * What it records — THREE node kinds for complete composition coverage:
19
+ * 1. 'subflow' — via onSubflowEntry (a mounted subflow boundary)
20
+ * 2. 'fork-branch' — via onFork (one node per child, synthesized)
21
+ * 3. 'decision-branch' — via onDecision (the chosen branch, synthesized)
22
+ *
23
+ * When a fork-branch or decision-branch target IS ALSO a subflow, the
24
+ * subsequent onSubflowEntry creates a subflow CHILD of the synthetic node.
25
+ * The layered shape preserves both "who branched" and "what the branch ran."
26
+ *
27
+ * Plain sequential stages are NOT nodes — that's StageContext's job.
28
+ * Topology is a graph of control-flow branching, not a full execution tree.
29
+ *
30
+ * Edges:
31
+ * One edge per traversal transition — `kind` matches the child's
32
+ * `incomingKind`. A consumer rendering "parallel columns" filters edges
33
+ * where `kind === 'fork-branch'` sharing the same `from`.
34
+ *
35
+ * @example
36
+ * ```typescript
37
+ * import { topologyRecorder } from 'footprintjs/trace';
38
+ *
39
+ * const topo = topologyRecorder();
40
+ * executor.attachCombinedRecorder(topo); // auto-routes to FlowRecorder channel
41
+ *
42
+ * await executor.run();
43
+ *
44
+ * const { nodes, edges, activeNodeId, rootId } = topo.getTopology();
45
+ * // Consumer queries:
46
+ * topo.getChildren('sf-parent'); // direct children (any kind)
47
+ * topo.getByKind('fork-branch'); // all parallel branches
48
+ * topo.getSubflowNodes(); // only mounted subflows
49
+ * ```
50
+ */
51
+ import type { FlowDecisionEvent, FlowForkEvent, FlowLoopEvent, FlowRecorder, FlowSubflowEvent } from '../engine/narrative/types.js';
52
+ /** The kind of composition unit a node represents. */
53
+ export type TopologyNodeKind = 'subflow' | 'fork-branch' | 'decision-branch';
54
+ /** How the traversal reached this node — drives consumer layout decisions. */
55
+ export type TopologyIncomingKind = 'root' | 'next' | 'fork-branch' | 'decision-branch' | 'loop-iteration';
56
+ /** A composition-significant point in the graph. */
57
+ export interface TopologyNode {
58
+ /** Unique id. Subflows use their subflowId (with `#n` suffix on re-entry).
59
+ * Synthetic nodes (fork-branch / decision-branch) use
60
+ * `fork-${runtimeStageId}-${i}` / `decision-${runtimeStageId}` form. */
61
+ readonly id: string;
62
+ /** What this node represents. */
63
+ readonly kind: TopologyNodeKind;
64
+ /** Display name. For subflows: `FlowSubflowEvent.name`. For fork-branches:
65
+ * the child name from `FlowForkEvent.children`. For decision-branches:
66
+ * the chosen name from `FlowDecisionEvent.chosen`. */
67
+ readonly name: string;
68
+ /** Parent node id. Undefined when this node sits at the run's top level. */
69
+ readonly parentId?: string;
70
+ /** Depth in the topology tree (0 = top-level). */
71
+ readonly depth: number;
72
+ /** How the traversal reached this node. */
73
+ readonly incomingKind: TopologyIncomingKind;
74
+ /** runtimeStageId at the moment the node was created. */
75
+ readonly enteredAt: string;
76
+ /** runtimeStageId when the corresponding subflow exited. Only meaningful
77
+ * for kind='subflow'; fork/decision-branch nodes are instantaneous. */
78
+ exitedAt?: string;
79
+ /** Kind-specific extras: forkParent, decider, rationale, description. */
80
+ readonly metadata?: Readonly<Record<string, unknown>>;
81
+ }
82
+ /** A traversal transition between two nodes. */
83
+ export interface TopologyEdge {
84
+ readonly from: string;
85
+ readonly to: string;
86
+ readonly kind: Exclude<TopologyIncomingKind, 'root'>;
87
+ readonly at: string;
88
+ }
89
+ /** Snapshot of the composition graph. */
90
+ export interface Topology {
91
+ readonly nodes: ReadonlyArray<TopologyNode>;
92
+ readonly edges: ReadonlyArray<TopologyEdge>;
93
+ /** Currently-active subflow (top of the subflow stack). Fork-branch and
94
+ * decision-branch nodes are instantaneous — they don't affect activeNodeId. */
95
+ readonly activeNodeId: string | null;
96
+ /** First node inserted. null before any composition event fires. */
97
+ readonly rootId: string | null;
98
+ }
99
+ export interface TopologyRecorderOptions {
100
+ /** Recorder id. Defaults to `topology-N` (auto-incremented). */
101
+ id?: string;
102
+ }
103
+ /**
104
+ * Factory — matches the `narrative()` / `metrics()` style.
105
+ */
106
+ export declare function topologyRecorder(options?: TopologyRecorderOptions): TopologyRecorder;
107
+ /**
108
+ * Stateful accumulator that watches FlowRecorder events and maintains a live
109
+ * composition graph. Attach via `executor.attachCombinedRecorder(recorder)` —
110
+ * footprintjs detects the `FlowRecorder` method shape and routes events.
111
+ */
112
+ export declare class TopologyRecorder implements FlowRecorder {
113
+ readonly id: string;
114
+ private readonly nodesById;
115
+ private readonly nodeOrder;
116
+ private readonly edges;
117
+ /** Stack of active SUBFLOW node ids. Fork/decision-branch nodes never push. */
118
+ private readonly subflowStack;
119
+ /** Map of childName → pending fork-branch synthetic node, consumed by
120
+ * the next matching `onSubflowEntry`. */
121
+ private readonly pendingForkByName;
122
+ /** Pending decision-branch synthetic node, consumed by a matching entry. */
123
+ private pendingDecision?;
124
+ constructor(options?: TopologyRecorderOptions);
125
+ onSubflowEntry(event: FlowSubflowEvent): void;
126
+ onSubflowExit(event: FlowSubflowEvent): void;
127
+ onFork(event: FlowForkEvent): void;
128
+ onDecision(event: FlowDecisionEvent): void;
129
+ onLoop(event: FlowLoopEvent): void;
130
+ /** Called by the executor before each `run()` — resets all state. */
131
+ clear(): void;
132
+ /** Live snapshot of the composition graph. Safe during or after a run. */
133
+ getTopology(): Topology;
134
+ /** Direct children of a node — insertion-ordered. */
135
+ getChildren(nodeId: string): TopologyNode[];
136
+ /** All nodes of a given kind. */
137
+ getByKind(kind: TopologyNodeKind): TopologyNode[];
138
+ /** All mounted subflow nodes. Convenience for agent-centric views. */
139
+ getSubflowNodes(): TopologyNode[];
140
+ /** All fork-branch nodes sharing the same parent as `nodeId` — i.e.,
141
+ * parallel siblings of a parallel branch. Empty if `nodeId` isn't a
142
+ * fork-branch or has no parent. */
143
+ getParallelSiblings(nodeId: string): TopologyNode[];
144
+ /** Emit a snapshot bundle for inclusion in `executor.getSnapshot()`. */
145
+ toSnapshot(): {
146
+ name: string;
147
+ description: string;
148
+ preferredOperation: "translate";
149
+ data: Topology;
150
+ };
151
+ }
@@ -27,6 +27,8 @@ export type { CausalChainOptions, CausalNode, KeysReadLookup } from './lib/memor
27
27
  export { causalChain, flattenCausalDAG, formatCausalChain } from './lib/memory/backtrack.js';
28
28
  export { KeyedRecorder } from './lib/recorder/KeyedRecorder.js';
29
29
  export { SequenceRecorder } from './lib/recorder/SequenceRecorder.js';
30
+ export type { Topology, TopologyEdge, TopologyIncomingKind, TopologyNode, TopologyRecorderOptions, } from './lib/recorder/TopologyRecorder.js';
31
+ export { TopologyRecorder, topologyRecorder } from './lib/recorder/TopologyRecorder.js';
30
32
  export type { QualityEntry, QualityRecorderOptions, QualityScoringFn } from './lib/recorder/QualityRecorder.js';
31
33
  export { QualityRecorder } from './lib/recorder/QualityRecorder.js';
32
34
  export type { QualityFrame, QualityStackTrace } from './lib/recorder/qualityTrace.js';
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "footprintjs",
3
- "version": "4.14.0",
3
+ "version": "4.15.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",