footprintjs 4.0.5 → 4.2.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.
Files changed (61) hide show
  1. package/CLAUDE.md +7 -0
  2. package/dist/esm/index.js +6 -1
  3. package/dist/esm/lib/builder/FlowChartBuilder.js +13 -8
  4. package/dist/esm/lib/engine/graph/StageNode.js +1 -1
  5. package/dist/esm/lib/engine/handlers/DeciderHandler.js +8 -7
  6. package/dist/esm/lib/engine/handlers/SelectorHandler.js +8 -5
  7. package/dist/esm/lib/engine/handlers/SubflowExecutor.js +28 -246
  8. package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +232 -139
  9. package/dist/esm/lib/engine/narrative/index.js +1 -1
  10. package/dist/esm/lib/engine/narrative/narrativeTypes.js +1 -1
  11. package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +64 -22
  12. package/dist/esm/lib/engine/types.js +1 -1
  13. package/dist/esm/lib/memory/StageContext.js +36 -9
  14. package/dist/esm/lib/reactive/createTypedScope.js +9 -6
  15. package/dist/esm/lib/reactive/types.js +1 -1
  16. package/dist/esm/lib/recorder/CompositeRecorder.js +172 -0
  17. package/dist/esm/lib/recorder/index.js +2 -0
  18. package/dist/esm/lib/runner/FlowChartExecutor.js +33 -3
  19. package/dist/esm/lib/scope/ScopeFacade.js +52 -4
  20. package/dist/esm/lib/scope/recorders/DebugRecorder.js +18 -2
  21. package/dist/esm/lib/scope/recorders/MetricRecorder.js +51 -4
  22. package/dist/esm/recorders.js +6 -5
  23. package/dist/index.js +21 -15
  24. package/dist/lib/builder/FlowChartBuilder.js +13 -8
  25. package/dist/lib/engine/graph/StageNode.js +1 -1
  26. package/dist/lib/engine/handlers/DeciderHandler.js +8 -7
  27. package/dist/lib/engine/handlers/SelectorHandler.js +8 -5
  28. package/dist/lib/engine/handlers/SubflowExecutor.js +27 -245
  29. package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +232 -139
  30. package/dist/lib/engine/narrative/index.js +1 -1
  31. package/dist/lib/engine/narrative/narrativeTypes.js +1 -1
  32. package/dist/lib/engine/traversal/FlowchartTraverser.js +64 -22
  33. package/dist/lib/engine/types.js +1 -1
  34. package/dist/lib/memory/StageContext.js +36 -9
  35. package/dist/lib/reactive/createTypedScope.js +9 -6
  36. package/dist/lib/reactive/types.js +1 -1
  37. package/dist/lib/recorder/CompositeRecorder.js +176 -0
  38. package/dist/lib/recorder/index.js +6 -0
  39. package/dist/lib/runner/FlowChartExecutor.js +33 -3
  40. package/dist/lib/scope/ScopeFacade.js +52 -4
  41. package/dist/lib/scope/recorders/DebugRecorder.js +18 -2
  42. package/dist/lib/scope/recorders/MetricRecorder.js +51 -4
  43. package/dist/recorders.js +8 -6
  44. package/dist/types/index.d.ts +4 -0
  45. package/dist/types/lib/engine/graph/StageNode.d.ts +4 -0
  46. package/dist/types/lib/engine/handlers/SubflowExecutor.d.ts +10 -29
  47. package/dist/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +18 -1
  48. package/dist/types/lib/engine/narrative/index.d.ts +1 -1
  49. package/dist/types/lib/engine/narrative/narrativeTypes.d.ts +92 -0
  50. package/dist/types/lib/engine/traversal/FlowchartTraverser.d.ts +20 -8
  51. package/dist/types/lib/engine/types.d.ts +51 -0
  52. package/dist/types/lib/memory/StageContext.d.ts +14 -3
  53. package/dist/types/lib/reactive/types.d.ts +2 -0
  54. package/dist/types/lib/recorder/CompositeRecorder.d.ts +95 -0
  55. package/dist/types/lib/recorder/index.d.ts +2 -0
  56. package/dist/types/lib/runner/FlowChartExecutor.d.ts +27 -1
  57. package/dist/types/lib/scope/ScopeFacade.d.ts +11 -0
  58. package/dist/types/lib/scope/recorders/DebugRecorder.d.ts +16 -0
  59. package/dist/types/lib/scope/recorders/MetricRecorder.d.ts +47 -2
  60. package/dist/types/recorders.d.ts +9 -4
  61. package/package.json +1 -1
@@ -16,7 +16,7 @@ import { summarizeValue } from '../../scope/recorders/summarizeValue.js';
16
16
  // ── Recorder ───────────────────────────────────────────────────────────────
