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.
- package/CLAUDE.md +7 -0
- package/dist/esm/index.js +6 -1
- package/dist/esm/lib/builder/FlowChartBuilder.js +13 -8
- package/dist/esm/lib/engine/graph/StageNode.js +1 -1
- package/dist/esm/lib/engine/handlers/DeciderHandler.js +8 -7
- package/dist/esm/lib/engine/handlers/SelectorHandler.js +8 -5
- package/dist/esm/lib/engine/handlers/SubflowExecutor.js +28 -246
- package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +232 -139
- package/dist/esm/lib/engine/narrative/index.js +1 -1
- package/dist/esm/lib/engine/narrative/narrativeTypes.js +1 -1
- package/dist/esm/lib/engine/traversal/FlowchartTraverser.js +64 -22
- package/dist/esm/lib/engine/types.js +1 -1
- package/dist/esm/lib/memory/StageContext.js +36 -9
- package/dist/esm/lib/reactive/createTypedScope.js +9 -6
- package/dist/esm/lib/reactive/types.js +1 -1
- package/dist/esm/lib/recorder/CompositeRecorder.js +172 -0
- package/dist/esm/lib/recorder/index.js +2 -0
- package/dist/esm/lib/runner/FlowChartExecutor.js +33 -3
- package/dist/esm/lib/scope/ScopeFacade.js +52 -4
- package/dist/esm/lib/scope/recorders/DebugRecorder.js +18 -2
- package/dist/esm/lib/scope/recorders/MetricRecorder.js +51 -4
- package/dist/esm/recorders.js +6 -5
- package/dist/index.js +21 -15
- package/dist/lib/builder/FlowChartBuilder.js +13 -8
- package/dist/lib/engine/graph/StageNode.js +1 -1
- package/dist/lib/engine/handlers/DeciderHandler.js +8 -7
- package/dist/lib/engine/handlers/SelectorHandler.js +8 -5
- package/dist/lib/engine/handlers/SubflowExecutor.js +27 -245
- package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +232 -139
- package/dist/lib/engine/narrative/index.js +1 -1
- package/dist/lib/engine/narrative/narrativeTypes.js +1 -1
- package/dist/lib/engine/traversal/FlowchartTraverser.js +64 -22
- package/dist/lib/engine/types.js +1 -1
- package/dist/lib/memory/StageContext.js +36 -9
- package/dist/lib/reactive/createTypedScope.js +9 -6
- package/dist/lib/reactive/types.js +1 -1
- package/dist/lib/recorder/CompositeRecorder.js +176 -0
- package/dist/lib/recorder/index.js +6 -0
- package/dist/lib/runner/FlowChartExecutor.js +33 -3
- package/dist/lib/scope/ScopeFacade.js +52 -4
- package/dist/lib/scope/recorders/DebugRecorder.js +18 -2
- package/dist/lib/scope/recorders/MetricRecorder.js +51 -4
- package/dist/recorders.js +8 -6
- package/dist/types/index.d.ts +4 -0
- package/dist/types/lib/engine/graph/StageNode.d.ts +4 -0
- package/dist/types/lib/engine/handlers/SubflowExecutor.d.ts +10 -29
- package/dist/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +18 -1
- package/dist/types/lib/engine/narrative/index.d.ts +1 -1
- package/dist/types/lib/engine/narrative/narrativeTypes.d.ts +92 -0
- package/dist/types/lib/engine/traversal/FlowchartTraverser.d.ts +20 -8
- package/dist/types/lib/engine/types.d.ts +51 -0
- package/dist/types/lib/memory/StageContext.d.ts +14 -3
- package/dist/types/lib/reactive/types.d.ts +2 -0
- package/dist/types/lib/recorder/CompositeRecorder.d.ts +95 -0
- package/dist/types/lib/recorder/index.d.ts +2 -0
- package/dist/types/lib/runner/FlowChartExecutor.d.ts +27 -1
- package/dist/types/lib/scope/ScopeFacade.d.ts +11 -0
- package/dist/types/lib/scope/recorders/DebugRecorder.d.ts +16 -0
- package/dist/types/lib/scope/recorders/MetricRecorder.d.ts +47 -2
- package/dist/types/recorders.d.ts +9 -4
- 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
|
-
|
|
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
|
-
|
|
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
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
: event.description
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
const sfId = (
|
|
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
|
|
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
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
: event.description
|
|
94
|
-
|
|
95
|
-
|
|
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:
|
|
100
|
+
text: stageText,
|
|
99
101
|
depth: 0,
|
|
100
102
|
stageName: event.decider,
|
|
101
103
|
stageId: deciderStageIdEarly,
|
|
102
|
-
subflowId: (
|
|
104
|
+
subflowId: (_g = event.traversalContext) === null || _g === void 0 ? void 0 : _g.subflowId,
|
|
103
105
|
});
|
|
104
|
-
this.flushOps(event.decider, (
|
|
105
|
-
// Emit the condition entry
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
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:
|
|
143
|
-
depth:
|
|
119
|
+
text: conditionText,
|
|
120
|
+
depth: 1,
|
|
144
121
|
stageName: event.decider,
|
|
145
122
|
stageId: deciderStageIdEarly,
|
|
146
|
-
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
|
|
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
|
|
136
|
+
text,
|
|
159
137
|
depth: 0,
|
|
160
|
-
stageId: (
|
|
161
|
-
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
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
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: (
|
|
191
|
-
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
|
|
201
|
-
|
|
202
|
-
:
|
|
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
|
-
|
|
208
|
-
|
|
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
|
|
188
|
+
text,
|
|
216
189
|
depth: 0,
|
|
217
|
-
|
|
218
|
-
|
|
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
|
|
224
|
-
|
|
225
|
-
:
|
|
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: (
|
|
231
|
-
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
|
|
217
|
+
text,
|
|
239
218
|
depth: 0,
|
|
240
219
|
stageName: event.stageName,
|
|
241
|
-
stageId: (
|
|
242
|
-
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
|
|
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
|
-
|
|
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
|
|
252
|
+
text,
|
|
269
253
|
depth: 0,
|
|
270
254
|
stageName: flowEvent.stageName,
|
|
271
|
-
stageId: (
|
|
272
|
-
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
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
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"]}
|