17
17
  export class CombinedNarrativeRecorder {
18
18
  constructor(options) {
19
- var _a, _b, _c, _d;
19
+ var _a, _b, _c, _d, _e;
20
20
  this.entries = [];
21
21
  /**
22
22
  * Pending scope ops keyed by stageName. Flushed in onStageExecuted/onDecision.
@@ -35,6 +35,8 @@ export class CombinedNarrativeRecorder {
35
35
  this.includeStepNumbers = (_b = options === null || options === void 0 ? void 0 : options.includeStepNumbers) !== null && _b !== void 0 ? _b : true;
36
36
  this.includeValues = (_c = options === null || options === void 0 ? void 0 : options.includeValues) !== null && _c !== void 0 ? _c : true;
37
37
  this.maxValueLength = (_d = options === null || options === void 0 ? void 0 : options.maxValueLength) !== null && _d !== void 0 ? _d : 80;
38
+ this.formatValue = (_e = options === null || options === void 0 ? void 0 : options.formatValue) !== null && _e !== void 0 ? _e : summarizeValue;
39
+ this.renderer = options === null || options === void 0 ? void 0 : options.renderer;
38
40
  }
39
41
  // ── Scope channel (fires first, during stage execution) ───────────────
40
42
  onRead(event) {
@@ -43,35 +45,35 @@ export class CombinedNarrativeRecorder {
43
45
  this.bufferOp(event.stageName, {
44
46
  type: 'read',
45
47
  key: event.key,
46
- valueSummary: summarizeValue(event.value, this.maxValueLength),
48
+ rawValue: event.value,
47
49
  });
48
50
  }
49
51
  onWrite(event) {
50
52
  this.bufferOp(event.stageName, {
51
53
  type: 'write',
52
54
  key: event.key,
53
- valueSummary: summarizeValue(event.value, this.maxValueLength),
55
+ rawValue: event.value,
54
56
  operation: event.operation,
55
57
  });
56
58
  }
57
59
  // ── Flow channel (fires after stage execution) ────────────────────────
58
60
  onStageExecuted(event) {
59
- var _a, _b, _c, _d;
61
+ var _a, _b, _c, _d, _e, _f, _g;
60
62
  const stageId = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId;
61
63
  const sfKey = (_c = (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId) !== null && _c !== void 0 ? _c : '';
62
64
  const stageNum = this.incrementStageCounter(sfKey);
63
65
  const isFirst = this.consumeFirstStageFlag(sfKey);
64
- const text = isFirst
65
- ? event.description
66
- ? `The process began: ${event.description}.`
67
- : `The process began with ${event.stageName}.`
68
- : event.description
69
- ? `Next step: ${event.description}.`
70
- : `Next, it moved on to ${event.stageName}.`;
71
- const sfId = (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.subflowId;
66
+ const ctx = {
67
+ stageName: event.stageName,
68
+ stageNumber: stageNum,
69
+ isFirst,
70
+ description: event.description,
71
+ };
72
+ const text = (_f = (_e = (_d = this.renderer) === null || _d === void 0 ? void 0 : _d.renderStage) === null || _e === void 0 ? void 0 : _e.call(_d, ctx)) !== null && _f !== void 0 ? _f : this.defaultRenderStage(ctx);
73
+ const sfId = (_g = event.traversalContext) === null || _g === void 0 ? void 0 : _g.subflowId;
72
74
  this.entries.push({
73
75
  type: 'stage',
74
- text: `Stage ${stageNum}: ${text}`,
76
+ text,
75
77
  depth: 0,
76
78
  stageName: event.stageName,
77
79
  stageId,
@@ -80,70 +82,45 @@ export class CombinedNarrativeRecorder {
80
82
  this.flushOps(event.stageName, sfId, stageId);
81
83
  }
82
84
  onDecision(event) {
83
- var _a, _b, _c, _d, _e, _f;
85
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m;
84
86
  const deciderStageIdEarly = (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId;
85
87
  // Emit the decider stage entry (deciders don't fire onStageExecuted)
86
88
  const sfKey = (_c = (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId) !== null && _c !== void 0 ? _c : '';
87
89
  const stageNum = this.incrementStageCounter(sfKey);
88
90
  const isFirst = this.consumeFirstStageFlag(sfKey);
89
- const stageText = isFirst
90
- ? event.description
91
- ? `The process began: ${event.description}.`
92
- : `The process began with ${event.decider}.`
93
- : event.description
94
- ? `Next step: ${event.description}.`
95
- : `Next, it moved on to ${event.decider}.`;
91
+ const stageCtx = {
92
+ stageName: event.decider,
93
+ stageNumber: stageNum,
94
+ isFirst,
95
+ description: event.description,
96
+ };
97
+ const stageText = (_f = (_e = (_d = this.renderer) === null || _d === void 0 ? void 0 : _d.renderStage) === null || _e === void 0 ? void 0 : _e.call(_d, stageCtx)) !== null && _f !== void 0 ? _f : this.defaultRenderStage(stageCtx);
96
98
  this.entries.push({
97
99
  type: 'stage',
98
- text: `Stage ${stageNum}: ${stageText}`,
100
+ text: stageText,
99
101
  depth: 0,
100
102
  stageName: event.decider,
101
103
  stageId: deciderStageIdEarly,
102
- subflowId: (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.subflowId,
104
+ subflowId: (_g = event.traversalContext) === null || _g === void 0 ? void 0 : _g.subflowId,
103
105
  });
104
- this.flushOps(event.decider, (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.subflowId, deciderStageIdEarly);
105
- // Emit the condition entry with evidence-aware rendering when available
106
- const branchName = event.chosen;
107
- let conditionText;
108
- if (event.evidence) {
109
- // Rich evidence from decide() helper
110
- const matchedRule = event.evidence.rules.find((r) => r.matched);
111
- if (matchedRule) {
112
- const label = matchedRule.label ? ` "${matchedRule.label}"` : '';
113
- if (matchedRule.type === 'filter') {
114
- const parts = matchedRule.conditions.map((c) => `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\u2713' : '\u2717'}`);
115
- conditionText = `It evaluated Rule ${matchedRule.ruleIndex}${label}: ${parts.join(', ')}, and chose ${branchName}.`;
116
- }
117
- else {
118
- const parts = matchedRule.inputs.map((i) => `${i.key}=${i.valueSummary}`);
119
- conditionText = `It examined${label}: ${parts.join(', ')}, and chose ${branchName}.`;
120
- }
121
- }
122
- else {
123
- const erroredCount = event.evidence.rules.filter((r) => r.matchError !== undefined).length;
124
- const errorNote = erroredCount > 0 ? ` (${erroredCount} rule${erroredCount > 1 ? 's' : ''} threw errors)` : '';
125
- conditionText = `No rules matched${errorNote}, fell back to default: ${branchName}.`;
126
- }
127
- }
128
- else if (event.description && event.rationale) {
129
- conditionText = `It ${event.description}: ${event.rationale}, so it chose ${branchName}.`;
130
- }
131
- else if (event.description) {
132
- conditionText = `It ${event.description} and chose ${branchName}.`;
133
- }
134
- else if (event.rationale) {
135
- conditionText = `A decision was made: ${event.rationale}, so the path taken was ${branchName}.`;
136
- }
137
- else {
138
- conditionText = `A decision was made, and the path taken was ${branchName}.`;
139
- }
106
+ this.flushOps(event.decider, (_h = event.traversalContext) === null || _h === void 0 ? void 0 : _h.subflowId, deciderStageIdEarly);
107
+ // Emit the condition entry as a nested sub-item (depth 1) of the stage above.
108
+ // Decision outcome is a detail of the decider stage, not a separate top-level entry.
109
+ const decisionCtx = {
110
+ decider: event.decider,
111
+ chosen: event.chosen,
112
+ description: event.description,
113
+ rationale: event.rationale,
114
+ evidence: event.evidence,
115
+ };
116
+ const conditionText = (_l = (_k = (_j = this.renderer) === null || _j === void 0 ? void 0 : _j.renderDecision) === null || _k === void 0 ? void 0 : _k.call(_j, decisionCtx)) !== null && _l !== void 0 ? _l : this.defaultRenderDecision(decisionCtx);
140
117
  this.entries.push({
141
118
  type: 'condition',
142
- text: `[Condition]: ${conditionText}`,
143
- depth: 0,
119
+ text: conditionText,
120
+ depth: 1,
144
121
  stageName: event.decider,
145
122
  stageId: deciderStageIdEarly,
146
- subflowId: (_f = event.traversalContext) === null || _f === void 0 ? void 0 : _f.subflowId,
123
+ subflowId: (_m = event.traversalContext) === null || _m === void 0 ? void 0 : _m.subflowId,
147
124
  });
148
125
  }
149
126
  onNext() {
@@ -151,95 +128,97 @@ export class CombinedNarrativeRecorder {
151
128
  // For deciders (no onStageExecuted), onDecision handles the announcement.
152
129
  }
153
130
  onFork(event) {
154
- var _a, _b;
155
- const names = event.children.join(', ');
131
+ var _a, _b, _c, _d, _e;
132
+ const ctx = { children: event.children };
133
+ const text = (_c = (_b = (_a = this.renderer) === null || _a === void 0 ? void 0 : _a.renderFork) === null || _b === void 0 ? void 0 : _b.call(_a, ctx)) !== null && _c !== void 0 ? _c : this.defaultRenderFork(ctx);
156
134
  this.entries.push({
157
135
  type: 'fork',
158
- text: `[Parallel]: Forking into ${event.children.length} parallel paths: ${names}.`,
136
+ text,
159
137
  depth: 0,
160
- stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
161
- subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
138
+ stageId: (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.stageId,
139
+ subflowId: (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.subflowId,
162
140
  });
163
141
  }
164
142
  onSelected(event) {
165
- var _a, _b;
166
- let text;
167
- if (event.evidence) {
168
- const matched = event.evidence.rules.filter((r) => r.matched);
169
- const parts = matched.map((r) => {
170
- const label = r.label ? ` "${r.label}"` : '';
171
- if (r.type === 'filter') {
172
- const conds = r.conditions
173
- .map((c) => `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\u2713' : '\u2717'}`)
174
- .join(', ');
175
- return `${r.branch}${label} (${conds})`;
176
- }
177
- const inputs = r.inputs.map((i) => `${i.key}=${i.valueSummary}`).join(', ');
178
- return `${r.branch}${label} (${inputs})`;
179
- });
180
- text = `[Selected]: ${event.selected.length} of ${event.total} paths selected: ${parts.join('; ')}.`;
181
- }
182
- else {
183
- const names = event.selected.join(', ');
184
- text = `[Selected]: ${event.selected.length} of ${event.total} paths selected for execution: ${names}.`;
185
- }
143
+ var _a, _b, _c, _d, _e;
144
+ const ctx = {
145
+ selected: event.selected,
146
+ total: event.total,
147
+ evidence: event.evidence,
148
+ };
149
+ const text = (_c = (_b = (_a = this.renderer) === null || _a === void 0 ? void 0 : _a.renderSelected) === null || _b === void 0 ? void 0 : _b.call(_a, ctx)) !== null && _c !== void 0 ? _c : this.defaultRenderSelected(ctx);
186
150
  this.entries.push({
187
151
  type: 'selector',
188
152
  text,
189
153
  depth: 0,
190
- stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
191
- subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
154
+ stageId: (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.stageId,
155
+ subflowId: (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.subflowId,
192
156
  });
193
157
  }
194
158
  onSubflowEntry(event) {
195
- var _a, _b, _c;
159
+ var _a, _b, _c, _d, _e, _f;
196
160
  // Reset stage counter for this subflow so stages start at "Stage 1" on re-entry
197
161
  const sfKey = (_a = event.subflowId) !== null && _a !== void 0 ? _a : '';
198
162
  this.stageCounters.delete(sfKey);
199
163
  this.firstStageFlags.delete(sfKey);
200
- const text = event.description
201
- ? `Entering the ${event.name} subflow: ${event.description}.`
202
- : `Entering the ${event.name} subflow.`;
164
+ const ctx = {
165
+ name: event.name,
166
+ direction: 'entry',
167
+ description: event.description,
168
+ };
169
+ const text = (_d = (_c = (_b = this.renderer) === null || _b === void 0 ? void 0 : _b.renderSubflow) === null || _c === void 0 ? void 0 : _c.call(_b, ctx)) !== null && _d !== void 0 ? _d : this.defaultRenderSubflow(ctx);
203
170
  this.entries.push({
204
171
  type: 'subflow',
205
172
  text,
206
173
  depth: 0,
207
- stageId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.stageId,
208
- subflowId: (_c = event.traversalContext) === null || _c === void 0 ? void 0 : _c.subflowId,
174
+ stageName: event.name,
175
+ stageId: (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.stageId,
176
+ subflowId: (_f = event.traversalContext) === null || _f === void 0 ? void 0 : _f.subflowId,
209
177
  });
210
178
  }
211
179
  onSubflowExit(event) {
212
- var _a, _b;
180
+ var _a, _b, _c, _d, _e;
181
+ const ctx = {
182
+ name: event.name,
183
+ direction: 'exit',
184
+ };
185
+ const text = (_c = (_b = (_a = this.renderer) === null || _a === void 0 ? void 0 : _a.renderSubflow) === null || _b === void 0 ? void 0 : _b.call(_a, ctx)) !== null && _c !== void 0 ? _c : this.defaultRenderSubflow(ctx);
213
186
  this.entries.push({
214
187
  type: 'subflow',
215
- text: `Exiting the ${event.name} subflow.`,
188
+ text,
216
189
  depth: 0,
217
- stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
218
- subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
190
+ stageName: event.name,
191
+ stageId: (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.stageId,
192
+ subflowId: (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.subflowId,
219
193
  });
220
194
  }
221
195
  onLoop(event) {
222
- var _a, _b;
223
- const text = event.description
224
- ? `On pass ${event.iteration}: ${event.description} again.`
225
- : `On pass ${event.iteration} through ${event.target}.`;
196
+ var _a, _b, _c, _d, _e;
197
+ const ctx = {
198
+ target: event.target,
199
+ iteration: event.iteration,
200
+ description: event.description,
201
+ };
202
+ const text = (_c = (_b = (_a = this.renderer) === null || _a === void 0 ? void 0 : _a.renderLoop) === null || _b === void 0 ? void 0 : _b.call(_a, ctx)) !== null && _c !== void 0 ? _c : this.defaultRenderLoop(ctx);
226
203
  this.entries.push({
227
204
  type: 'loop',
228
205
  text,
229
206
  depth: 0,
230
- stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
231
- subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
207
+ stageId: (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.stageId,
208
+ subflowId: (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.subflowId,
232
209
  });
233
210
  }
234
211
  onBreak(event) {
235
- var _a, _b;
212
+ var _a, _b, _c, _d, _e;
213
+ const ctx = { stageName: event.stageName };
214
+ const text = (_c = (_b = (_a = this.renderer) === null || _a === void 0 ? void 0 : _a.renderBreak) === null || _b === void 0 ? void 0 : _b.call(_a, ctx)) !== null && _c !== void 0 ? _c : this.defaultRenderBreak(ctx);
236
215
  this.entries.push({
237
216
  type: 'break',
238
- text: `Execution stopped at ${event.stageName}.`,
217
+ text,
239
218
  depth: 0,
240
219
  stageName: event.stageName,
241
- stageId: (_a = event.traversalContext) === null || _a === void 0 ? void 0 : _a.stageId,
242
- subflowId: (_b = event.traversalContext) === null || _b === void 0 ? void 0 : _b.subflowId,
220
+ stageId: (_d = event.traversalContext) === null || _d === void 0 ? void 0 : _d.stageId,
221
+ subflowId: (_e = event.traversalContext) === null || _e === void 0 ? void 0 : _e.subflowId,
243
222
  });
244
223
  }
245
224
  /**
@@ -248,28 +227,33 @@ export class CombinedNarrativeRecorder {
248
227
  * - Recorder.onError (ErrorEvent from scope system — ignored for narrative)
249
228
  */
250
229
  onError(event) {
251
- var _a, _b, _c, _d;
230
+ var _a, _b, _c, _d, _e, _f, _g;
252
231
  // Only handle flow errors (which have `message` and `structuredError`)
253
232
  if (typeof event.message !== 'string')
254
233
  return;
255
234
  const flowEvent = event;
256
- let text = `An error occurred at ${flowEvent.stageName}: ${flowEvent.message}.`;
235
+ let validationIssues;
257
236
  if ((_b = (_a = flowEvent.structuredError) === null || _a === void 0 ? void 0 : _a.issues) === null || _b === void 0 ? void 0 : _b.length) {
258
- const details = flowEvent.structuredError.issues
237
+ validationIssues = flowEvent.structuredError.issues
259
238
  .map((issue) => {
260
239
  const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';
261
240
  return `${path}: ${issue.message}`;
262
241
  })
263
242
  .join('; ');
264
- text += ` Validation issues: ${details}.`;
265
243
  }
244
+ const ctx = {
245
+ stageName: flowEvent.stageName,
246
+ message: flowEvent.message,
247
+ validationIssues,
248
+ };
249
+ const text = (_e = (_d = (_c = this.renderer) === null || _c === void 0 ? void 0 : _c.renderError) === null || _d === void 0 ? void 0 : _d.call(_c, ctx)) !== null && _e !== void 0 ? _e : this.defaultRenderError(ctx);
266
250
  this.entries.push({
267
251
  type: 'error',
268
- text: `[Error]: ${text}`,
252
+ text,
269
253
  depth: 0,
270
254
  stageName: flowEvent.stageName,
271
- stageId: (_c = flowEvent.traversalContext) === null || _c === void 0 ? void 0 : _c.stageId,
272
- subflowId: (_d = flowEvent.traversalContext) === null || _d === void 0 ? void 0 : _d.subflowId,
255
+ stageId: (_f = flowEvent.traversalContext) === null || _f === void 0 ? void 0 : _f.stageId,
256
+ subflowId: (_g = flowEvent.traversalContext) === null || _g === void 0 ? void 0 : _g.subflowId,
273
257
  });
274
258
  }
275
259
  // ── Output ────────────────────────────────────────────────────────────
@@ -329,31 +313,23 @@ export class CombinedNarrativeRecorder {
329
313
  ops.push({ ...op, stepNumber: ops.length + 1 });
330
314
  }
331
315
  flushOps(stageName, subflowId, stageId) {
316
+ var _a;
332
317
  const ops = this.pendingOps.get(stageName);
333
318
  if (!ops || ops.length === 0)
334
319
  return;
335
320
  for (const op of ops) {
336
- const stepPrefix = this.includeStepNumbers ? `Step ${op.stepNumber}: ` : '';
337
- let text;
338
- if (op.type === 'read') {
339
- text =
340
- this.includeValues && op.valueSummary
341
- ? `${stepPrefix}Read ${op.key} = ${op.valueSummary}`
342
- : `${stepPrefix}Read ${op.key}`;
343
- }
344
- else if (op.operation === 'delete') {
345
- text = `${stepPrefix}Delete ${op.key}`;
346
- }
347
- else if (op.operation === 'update') {
348
- text = this.includeValues
349
- ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`
350
- : `${stepPrefix}Update ${op.key}`;
351
- }
352
- else {
353
- text = this.includeValues
354
- ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`
355
- : `${stepPrefix}Write ${op.key}`;
356
- }
321
+ const valueSummary = this.formatValue(op.rawValue, this.maxValueLength);
322
+ const opCtx = {
323
+ type: op.type,
324
+ key: op.key,
325
+ rawValue: op.rawValue,
326
+ valueSummary,
327
+ operation: op.operation,
328
+ stepNumber: op.stepNumber,
329
+ };
330
+ const text = ((_a = this.renderer) === null || _a === void 0 ? void 0 : _a.renderOp) ? this.renderer.renderOp(opCtx) : this.defaultRenderOp(opCtx);
331
+ if (text == null)
332
+ continue; // renderer excluded this op (null or undefined)
357
333
  this.entries.push({
358
334
  type: 'step',
359
335
  text,
@@ -362,9 +338,126 @@ export class CombinedNarrativeRecorder {
362
338
  stageId,
363
339
  stepNumber: op.stepNumber,
364
340
  subflowId,
341
+ rawValue: op.rawValue,
365
342
  });
366
343
  }
367
344
  this.pendingOps.delete(stageName);
368
345
  }
346
+ // ── Default renderers (used when no custom renderer is provided) ────
347
+ defaultRenderStage(ctx) {
348
+ const inner = ctx.isFirst
349
+ ? ctx.description
350
+ ? `The process began: ${ctx.description}.`
351
+ : `The process began with ${ctx.stageName}.`
352
+ : ctx.description
353
+ ? `Next step: ${ctx.description}.`
354
+ : `Next, it moved on to ${ctx.stageName}.`;
355
+ return `Stage ${ctx.stageNumber}: ${inner}`;
356
+ }
357
+ defaultRenderOp(ctx) {
358
+ const stepPrefix = this.includeStepNumbers ? `Step ${ctx.stepNumber}: ` : '';
359
+ if (ctx.type === 'read') {
360
+ return this.includeValues && ctx.valueSummary
361
+ ? `${stepPrefix}Read ${ctx.key} = ${ctx.valueSummary}`
362
+ : `${stepPrefix}Read ${ctx.key}`;
363
+ }
364
+ if (ctx.operation === 'delete') {
365
+ return `${stepPrefix}Delete ${ctx.key}`;
366
+ }
367
+ if (ctx.operation === 'update') {
368
+ return this.includeValues
369
+ ? `${stepPrefix}Update ${ctx.key} = ${ctx.valueSummary}`
370
+ : `${stepPrefix}Update ${ctx.key}`;
371
+ }
372
+ return this.includeValues ? `${stepPrefix}Write ${ctx.key} = ${ctx.valueSummary}` : `${stepPrefix}Write ${ctx.key}`;
373
+ }
374
+ defaultRenderDecision(ctx) {
375
+ var _a, _b, _c;
376
+ const branchName = ctx.chosen;
377
+ let conditionText;
378
+ if (ctx.evidence) {
379
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
380
+ const evidence = ctx.evidence;
381
+ const matchedRule = (_a = evidence.rules) === null || _a === void 0 ? void 0 : _a.find((r) => r.matched);
382
+ if (matchedRule) {
383
+ const label = matchedRule.label ? ` "${matchedRule.label}"` : '';
384
+ if (matchedRule.type === 'filter') {
385
+ const parts = matchedRule.conditions.map((c) => `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\u2713' : '\u2717'}`);
386
+ conditionText = `It evaluated Rule ${matchedRule.ruleIndex}${label}: ${parts.join(', ')}, and chose ${branchName}.`;
387
+ }
388
+ else {
389
+ const parts = matchedRule.inputs.map((i) => `${i.key}=${i.valueSummary}`);
390
+ conditionText = `It examined${label}: ${parts.join(', ')}, and chose ${branchName}.`;
391
+ }
392
+ }
393
+ else {
394
+ const erroredCount = (_c = (_b = evidence.rules) === null || _b === void 0 ? void 0 : _b.filter((r) => r.matchError !== undefined).length) !== null && _c !== void 0 ? _c : 0;
395
+ const errorNote = erroredCount > 0 ? ` (${erroredCount} rule${erroredCount > 1 ? 's' : ''} threw errors)` : '';
396
+ conditionText = `No rules matched${errorNote}, fell back to default: ${branchName}.`;
397
+ }
398
+ }
399
+ else if (ctx.description && ctx.rationale) {
400
+ conditionText = `It ${ctx.description}: ${ctx.rationale}, so it chose ${branchName}.`;
401
+ }
402
+ else if (ctx.description) {
403
+ conditionText = `It ${ctx.description} and chose ${branchName}.`;
404
+ }
405
+ else if (ctx.rationale) {
406
+ conditionText = `A decision was made: ${ctx.rationale}, so the path taken was ${branchName}.`;
407
+ }
408
+ else {
409
+ conditionText = `A decision was made, and the path taken was ${branchName}.`;
410
+ }
411
+ return `[Condition]: ${conditionText}`;
412
+ }
413
+ defaultRenderFork(ctx) {
414
+ const names = ctx.children.join(', ');
415
+ return `[Parallel]: Forking into ${ctx.children.length} parallel paths: ${names}.`;
416
+ }
417
+ defaultRenderSelected(ctx) {
418
+ var _a, _b;
419
+ if (ctx.evidence) {
420
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
421
+ const evidence = ctx.evidence;
422
+ const matched = (_b = (_a = evidence.rules) === null || _a === void 0 ? void 0 : _a.filter((r) => r.matched)) !== null && _b !== void 0 ? _b : [];
423
+ const parts = matched.map((r) => {
424
+ const label = r.label ? ` "${r.label}"` : '';
425
+ if (r.type === 'filter') {
426
+ const conds = r.conditions
427
+ .map((c) => `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\u2713' : '\u2717'}`)
428
+ .join(', ');
429
+ return `${r.branch}${label} (${conds})`;
430
+ }
431
+ const inputs = r.inputs.map((i) => `${i.key}=${i.valueSummary}`).join(', ');
432
+ return `${r.branch}${label} (${inputs})`;
433
+ });
434
+ return `[Selected]: ${ctx.selected.length} of ${ctx.total} paths selected: ${parts.join('; ')}.`;
435
+ }
436
+ const names = ctx.selected.join(', ');
437
+ return `[Selected]: ${ctx.selected.length} of ${ctx.total} paths selected for execution: ${names}.`;
438
+ }
439
+ defaultRenderSubflow(ctx) {
440
+ if (ctx.direction === 'exit') {
441
+ return `Exiting the ${ctx.name} subflow.`;
442
+ }
443
+ return ctx.description
444
+ ? `Entering the ${ctx.name} subflow: ${ctx.description}.`
445
+ : `Entering the ${ctx.name} subflow.`;
446
+ }
447
+ defaultRenderLoop(ctx) {
448
+ return ctx.description
449
+ ? `On pass ${ctx.iteration}: ${ctx.description} again.`
450
+ : `On pass ${ctx.iteration} through ${ctx.target}.`;
451
+ }
452
+ defaultRenderBreak(ctx) {
453
+ return `Execution stopped at ${ctx.stageName}.`;
454
+ }
455
+ defaultRenderError(ctx) {
456
+ let text = `An error occurred at ${ctx.stageName}: ${ctx.message}.`;
457
+ if (ctx.validationIssues) {
458
+ text += ` Validation issues: ${ctx.validationIssues}.`;
459
+ }
460
+ return `[Error]: ${text}`;
461
+ }
369
462
  }
370
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/CombinedNarrativeRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeRecorder — Inline narrative builder that merges flow + data during traversal.\n *\n * Replaces the post-processing CombinedNarrativeBuilder by implementing BOTH\n * FlowRecorder (control-flow events) and Recorder (scope data events).\n *\n * Event ordering guarantees this works:\n *   1. Scope events (onRead, onWrite) fire DURING stage execution\n *   2. Flow events (onStageExecuted, onDecision) fire AFTER stage execution\n *   3. Both carry the same `stageName` — no matching ambiguity\n *\n * So we buffer scope ops per-stage, then when the flow event arrives,\n * emit the stage entry + flush the buffered ops in one pass.\n */\n\nimport { summarizeValue } from '../../scope/recorders/summarizeValue.js';\nimport type { ReadEvent, Recorder, WriteEvent } from '../../scope/types.js';\nimport type { CombinedNarrativeEntry } from './narrativeTypes.js';\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\n// ── Types ──────────────────────────────────────────────────────────────────\n\ninterface BufferedOp {\n  type: 'read' | 'write';\n  key: string;\n  valueSummary: string;\n  operation?: 'set' | 'update' | 'delete';\n  stepNumber: number;\n}\n\nexport interface CombinedNarrativeRecorderOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  maxValueLength?: number;\n}\n\n// ── Recorder ───────────────────────────────────────────────────────────────\n\nexport class CombinedNarrativeRecorder implements FlowRecorder, Recorder {\n  readonly id: string;\n\n  private entries: CombinedNarrativeEntry[] = [];\n  /**\n   * Pending scope ops keyed by stageName. Flushed in onStageExecuted/onDecision.\n   *\n   * Name collisions (two stages with the same name, different IDs) are prevented by\n   * the event ordering contract: scope events (onRead/onWrite) for stage N are always\n   * flushed by onStageExecuted for stage N before stage N+1's scope events begin.\n   * So the key is always uniquely bound to the currently-executing stage.\n   */\n  private pendingOps = new Map<string, BufferedOp[]>();\n  /** Per-subflow stage counters. Key '' = root flow. */\n  private stageCounters = new Map<string, number>();\n  /** Per-subflow first-stage flags. Key '' = root flow. */\n  private firstStageFlags = new Map<string, boolean>();\n\n  private includeStepNumbers: boolean;\n  private includeValues: boolean;\n  private maxValueLength: number;\n\n  constructor(options?: CombinedNarrativeRecorderOptions & { id?: string }) {\n    this.id = options?.id ?? 'combined-narrative';\n    this.includeStepNumbers = options?.includeStepNumbers ?? true;\n    this.includeValues = options?.includeValues ?? true;\n    this.maxValueLength = options?.maxValueLength ?? 80;\n  }\n\n  // ── Scope channel (fires first, during stage execution) ───────────────\n\n  onRead(event: ReadEvent): void {\n    if (!event.key) return;\n    this.bufferOp(event.stageName, {\n      type: 'read',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n    });\n  }\n\n  onWrite(event: WriteEvent): void {\n    this.bufferOp(event.stageName, {\n      type: 'write',\n      key: event.key,\n      valueSummary: summarizeValue(event.value, this.maxValueLength),\n      operation: event.operation,\n    });\n  }\n\n  // ── Flow channel (fires after stage execution) ────────────────────────\n\n  onStageExecuted(event: FlowStageEvent): void {\n    const stageId = event.traversalContext?.stageId;\n\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const text = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.stageName}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.stageName}.`;\n\n    const sfId = event.traversalContext?.subflowId;\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId, stageId);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const deciderStageIdEarly = event.traversalContext?.stageId;\n\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n    const stageText = isFirst\n      ? event.description\n        ? `The process began: ${event.description}.`\n        : `The process began with ${event.decider}.`\n      : event.description\n      ? `Next step: ${event.description}.`\n      : `Next, it moved on to ${event.decider}.`;\n\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${stageNum}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageIdEarly,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider, event.traversalContext?.subflowId, deciderStageIdEarly);\n\n    // Emit the condition entry — with evidence-aware rendering when available\n    const branchName = event.chosen;\n    let conditionText: string;\n    if (event.evidence) {\n      // Rich evidence from decide() helper\n      const matchedRule = event.evidence.rules.find((r) => r.matched);\n      if (matchedRule) {\n        const label = matchedRule.label ? ` \"${matchedRule.label}\"` : '';\n        if (matchedRule.type === 'filter') {\n          const parts = matchedRule.conditions.map(\n            (c) =>\n              `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\\u2713' : '\\u2717'}`,\n          );\n          conditionText = `It evaluated Rule ${matchedRule.ruleIndex}${label}: ${parts.join(\n            ', ',\n          )}, and chose ${branchName}.`;\n        } else {\n          const parts = matchedRule.inputs.map((i) => `${i.key}=${i.valueSummary}`);\n          conditionText = `It examined${label}: ${parts.join(', ')}, and chose ${branchName}.`;\n        }\n      } else {\n        const erroredCount = event.evidence.rules.filter((r) => r.matchError !== undefined).length;\n        const errorNote = erroredCount > 0 ? ` (${erroredCount} rule${erroredCount > 1 ? 's' : ''} threw errors)` : '';\n        conditionText = `No rules matched${errorNote}, fell back to default: ${branchName}.`;\n      }\n    } else if (event.description && event.rationale) {\n      conditionText = `It ${event.description}: ${event.rationale}, so it chose ${branchName}.`;\n    } else if (event.description) {\n      conditionText = `It ${event.description} and chose ${branchName}.`;\n    } else if (event.rationale) {\n      conditionText = `A decision was made: ${event.rationale}, so the path taken was ${branchName}.`;\n    } else {\n      conditionText = `A decision was made, and the path taken was ${branchName}.`;\n    }\n    this.entries.push({\n      type: 'condition',\n      text: `[Condition]: ${conditionText}`,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageIdEarly,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onNext(): void {\n    // No-op. onStageExecuted already has the description for the next stage.\n    // For deciders (no onStageExecuted), onDecision handles the announcement.\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const names = event.children.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Parallel]: Forking into ${event.children.length} parallel paths: ${names}.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    let text: string;\n    if (event.evidence) {\n      const matched = event.evidence.rules.filter((r) => r.matched);\n      const parts = matched.map((r) => {\n        const label = r.label ? ` \"${r.label}\"` : '';\n        if (r.type === 'filter') {\n          const conds = r.conditions\n            .map(\n              (c) =>\n                `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\\u2713' : '\\u2717'}`,\n            )\n            .join(', ');\n          return `${r.branch}${label} (${conds})`;\n        }\n        const inputs = r.inputs.map((i) => `${i.key}=${i.valueSummary}`).join(', ');\n        return `${r.branch}${label} (${inputs})`;\n      });\n      text = `[Selected]: ${event.selected.length} of ${event.total} paths selected: ${parts.join('; ')}.`;\n    } else {\n      const names = event.selected.join(', ');\n      text = `[Selected]: ${event.selected.length} of ${event.total} paths selected for execution: ${names}.`;\n    }\n    this.entries.push({\n      type: 'selector',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    // Reset stage counter for this subflow so stages start at \"Stage 1\" on re-entry\n    const sfKey = event.subflowId ?? '';\n    this.stageCounters.delete(sfKey);\n    this.firstStageFlags.delete(sfKey);\n\n    const text = event.description\n      ? `Entering the ${event.name} subflow: ${event.description}.`\n      : `Entering the ${event.name} subflow.`;\n    this.entries.push({\n      type: 'subflow',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    this.entries.push({\n      type: 'subflow',\n      text: `Exiting the ${event.name} subflow.`,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    const text = event.description\n      ? `On pass ${event.iteration}: ${event.description} again.`\n      : `On pass ${event.iteration} through ${event.target}.`;\n    this.entries.push({\n      type: 'loop',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    this.entries.push({\n      type: 'break',\n      text: `Execution stopped at ${event.stageName}.`,\n      depth: 0,\n      stageName: event.stageName,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  /**\n   * Handles errors from both channels:\n   * - FlowRecorder.onError (FlowErrorEvent with message + structuredError)\n   * - Recorder.onError (ErrorEvent from scope system — ignored for narrative)\n   */\n  onError(event: FlowErrorEvent | { stageName?: string; message?: string }): void {\n    // Only handle flow errors (which have `message` and `structuredError`)\n    if (typeof (event as FlowErrorEvent).message !== 'string') return;\n    const flowEvent = event as FlowErrorEvent;\n\n    let text = `An error occurred at ${flowEvent.stageName}: ${flowEvent.message}.`;\n    if (flowEvent.structuredError?.issues?.length) {\n      const details = flowEvent.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n      text += ` Validation issues: ${details}.`;\n    }\n    this.entries.push({\n      type: 'error',\n      text: `[Error]: ${text}`,\n      depth: 0,\n      stageName: flowEvent.stageName,\n      stageId: flowEvent.traversalContext?.stageId,\n      subflowId: flowEvent.traversalContext?.subflowId,\n    });\n  }\n\n  // ── Output ────────────────────────────────────────────────────────────\n\n  /** Returns structured entries for programmatic consumption. */\n  getEntries(): CombinedNarrativeEntry[] {\n    return [...this.entries];\n  }\n\n  /** Returns formatted narrative lines (same output as CombinedNarrativeBuilder.build). */\n  getNarrative(indent = '  '): string[] {\n    return this.entries.map((entry) => `${indent.repeat(entry.depth)}${entry.text}`);\n  }\n\n  /**\n   * Returns entries grouped by subflowId for structured access.\n   * Root-level entries have subflowId = undefined.\n   */\n  getEntriesBySubflow(): Record<string, CombinedNarrativeEntry[]> {\n    const result: Record<string, CombinedNarrativeEntry[]> = { '': [] };\n    for (const entry of this.entries) {\n      const key = entry.subflowId ?? '';\n      if (!result[key]) result[key] = [];\n      result[key].push(entry);\n    }\n    return result;\n  }\n\n  /** Clears all state. Called automatically before each run. */\n  clear(): void {\n    this.entries = [];\n    this.pendingOps.clear();\n    this.stageCounters.clear();\n    this.firstStageFlags.clear();\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\n\n  /** Increment and return the stage counter for a given subflow ('' = root). */\n  private incrementStageCounter(subflowKey: string): number {\n    const current = this.stageCounters.get(subflowKey) ?? 0;\n    const next = current + 1;\n    this.stageCounters.set(subflowKey, next);\n    return next;\n  }\n\n  /** Returns true if this is the first stage for the given subflow, consuming the flag. */\n  private consumeFirstStageFlag(subflowKey: string): boolean {\n    if (!this.firstStageFlags.has(subflowKey)) {\n      this.firstStageFlags.set(subflowKey, false);\n      return true;\n    }\n    return false;\n  }\n\n  private bufferOp(stageName: string, op: Omit<BufferedOp, 'stepNumber'>): void {\n    let ops = this.pendingOps.get(stageName);\n    if (!ops) {\n      ops = [];\n      this.pendingOps.set(stageName, ops);\n    }\n    ops.push({ ...op, stepNumber: ops.length + 1 });\n  }\n\n  private flushOps(stageName: string, subflowId?: string, stageId?: string): void {\n    const ops = this.pendingOps.get(stageName);\n    if (!ops || ops.length === 0) return;\n\n    for (const op of ops) {\n      const stepPrefix = this.includeStepNumbers ? `Step ${op.stepNumber}: ` : '';\n\n      let text: string;\n      if (op.type === 'read') {\n        text =\n          this.includeValues && op.valueSummary\n            ? `${stepPrefix}Read ${op.key} = ${op.valueSummary}`\n            : `${stepPrefix}Read ${op.key}`;\n      } else if (op.operation === 'delete') {\n        text = `${stepPrefix}Delete ${op.key}`;\n      } else if (op.operation === 'update') {\n        text = this.includeValues\n          ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Update ${op.key}`;\n      } else {\n        text = this.includeValues\n          ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Write ${op.key}`;\n      }\n\n      this.entries.push({\n        type: 'step',\n        text,\n        depth: 1,\n        stageName,\n        stageId,\n        stepNumber: op.stepNumber,\n        subflowId,\n      });\n    }\n\n    this.pendingOps.delete(stageName);\n  }\n}\n"]}
463
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeRecorder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/CombinedNarrativeRecorder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeRecorder — Inline narrative builder that merges flow + data during traversal.\n *\n * Replaces the post-processing CombinedNarrativeBuilder by implementing BOTH\n * FlowRecorder (control-flow events) and Recorder (scope data events).\n *\n * Event ordering guarantees this works:\n *   1. Scope events (onRead, onWrite) fire DURING stage execution\n *   2. Flow events (onStageExecuted, onDecision) fire AFTER stage execution\n *   3. Both carry the same `stageName` — no matching ambiguity\n *\n * So we buffer scope ops per-stage, then when the flow event arrives,\n * emit the stage entry + flush the buffered ops in one pass.\n */\n\nimport { summarizeValue } from '../../scope/recorders/summarizeValue.js';\nimport type { ReadEvent, Recorder, WriteEvent } from '../../scope/types.js';\nimport type {\n  BreakRenderContext,\n  CombinedNarrativeEntry,\n  DecisionRenderContext,\n  ErrorRenderContext,\n  ForkRenderContext,\n  LoopRenderContext,\n  NarrativeRenderer,\n  OpRenderContext,\n  SelectedRenderContext,\n  StageRenderContext,\n  SubflowRenderContext,\n} from './narrativeTypes.js';\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types.js';\n\n// ── Types ──────────────────────────────────────────────────────────────────\n\ninterface BufferedOp {\n  type: 'read' | 'write';\n  key: string;\n  rawValue: unknown;\n  operation?: 'set' | 'update' | 'delete';\n  stepNumber: number;\n}\n\nexport interface CombinedNarrativeRecorderOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  maxValueLength?: number;\n  /** Custom value formatter. Called at render time (flushOps), not capture time.\n   *  Receives the raw value and maxValueLength. Defaults to summarizeValue(). */\n  formatValue?: (value: unknown, maxLen: number) => string;\n  /** Pluggable renderer for customizing narrative output. Unimplemented methods\n   *  fall back to the default English renderer. See NarrativeRenderer docs. */\n  renderer?: NarrativeRenderer;\n}\n\n// ── Recorder ───────────────────────────────────────────────────────────────\n\nexport class CombinedNarrativeRecorder implements FlowRecorder, Recorder {\n  readonly id: string;\n\n  private entries: CombinedNarrativeEntry[] = [];\n  /**\n   * Pending scope ops keyed by stageName. Flushed in onStageExecuted/onDecision.\n   *\n   * Name collisions (two stages with the same name, different IDs) are prevented by\n   * the event ordering contract: scope events (onRead/onWrite) for stage N are always\n   * flushed by onStageExecuted for stage N before stage N+1's scope events begin.\n   * So the key is always uniquely bound to the currently-executing stage.\n   */\n  private pendingOps = new Map<string, BufferedOp[]>();\n  /** Per-subflow stage counters. Key '' = root flow. */\n  private stageCounters = new Map<string, number>();\n  /** Per-subflow first-stage flags. Key '' = root flow. */\n  private firstStageFlags = new Map<string, boolean>();\n\n  private includeStepNumbers: boolean;\n  private includeValues: boolean;\n  private maxValueLength: number;\n  private formatValue: (value: unknown, maxLen: number) => string;\n  private renderer?: NarrativeRenderer;\n\n  constructor(options?: CombinedNarrativeRecorderOptions & { id?: string }) {\n    this.id = options?.id ?? 'combined-narrative';\n    this.includeStepNumbers = options?.includeStepNumbers ?? true;\n    this.includeValues = options?.includeValues ?? true;\n    this.maxValueLength = options?.maxValueLength ?? 80;\n    this.formatValue = options?.formatValue ?? summarizeValue;\n    this.renderer = options?.renderer;\n  }\n\n  // ── Scope channel (fires first, during stage execution) ───────────────\n\n  onRead(event: ReadEvent): void {\n    if (!event.key) return;\n    this.bufferOp(event.stageName, {\n      type: 'read',\n      key: event.key,\n      rawValue: event.value,\n    });\n  }\n\n  onWrite(event: WriteEvent): void {\n    this.bufferOp(event.stageName, {\n      type: 'write',\n      key: event.key,\n      rawValue: event.value,\n      operation: event.operation,\n    });\n  }\n\n  // ── Flow channel (fires after stage execution) ────────────────────────\n\n  onStageExecuted(event: FlowStageEvent): void {\n    const stageId = event.traversalContext?.stageId;\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n\n    const ctx: StageRenderContext = {\n      stageName: event.stageName,\n      stageNumber: stageNum,\n      isFirst,\n      description: event.description,\n    };\n    const text = this.renderer?.renderStage?.(ctx) ?? this.defaultRenderStage(ctx);\n\n    const sfId = event.traversalContext?.subflowId;\n    this.entries.push({\n      type: 'stage',\n      text,\n      depth: 0,\n      stageName: event.stageName,\n      stageId,\n      subflowId: sfId,\n    });\n    this.flushOps(event.stageName, sfId, stageId);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    const deciderStageIdEarly = event.traversalContext?.stageId;\n\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    const sfKey = event.traversalContext?.subflowId ?? '';\n    const stageNum = this.incrementStageCounter(sfKey);\n    const isFirst = this.consumeFirstStageFlag(sfKey);\n\n    const stageCtx: StageRenderContext = {\n      stageName: event.decider,\n      stageNumber: stageNum,\n      isFirst,\n      description: event.description,\n    };\n    const stageText = this.renderer?.renderStage?.(stageCtx) ?? this.defaultRenderStage(stageCtx);\n\n    this.entries.push({\n      type: 'stage',\n      text: stageText,\n      depth: 0,\n      stageName: event.decider,\n      stageId: deciderStageIdEarly,\n      subflowId: event.traversalContext?.subflowId,\n    });\n    this.flushOps(event.decider, event.traversalContext?.subflowId, deciderStageIdEarly);\n\n    // Emit the condition entry as a nested sub-item (depth 1) of the stage above.\n    // Decision outcome is a detail of the decider stage, not a separate top-level entry.\n    const decisionCtx: DecisionRenderContext = {\n      decider: event.decider,\n      chosen: event.chosen,\n      description: event.description,\n      rationale: event.rationale,\n      evidence: event.evidence,\n    };\n    const conditionText = this.renderer?.renderDecision?.(decisionCtx) ?? this.defaultRenderDecision(decisionCtx);\n    this.entries.push({\n      type: 'condition',\n      text: conditionText,\n      depth: 1,\n      stageName: event.decider,\n      stageId: deciderStageIdEarly,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onNext(): void {\n    // No-op. onStageExecuted already has the description for the next stage.\n    // For deciders (no onStageExecuted), onDecision handles the announcement.\n  }\n\n  onFork(event: FlowForkEvent): void {\n    const ctx: ForkRenderContext = { children: event.children };\n    const text = this.renderer?.renderFork?.(ctx) ?? this.defaultRenderFork(ctx);\n    this.entries.push({\n      type: 'fork',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const ctx: SelectedRenderContext = {\n      selected: event.selected,\n      total: event.total,\n      evidence: event.evidence,\n    };\n    const text = this.renderer?.renderSelected?.(ctx) ?? this.defaultRenderSelected(ctx);\n    this.entries.push({\n      type: 'selector',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    // Reset stage counter for this subflow so stages start at \"Stage 1\" on re-entry\n    const sfKey = event.subflowId ?? '';\n    this.stageCounters.delete(sfKey);\n    this.firstStageFlags.delete(sfKey);\n\n    const ctx: SubflowRenderContext = {\n      name: event.name,\n      direction: 'entry',\n      description: event.description,\n    };\n    const text = this.renderer?.renderSubflow?.(ctx) ?? this.defaultRenderSubflow(ctx);\n    this.entries.push({\n      type: 'subflow',\n      text,\n      depth: 0,\n      stageName: event.name,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onSubflowExit(event: FlowSubflowEvent): void {\n    const ctx: SubflowRenderContext = {\n      name: event.name,\n      direction: 'exit',\n    };\n    const text = this.renderer?.renderSubflow?.(ctx) ?? this.defaultRenderSubflow(ctx);\n    this.entries.push({\n      type: 'subflow',\n      text,\n      depth: 0,\n      stageName: event.name,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onLoop(event: FlowLoopEvent): void {\n    const ctx: LoopRenderContext = {\n      target: event.target,\n      iteration: event.iteration,\n      description: event.description,\n    };\n    const text = this.renderer?.renderLoop?.(ctx) ?? this.defaultRenderLoop(ctx);\n    this.entries.push({\n      type: 'loop',\n      text,\n      depth: 0,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  onBreak(event: FlowBreakEvent): void {\n    const ctx: BreakRenderContext = { stageName: event.stageName };\n    const text = this.renderer?.renderBreak?.(ctx) ?? this.defaultRenderBreak(ctx);\n    this.entries.push({\n      type: 'break',\n      text,\n      depth: 0,\n      stageName: event.stageName,\n      stageId: event.traversalContext?.stageId,\n      subflowId: event.traversalContext?.subflowId,\n    });\n  }\n\n  /**\n   * Handles errors from both channels:\n   * - FlowRecorder.onError (FlowErrorEvent with message + structuredError)\n   * - Recorder.onError (ErrorEvent from scope system — ignored for narrative)\n   */\n  onError(event: FlowErrorEvent | { stageName?: string; message?: string }): void {\n    // Only handle flow errors (which have `message` and `structuredError`)\n    if (typeof (event as FlowErrorEvent).message !== 'string') return;\n    const flowEvent = event as FlowErrorEvent;\n\n    let validationIssues: string | undefined;\n    if (flowEvent.structuredError?.issues?.length) {\n      validationIssues = flowEvent.structuredError.issues\n        .map((issue) => {\n          const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';\n          return `${path}: ${issue.message}`;\n        })\n        .join('; ');\n    }\n\n    const ctx: ErrorRenderContext = {\n      stageName: flowEvent.stageName,\n      message: flowEvent.message,\n      validationIssues,\n    };\n    const text = this.renderer?.renderError?.(ctx) ?? this.defaultRenderError(ctx);\n    this.entries.push({\n      type: 'error',\n      text,\n      depth: 0,\n      stageName: flowEvent.stageName,\n      stageId: flowEvent.traversalContext?.stageId,\n      subflowId: flowEvent.traversalContext?.subflowId,\n    });\n  }\n\n  // ── Output ────────────────────────────────────────────────────────────\n\n  /** Returns structured entries for programmatic consumption. */\n  getEntries(): CombinedNarrativeEntry[] {\n    return [...this.entries];\n  }\n\n  /** Returns formatted narrative lines (same output as CombinedNarrativeBuilder.build). */\n  getNarrative(indent = '  '): string[] {\n    return this.entries.map((entry) => `${indent.repeat(entry.depth)}${entry.text}`);\n  }\n\n  /**\n   * Returns entries grouped by subflowId for structured access.\n   * Root-level entries have subflowId = undefined.\n   */\n  getEntriesBySubflow(): Record<string, CombinedNarrativeEntry[]> {\n    const result: Record<string, CombinedNarrativeEntry[]> = { '': [] };\n    for (const entry of this.entries) {\n      const key = entry.subflowId ?? '';\n      if (!result[key]) result[key] = [];\n      result[key].push(entry);\n    }\n    return result;\n  }\n\n  /** Clears all state. Called automatically before each run. */\n  clear(): void {\n    this.entries = [];\n    this.pendingOps.clear();\n    this.stageCounters.clear();\n    this.firstStageFlags.clear();\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\n\n  /** Increment and return the stage counter for a given subflow ('' = root). */\n  private incrementStageCounter(subflowKey: string): number {\n    const current = this.stageCounters.get(subflowKey) ?? 0;\n    const next = current + 1;\n    this.stageCounters.set(subflowKey, next);\n    return next;\n  }\n\n  /** Returns true if this is the first stage for the given subflow, consuming the flag. */\n  private consumeFirstStageFlag(subflowKey: string): boolean {\n    if (!this.firstStageFlags.has(subflowKey)) {\n      this.firstStageFlags.set(subflowKey, false);\n      return true;\n    }\n    return false;\n  }\n\n  private bufferOp(stageName: string, op: Omit<BufferedOp, 'stepNumber'>): void {\n    let ops = this.pendingOps.get(stageName);\n    if (!ops) {\n      ops = [];\n      this.pendingOps.set(stageName, ops);\n    }\n    ops.push({ ...op, stepNumber: ops.length + 1 });\n  }\n\n  private flushOps(stageName: string, subflowId?: string, stageId?: string): void {\n    const ops = this.pendingOps.get(stageName);\n    if (!ops || ops.length === 0) return;\n\n    for (const op of ops) {\n      const valueSummary = this.formatValue(op.rawValue, this.maxValueLength);\n      const opCtx: OpRenderContext = {\n        type: op.type,\n        key: op.key,\n        rawValue: op.rawValue,\n        valueSummary,\n        operation: op.operation,\n        stepNumber: op.stepNumber,\n      };\n\n      const text = this.renderer?.renderOp ? this.renderer.renderOp(opCtx) : this.defaultRenderOp(opCtx);\n\n      if (text == null) continue; // renderer excluded this op (null or undefined)\n\n      this.entries.push({\n        type: 'step',\n        text,\n        depth: 1,\n        stageName,\n        stageId,\n        stepNumber: op.stepNumber,\n        subflowId,\n        rawValue: op.rawValue,\n      });\n    }\n\n    this.pendingOps.delete(stageName);\n  }\n\n  // ── Default renderers (used when no custom renderer is provided) ────\n\n  private defaultRenderStage(ctx: StageRenderContext): string {\n    const inner = ctx.isFirst\n      ? ctx.description\n        ? `The process began: ${ctx.description}.`\n        : `The process began with ${ctx.stageName}.`\n      : ctx.description\n      ? `Next step: ${ctx.description}.`\n      : `Next, it moved on to ${ctx.stageName}.`;\n    return `Stage ${ctx.stageNumber}: ${inner}`;\n  }\n\n  private defaultRenderOp(ctx: OpRenderContext): string {\n    const stepPrefix = this.includeStepNumbers ? `Step ${ctx.stepNumber}: ` : '';\n    if (ctx.type === 'read') {\n      return this.includeValues && ctx.valueSummary\n        ? `${stepPrefix}Read ${ctx.key} = ${ctx.valueSummary}`\n        : `${stepPrefix}Read ${ctx.key}`;\n    }\n    if (ctx.operation === 'delete') {\n      return `${stepPrefix}Delete ${ctx.key}`;\n    }\n    if (ctx.operation === 'update') {\n      return this.includeValues\n        ? `${stepPrefix}Update ${ctx.key} = ${ctx.valueSummary}`\n        : `${stepPrefix}Update ${ctx.key}`;\n    }\n    return this.includeValues ? `${stepPrefix}Write ${ctx.key} = ${ctx.valueSummary}` : `${stepPrefix}Write ${ctx.key}`;\n  }\n\n  private defaultRenderDecision(ctx: DecisionRenderContext): string {\n    const branchName = ctx.chosen;\n    let conditionText: string;\n    if (ctx.evidence) {\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      const evidence = ctx.evidence as any;\n      const matchedRule = evidence.rules?.find((r: any) => r.matched);\n      if (matchedRule) {\n        const label = matchedRule.label ? ` \"${matchedRule.label}\"` : '';\n        if (matchedRule.type === 'filter') {\n          const parts = matchedRule.conditions.map(\n            (c: any) =>\n              `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\\u2713' : '\\u2717'}`,\n          );\n          conditionText = `It evaluated Rule ${matchedRule.ruleIndex}${label}: ${parts.join(\n            ', ',\n          )}, and chose ${branchName}.`;\n        } else {\n          const parts = matchedRule.inputs.map((i: any) => `${i.key}=${i.valueSummary}`);\n          conditionText = `It examined${label}: ${parts.join(', ')}, and chose ${branchName}.`;\n        }\n      } else {\n        const erroredCount = evidence.rules?.filter((r: any) => r.matchError !== undefined).length ?? 0;\n        const errorNote = erroredCount > 0 ? ` (${erroredCount} rule${erroredCount > 1 ? 's' : ''} threw errors)` : '';\n        conditionText = `No rules matched${errorNote}, fell back to default: ${branchName}.`;\n      }\n    } else if (ctx.description && ctx.rationale) {\n      conditionText = `It ${ctx.description}: ${ctx.rationale}, so it chose ${branchName}.`;\n    } else if (ctx.description) {\n      conditionText = `It ${ctx.description} and chose ${branchName}.`;\n    } else if (ctx.rationale) {\n      conditionText = `A decision was made: ${ctx.rationale}, so the path taken was ${branchName}.`;\n    } else {\n      conditionText = `A decision was made, and the path taken was ${branchName}.`;\n    }\n    return `[Condition]: ${conditionText}`;\n  }\n\n  private defaultRenderFork(ctx: ForkRenderContext): string {\n    const names = ctx.children.join(', ');\n    return `[Parallel]: Forking into ${ctx.children.length} parallel paths: ${names}.`;\n  }\n\n  private defaultRenderSelected(ctx: SelectedRenderContext): string {\n    if (ctx.evidence) {\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      const evidence = ctx.evidence as any;\n      const matched = evidence.rules?.filter((r: any) => r.matched) ?? [];\n      const parts = matched.map((r: any) => {\n        const label = r.label ? ` \"${r.label}\"` : '';\n        if (r.type === 'filter') {\n          const conds = r.conditions\n            .map(\n              (c: any) =>\n                `${c.key} ${c.actualSummary} ${c.op} ${JSON.stringify(c.threshold)} ${c.result ? '\\u2713' : '\\u2717'}`,\n            )\n            .join(', ');\n          return `${r.branch}${label} (${conds})`;\n        }\n        const inputs = r.inputs.map((i: any) => `${i.key}=${i.valueSummary}`).join(', ');\n        return `${r.branch}${label} (${inputs})`;\n      });\n      return `[Selected]: ${ctx.selected.length} of ${ctx.total} paths selected: ${parts.join('; ')}.`;\n    }\n    const names = ctx.selected.join(', ');\n    return `[Selected]: ${ctx.selected.length} of ${ctx.total} paths selected for execution: ${names}.`;\n  }\n\n  private defaultRenderSubflow(ctx: SubflowRenderContext): string {\n    if (ctx.direction === 'exit') {\n      return `Exiting the ${ctx.name} subflow.`;\n    }\n    return ctx.description\n      ? `Entering the ${ctx.name} subflow: ${ctx.description}.`\n      : `Entering the ${ctx.name} subflow.`;\n  }\n\n  private defaultRenderLoop(ctx: LoopRenderContext): string {\n    return ctx.description\n      ? `On pass ${ctx.iteration}: ${ctx.description} again.`\n      : `On pass ${ctx.iteration} through ${ctx.target}.`;\n  }\n\n  private defaultRenderBreak(ctx: BreakRenderContext): string {\n    return `Execution stopped at ${ctx.stageName}.`;\n  }\n\n  private defaultRenderError(ctx: ErrorRenderContext): string {\n    let text = `An error occurred at ${ctx.stageName}: ${ctx.message}.`;\n    if (ctx.validationIssues) {\n      text += ` Validation issues: ${ctx.validationIssues}.`;\n    }\n    return `[Error]: ${text}`;\n  }\n}\n"]}