footprintjs 0.10.0 → 0.10.1
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/dist/esm/lib/engine/narrative/CombinedNarrativeBuilder.js +43 -14
- package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +267 -0
- package/dist/esm/lib/engine/narrative/NarrativeFlowRecorder.js +25 -1
- package/dist/esm/lib/engine/narrative/index.js +2 -1
- package/dist/esm/lib/runner/FlowChartExecutor.js +33 -16
- package/dist/esm/lib/scope/recorders/NarrativeRecorder.js +6 -1
- package/dist/esm/types/lib/engine/narrative/CombinedNarrativeBuilder.d.ts +15 -2
- package/dist/esm/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +63 -0
- package/dist/esm/types/lib/engine/narrative/NarrativeFlowRecorder.d.ts +11 -0
- package/dist/esm/types/lib/engine/narrative/index.d.ts +2 -0
- package/dist/esm/types/lib/runner/FlowChartExecutor.d.ts +8 -1
- package/dist/esm/types/lib/scope/recorders/NarrativeRecorder.d.ts +5 -0
- package/dist/lib/engine/narrative/CombinedNarrativeBuilder.js +43 -14
- package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +271 -0
- package/dist/lib/engine/narrative/NarrativeFlowRecorder.js +25 -1
- package/dist/lib/engine/narrative/index.js +4 -2
- package/dist/lib/runner/FlowChartExecutor.js +33 -16
- package/dist/lib/scope/recorders/NarrativeRecorder.js +6 -1
- package/dist/types/lib/engine/narrative/CombinedNarrativeBuilder.d.ts +15 -2
- package/dist/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +63 -0
- package/dist/types/lib/engine/narrative/NarrativeFlowRecorder.d.ts +11 -0
- package/dist/types/lib/engine/narrative/index.d.ts +2 -0
- package/dist/types/lib/runner/FlowChartExecutor.d.ts +8 -1
- package/dist/types/lib/scope/recorders/NarrativeRecorder.d.ts +5 -0
- package/package.json +1 -1
|
@@ -8,6 +8,11 @@
|
|
|
8
8
|
* Stage 1: "Receive Application"
|
|
9
9
|
* Step 1: Write applicantName = 'Bob'
|
|
10
10
|
* [Condition]: risk tier is high → chose "Reject Application"
|
|
11
|
+
*
|
|
12
|
+
* @deprecated Since 0.9.x — Superseded by {@link CombinedNarrativeRecorder} which builds the
|
|
13
|
+
* combined narrative inline during traversal (no post-processing). This class is retained for
|
|
14
|
+
* backward compatibility with consumers that use CombinedNarrativeBuilder directly.
|
|
15
|
+
* Will be removed in v1.0.
|
|
11
16
|
*/
|
|
12
17
|
// ---------------------------------------------------------------------------
|
|
13
18
|
// Builder
|
|
@@ -21,23 +26,35 @@ export class CombinedNarrativeBuilder {
|
|
|
21
26
|
indent: (_c = options === null || options === void 0 ? void 0 : options.indent) !== null && _c !== void 0 ? _c : ' ',
|
|
22
27
|
};
|
|
23
28
|
}
|
|
24
|
-
|
|
29
|
+
/**
|
|
30
|
+
* @param flowSentences Sentences from the flow recorder (NarrativeFlowRecorder.getSentences()).
|
|
31
|
+
* @param recorder Data-level recorder with per-stage read/write operations.
|
|
32
|
+
* @param sentenceStageNames Optional parallel array of actual stage names for each sentence
|
|
33
|
+
* (from NarrativeFlowRecorder.getSentenceStageNames()). When provided,
|
|
34
|
+
* enables direct lookup instead of fragile regex-based matching.
|
|
35
|
+
* @deprecated Use {@link CombinedNarrativeRecorder} instead, which builds entries inline during traversal.
|
|
36
|
+
*/
|
|
37
|
+
buildEntries(flowSentences, recorder, sentenceStageNames) {
|
|
38
|
+
var _a;
|
|
25
39
|
const entries = [];
|
|
26
40
|
const stageData = recorder.getStageData();
|
|
27
41
|
const usedStages = new Set();
|
|
28
42
|
let stageCounter = 0;
|
|
29
|
-
for (
|
|
43
|
+
for (let i = 0; i < flowSentences.length; i++) {
|
|
44
|
+
const sentence = flowSentences[i];
|
|
30
45
|
const parsed = this.parseSentence(sentence);
|
|
46
|
+
// Prefer the actual stage name from the recorder over regex-parsed description text
|
|
47
|
+
const resolvedStageName = (_a = sentenceStageNames === null || sentenceStageNames === void 0 ? void 0 : sentenceStageNames[i]) !== null && _a !== void 0 ? _a : parsed.stageName;
|
|
31
48
|
if (parsed.type === 'stage') {
|
|
32
49
|
stageCounter++;
|
|
33
50
|
entries.push({
|
|
34
51
|
type: 'stage',
|
|
35
52
|
text: `Stage ${stageCounter}: ${sentence}`,
|
|
36
53
|
depth: 0,
|
|
37
|
-
stageName:
|
|
54
|
+
stageName: resolvedStageName,
|
|
38
55
|
});
|
|
39
|
-
if (
|
|
40
|
-
const data = this.findStageData(
|
|
56
|
+
if (resolvedStageName) {
|
|
57
|
+
const data = this.findStageData(resolvedStageName, stageData);
|
|
41
58
|
if (data) {
|
|
42
59
|
usedStages.add(data.stageName);
|
|
43
60
|
this.addStepEntries(entries, data);
|
|
@@ -45,7 +62,8 @@ export class CombinedNarrativeBuilder {
|
|
|
45
62
|
}
|
|
46
63
|
}
|
|
47
64
|
else if (parsed.type === 'condition') {
|
|
48
|
-
entries.push({ type: 'condition', text: `[Condition]: ${sentence}`, depth: 0 });
|
|
65
|
+
entries.push({ type: 'condition', text: `[Condition]: ${sentence}`, depth: 0, stageName: resolvedStageName });
|
|
66
|
+
// For conditions, also look up the chosen branch's data
|
|
49
67
|
if (parsed.stageName) {
|
|
50
68
|
const data = this.findStageData(parsed.stageName, stageData);
|
|
51
69
|
if (data) {
|
|
@@ -53,6 +71,17 @@ export class CombinedNarrativeBuilder {
|
|
|
53
71
|
this.addStepEntries(entries, data);
|
|
54
72
|
}
|
|
55
73
|
}
|
|
74
|
+
// Also mark the decider stage itself as used (the resolvedStageName)
|
|
75
|
+
if (resolvedStageName) {
|
|
76
|
+
const data = this.findStageData(resolvedStageName, stageData);
|
|
77
|
+
if (data) {
|
|
78
|
+
usedStages.add(data.stageName);
|
|
79
|
+
// Only add step entries if the chosen branch didn't already cover it
|
|
80
|
+
if (!parsed.stageName || parsed.stageName !== resolvedStageName) {
|
|
81
|
+
this.addStepEntries(entries, data);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
56
85
|
}
|
|
57
86
|
else if (parsed.type === 'fork') {
|
|
58
87
|
entries.push({ type: 'fork', text: `[Parallel]: ${sentence}`, depth: 0 });
|
|
@@ -80,8 +109,8 @@ export class CombinedNarrativeBuilder {
|
|
|
80
109
|
}
|
|
81
110
|
return entries;
|
|
82
111
|
}
|
|
83
|
-
build(flowSentences, recorder) {
|
|
84
|
-
return this.buildEntries(flowSentences, recorder).map((entry) => {
|
|
112
|
+
build(flowSentences, recorder, sentenceStageNames) {
|
|
113
|
+
return this.buildEntries(flowSentences, recorder, sentenceStageNames).map((entry) => {
|
|
85
114
|
const indent = this.options.indent.repeat(entry.depth);
|
|
86
115
|
return `${indent}${entry.text}`;
|
|
87
116
|
});
|
|
@@ -125,22 +154,22 @@ export class CombinedNarrativeBuilder {
|
|
|
125
154
|
return undefined;
|
|
126
155
|
}
|
|
127
156
|
parseSentence(sentence) {
|
|
128
|
-
var _a, _b, _c, _d;
|
|
157
|
+
var _a, _b, _c, _d, _e, _f;
|
|
129
158
|
if (sentence.startsWith('The process began')) {
|
|
130
159
|
const match = sentence.match(/The process began(?:: (.+)| with (.+))\./);
|
|
131
|
-
return { type: 'stage', stageName: (_a = match === null || match === void 0 ? void 0 : match[2]) === null ||
|
|
160
|
+
return { type: 'stage', stageName: (_b = ((_a = match === null || match === void 0 ? void 0 : match[1]) !== null && _a !== void 0 ? _a : match === null || match === void 0 ? void 0 : match[2])) === null || _b === void 0 ? void 0 : _b.trim() };
|
|
132
161
|
}
|
|
133
162
|
if (sentence.startsWith('Next')) {
|
|
134
163
|
const match = sentence.match(/Next(?:,? it moved on to (.+)| step: (.+))\./);
|
|
135
|
-
return { type: 'stage', stageName: (
|
|
164
|
+
return { type: 'stage', stageName: (_d = ((_c = match === null || match === void 0 ? void 0 : match[1]) !== null && _c !== void 0 ? _c : match === null || match === void 0 ? void 0 : match[2])) === null || _d === void 0 ? void 0 : _d.trim() };
|
|
136
165
|
}
|
|
137
166
|
if (sentence.includes('decision was made') || sentence.includes('it chose') || sentence.includes('so it chose')) {
|
|
138
167
|
const match = sentence.match(/chose (.+)\./);
|
|
139
|
-
return { type: 'condition', stageName: (
|
|
168
|
+
return { type: 'condition', stageName: (_e = match === null || match === void 0 ? void 0 : match[1]) === null || _e === void 0 ? void 0 : _e.trim() };
|
|
140
169
|
}
|
|
141
170
|
if (sentence.startsWith('It ') && sentence.includes('chose')) {
|
|
142
171
|
const match = sentence.match(/chose (.+)\./);
|
|
143
|
-
return { type: 'condition', stageName: (
|
|
172
|
+
return { type: 'condition', stageName: (_f = match === null || match === void 0 ? void 0 : match[1]) === null || _f === void 0 ? void 0 : _f.trim() };
|
|
144
173
|
}
|
|
145
174
|
if (sentence.includes('paths were executed in parallel') || sentence.includes('paths were selected'))
|
|
146
175
|
return { type: 'fork' };
|
|
@@ -155,4 +184,4 @@ export class CombinedNarrativeBuilder {
|
|
|
155
184
|
return { type: 'stage' };
|
|
156
185
|
}
|
|
157
186
|
}
|
|
158
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeBuilder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/CombinedNarrativeBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeBuilder — Merges flow-level narrative with data-level operations.\n *\n * ControlFlowNarrativeGenerator captures FLOW — what stages ran, which branches were taken.\n * NarrativeRecorder (scope/) captures DATA — what values were read, written, updated.\n *\n * This builder weaves both into a single unified narrative:\n *   Stage 1: \"Receive Application\"\n *     Step 1: Write applicantName = 'Bob'\n *   [Condition]: risk tier is high → chose \"Reject Application\"\n */\n\nimport type { NarrativeRecorder, StageNarrativeData } from '../../scope/recorders/NarrativeRecorder';\n\n// ---------------------------------------------------------------------------\n// Types\n// ---------------------------------------------------------------------------\n\nexport interface CombinedNarrativeEntry {\n  type: 'stage' | 'step' | 'condition' | 'fork' | 'subflow' | 'loop' | 'break' | 'error';\n  text: string;\n  depth: number;\n  stageName?: string;\n  stepNumber?: number;\n}\n\nexport interface CombinedNarrativeOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  indent?: string;\n}\n\n// ---------------------------------------------------------------------------\n// Builder\n// ---------------------------------------------------------------------------\n\nexport class CombinedNarrativeBuilder {\n  private options: Required<CombinedNarrativeOptions>;\n\n  constructor(options?: CombinedNarrativeOptions) {\n    this.options = {\n      includeStepNumbers: options?.includeStepNumbers ?? true,\n      includeValues: options?.includeValues ?? true,\n      indent: options?.indent ?? '  ',\n    };\n  }\n\n  buildEntries(flowSentences: string[], recorder: NarrativeRecorder): CombinedNarrativeEntry[] {\n    const entries: CombinedNarrativeEntry[] = [];\n    const stageData = recorder.getStageData();\n    const usedStages = new Set<string>();\n    let stageCounter = 0;\n\n    for (const sentence of flowSentences) {\n      const parsed = this.parseSentence(sentence);\n\n      if (parsed.type === 'stage') {\n        stageCounter++;\n        entries.push({\n          type: 'stage',\n          text: `Stage ${stageCounter}: ${sentence}`,\n          depth: 0,\n          stageName: parsed.stageName,\n        });\n        if (parsed.stageName) {\n          const data = this.findStageData(parsed.stageName, stageData);\n          if (data) {\n            usedStages.add(data.stageName);\n            this.addStepEntries(entries, data);\n          }\n        }\n      } else if (parsed.type === 'condition') {\n        entries.push({ type: 'condition', text: `[Condition]: ${sentence}`, depth: 0 });\n        if (parsed.stageName) {\n          const data = this.findStageData(parsed.stageName, stageData);\n          if (data) {\n            usedStages.add(data.stageName);\n            this.addStepEntries(entries, data);\n          }\n        }\n      } else if (parsed.type === 'fork') {\n        entries.push({ type: 'fork', text: `[Parallel]: ${sentence}`, depth: 0 });\n      } else if (parsed.type === 'subflow') {\n        entries.push({ type: 'subflow', text: sentence, depth: 0 });\n      } else if (parsed.type === 'loop') {\n        entries.push({ type: 'loop', text: sentence, depth: 0 });\n      } else if (parsed.type === 'break') {\n        entries.push({ type: 'break', text: sentence, depth: 0 });\n      } else if (parsed.type === 'error') {\n        entries.push({ type: 'error', text: `[Error]: ${sentence}`, depth: 0 });\n      }\n    }\n\n    // Add stages with operations that weren't referenced in flow sentences\n    for (const [stageName, data] of Array.from(stageData.entries())) {\n      if (!usedStages.has(stageName) && data.operations.length > 0) {\n        stageCounter++;\n        entries.push({ type: 'stage', text: `Stage ${stageCounter}: ${stageName}`, depth: 0, stageName });\n        this.addStepEntries(entries, data);\n      }\n    }\n\n    return entries;\n  }\n\n  build(flowSentences: string[], recorder: NarrativeRecorder): string[] {\n    return this.buildEntries(flowSentences, recorder).map((entry) => {\n      const indent = this.options.indent.repeat(entry.depth);\n      return `${indent}${entry.text}`;\n    });\n  }\n\n  // ── Private helpers ──\n\n  private addStepEntries(entries: CombinedNarrativeEntry[], data: StageNarrativeData): void {\n    for (const op of data.operations) {\n      const stepPrefix = this.options.includeStepNumbers && op.stepNumber ? `Step ${op.stepNumber}: ` : '';\n\n      let text: string;\n      if (op.type === 'read') {\n        text =\n          this.options.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.options.includeValues\n          ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Update ${op.key}`;\n      } else {\n        text = this.options.includeValues\n          ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Write ${op.key}`;\n      }\n\n      entries.push({ type: 'step', text, depth: 1, stageName: data.stageName, stepNumber: op.stepNumber });\n    }\n  }\n\n  private findStageData(name: string, stageData: Map<string, StageNarrativeData>): StageNarrativeData | undefined {\n    if (stageData.has(name)) return stageData.get(name);\n    for (const [stageName, data] of Array.from(stageData.entries())) {\n      if (\n        stageName.toLowerCase().includes(name.toLowerCase()) ||\n        name.toLowerCase().includes(stageName.toLowerCase())\n      ) {\n        return data;\n      }\n    }\n    return undefined;\n  }\n\n  private parseSentence(sentence: string): { type: string; stageName?: string } {\n    if (sentence.startsWith('The process began')) {\n      const match = sentence.match(/The process began(?:: (.+)| with (.+))\\./);\n      return { type: 'stage', stageName: match?.[2]?.trim() };\n    }\n    if (sentence.startsWith('Next')) {\n      const match = sentence.match(/Next(?:,? it moved on to (.+)| step: (.+))\\./);\n      return { type: 'stage', stageName: match?.[1]?.trim() };\n    }\n    if (sentence.includes('decision was made') || sentence.includes('it chose') || sentence.includes('so it chose')) {\n      const match = sentence.match(/chose (.+)\\./);\n      return { type: 'condition', stageName: match?.[1]?.trim() };\n    }\n    if (sentence.startsWith('It ') && sentence.includes('chose')) {\n      const match = sentence.match(/chose (.+)\\./);\n      return { type: 'condition', stageName: match?.[1]?.trim() };\n    }\n    if (sentence.includes('paths were executed in parallel') || sentence.includes('paths were selected'))\n      return { type: 'fork' };\n    if (sentence.startsWith('Entering') || sentence.startsWith('Exiting')) return { type: 'subflow' };\n    if (sentence.startsWith('On pass')) return { type: 'loop' };\n    if (sentence.startsWith('Execution stopped')) return { type: 'break' };\n    if (sentence.startsWith('An error occurred')) return { type: 'error' };\n    return { type: 'stage' };\n  }\n}\n"]}
|
|
187
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CombinedNarrativeBuilder.js","sourceRoot":"","sources":["../../../../../src/lib/engine/narrative/CombinedNarrativeBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * CombinedNarrativeBuilder — Merges flow-level narrative with data-level operations.\n *\n * ControlFlowNarrativeGenerator captures FLOW — what stages ran, which branches were taken.\n * NarrativeRecorder (scope/) captures DATA — what values were read, written, updated.\n *\n * This builder weaves both into a single unified narrative:\n *   Stage 1: \"Receive Application\"\n *     Step 1: Write applicantName = 'Bob'\n *   [Condition]: risk tier is high → chose \"Reject Application\"\n *\n * @deprecated Since 0.9.x — Superseded by {@link CombinedNarrativeRecorder} which builds the\n * combined narrative inline during traversal (no post-processing). This class is retained for\n * backward compatibility with consumers that use CombinedNarrativeBuilder directly.\n * Will be removed in v1.0.\n */\n\nimport type { NarrativeRecorder, StageNarrativeData } from '../../scope/recorders/NarrativeRecorder';\n\n// ---------------------------------------------------------------------------\n// Types\n// ---------------------------------------------------------------------------\n\nexport interface CombinedNarrativeEntry {\n  type: 'stage' | 'step' | 'condition' | 'fork' | 'subflow' | 'loop' | 'break' | 'error';\n  text: string;\n  depth: number;\n  stageName?: string;\n  stepNumber?: number;\n}\n\nexport interface CombinedNarrativeOptions {\n  includeStepNumbers?: boolean;\n  includeValues?: boolean;\n  indent?: string;\n}\n\n// ---------------------------------------------------------------------------\n// Builder\n// ---------------------------------------------------------------------------\n\nexport class CombinedNarrativeBuilder {\n  private options: Required<CombinedNarrativeOptions>;\n\n  constructor(options?: CombinedNarrativeOptions) {\n    this.options = {\n      includeStepNumbers: options?.includeStepNumbers ?? true,\n      includeValues: options?.includeValues ?? true,\n      indent: options?.indent ?? '  ',\n    };\n  }\n\n  /**\n   * @param flowSentences  Sentences from the flow recorder (NarrativeFlowRecorder.getSentences()).\n   * @param recorder       Data-level recorder with per-stage read/write operations.\n   * @param sentenceStageNames  Optional parallel array of actual stage names for each sentence\n   *                            (from NarrativeFlowRecorder.getSentenceStageNames()). When provided,\n   *                            enables direct lookup instead of fragile regex-based matching.\n   * @deprecated Use {@link CombinedNarrativeRecorder} instead, which builds entries inline during traversal.\n   */\n  buildEntries(\n    flowSentences: string[],\n    recorder: NarrativeRecorder,\n    sentenceStageNames?: (string | undefined)[],\n  ): CombinedNarrativeEntry[] {\n    const entries: CombinedNarrativeEntry[] = [];\n    const stageData = recorder.getStageData();\n    const usedStages = new Set<string>();\n    let stageCounter = 0;\n\n    for (let i = 0; i < flowSentences.length; i++) {\n      const sentence = flowSentences[i];\n      const parsed = this.parseSentence(sentence);\n      // Prefer the actual stage name from the recorder over regex-parsed description text\n      const resolvedStageName = sentenceStageNames?.[i] ?? parsed.stageName;\n\n      if (parsed.type === 'stage') {\n        stageCounter++;\n        entries.push({\n          type: 'stage',\n          text: `Stage ${stageCounter}: ${sentence}`,\n          depth: 0,\n          stageName: resolvedStageName,\n        });\n        if (resolvedStageName) {\n          const data = this.findStageData(resolvedStageName, stageData);\n          if (data) {\n            usedStages.add(data.stageName);\n            this.addStepEntries(entries, data);\n          }\n        }\n      } else if (parsed.type === 'condition') {\n        entries.push({ type: 'condition', text: `[Condition]: ${sentence}`, depth: 0, stageName: resolvedStageName });\n        // For conditions, also look up the chosen branch's data\n        if (parsed.stageName) {\n          const data = this.findStageData(parsed.stageName, stageData);\n          if (data) {\n            usedStages.add(data.stageName);\n            this.addStepEntries(entries, data);\n          }\n        }\n        // Also mark the decider stage itself as used (the resolvedStageName)\n        if (resolvedStageName) {\n          const data = this.findStageData(resolvedStageName, stageData);\n          if (data) {\n            usedStages.add(data.stageName);\n            // Only add step entries if the chosen branch didn't already cover it\n            if (!parsed.stageName || parsed.stageName !== resolvedStageName) {\n              this.addStepEntries(entries, data);\n            }\n          }\n        }\n      } else if (parsed.type === 'fork') {\n        entries.push({ type: 'fork', text: `[Parallel]: ${sentence}`, depth: 0 });\n      } else if (parsed.type === 'subflow') {\n        entries.push({ type: 'subflow', text: sentence, depth: 0 });\n      } else if (parsed.type === 'loop') {\n        entries.push({ type: 'loop', text: sentence, depth: 0 });\n      } else if (parsed.type === 'break') {\n        entries.push({ type: 'break', text: sentence, depth: 0 });\n      } else if (parsed.type === 'error') {\n        entries.push({ type: 'error', text: `[Error]: ${sentence}`, depth: 0 });\n      }\n    }\n\n    // Add stages with operations that weren't referenced in flow sentences\n    for (const [stageName, data] of Array.from(stageData.entries())) {\n      if (!usedStages.has(stageName) && data.operations.length > 0) {\n        stageCounter++;\n        entries.push({ type: 'stage', text: `Stage ${stageCounter}: ${stageName}`, depth: 0, stageName });\n        this.addStepEntries(entries, data);\n      }\n    }\n\n    return entries;\n  }\n\n  build(flowSentences: string[], recorder: NarrativeRecorder, sentenceStageNames?: (string | undefined)[]): string[] {\n    return this.buildEntries(flowSentences, recorder, sentenceStageNames).map((entry) => {\n      const indent = this.options.indent.repeat(entry.depth);\n      return `${indent}${entry.text}`;\n    });\n  }\n\n  // ── Private helpers ──\n\n  private addStepEntries(entries: CombinedNarrativeEntry[], data: StageNarrativeData): void {\n    for (const op of data.operations) {\n      const stepPrefix = this.options.includeStepNumbers && op.stepNumber ? `Step ${op.stepNumber}: ` : '';\n\n      let text: string;\n      if (op.type === 'read') {\n        text =\n          this.options.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.options.includeValues\n          ? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Update ${op.key}`;\n      } else {\n        text = this.options.includeValues\n          ? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`\n          : `${stepPrefix}Write ${op.key}`;\n      }\n\n      entries.push({ type: 'step', text, depth: 1, stageName: data.stageName, stepNumber: op.stepNumber });\n    }\n  }\n\n  private findStageData(name: string, stageData: Map<string, StageNarrativeData>): StageNarrativeData | undefined {\n    if (stageData.has(name)) return stageData.get(name);\n    for (const [stageName, data] of Array.from(stageData.entries())) {\n      if (\n        stageName.toLowerCase().includes(name.toLowerCase()) ||\n        name.toLowerCase().includes(stageName.toLowerCase())\n      ) {\n        return data;\n      }\n    }\n    return undefined;\n  }\n\n  private parseSentence(sentence: string): { type: string; stageName?: string } {\n    if (sentence.startsWith('The process began')) {\n      const match = sentence.match(/The process began(?:: (.+)| with (.+))\\./);\n      return { type: 'stage', stageName: (match?.[1] ?? match?.[2])?.trim() };\n    }\n    if (sentence.startsWith('Next')) {\n      const match = sentence.match(/Next(?:,? it moved on to (.+)| step: (.+))\\./);\n      return { type: 'stage', stageName: (match?.[1] ?? match?.[2])?.trim() };\n    }\n    if (sentence.includes('decision was made') || sentence.includes('it chose') || sentence.includes('so it chose')) {\n      const match = sentence.match(/chose (.+)\\./);\n      return { type: 'condition', stageName: match?.[1]?.trim() };\n    }\n    if (sentence.startsWith('It ') && sentence.includes('chose')) {\n      const match = sentence.match(/chose (.+)\\./);\n      return { type: 'condition', stageName: match?.[1]?.trim() };\n    }\n    if (sentence.includes('paths were executed in parallel') || sentence.includes('paths were selected'))\n      return { type: 'fork' };\n    if (sentence.startsWith('Entering') || sentence.startsWith('Exiting')) return { type: 'subflow' };\n    if (sentence.startsWith('On pass')) return { type: 'loop' };\n    if (sentence.startsWith('Execution stopped')) return { type: 'break' };\n    if (sentence.startsWith('An error occurred')) return { type: 'error' };\n    return { type: 'stage' };\n  }\n}\n"]}
|
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CombinedNarrativeRecorder — Inline narrative builder that merges flow + data during traversal.
|
|
3
|
+
*
|
|
4
|
+
* Replaces the post-processing CombinedNarrativeBuilder by implementing BOTH
|
|
5
|
+
* FlowRecorder (control-flow events) and Recorder (scope data events).
|
|
6
|
+
*
|
|
7
|
+
* Event ordering guarantees this works:
|
|
8
|
+
* 1. Scope events (onRead, onWrite) fire DURING stage execution
|
|
9
|
+
* 2. Flow events (onStageExecuted, onDecision) fire AFTER stage execution
|
|
10
|
+
* 3. Both carry the same `stageName` — no matching ambiguity
|
|
11
|
+
*
|
|
12
|
+
* So we buffer scope ops per-stage, then when the flow event arrives,
|
|
13
|
+
* emit the stage entry + flush the buffered ops in one pass.
|
|
14
|
+
*/
|
|
15
|
+
// ── Recorder ───────────────────────────────────────────────────────────────
|
|
16
|
+
export class CombinedNarrativeRecorder {
|
|
17
|
+
constructor(options) {
|
|
18
|
+
var _a, _b, _c, _d;
|
|
19
|
+
this.entries = [];
|
|
20
|
+
this.pendingOps = new Map();
|
|
21
|
+
this.stageCounter = 0;
|
|
22
|
+
this.isFirstStage = true;
|
|
23
|
+
this.id = (_a = options === null || options === void 0 ? void 0 : options.id) !== null && _a !== void 0 ? _a : 'combined-narrative';
|
|
24
|
+
this.includeStepNumbers = (_b = options === null || options === void 0 ? void 0 : options.includeStepNumbers) !== null && _b !== void 0 ? _b : true;
|
|
25
|
+
this.includeValues = (_c = options === null || options === void 0 ? void 0 : options.includeValues) !== null && _c !== void 0 ? _c : true;
|
|
26
|
+
this.maxValueLength = (_d = options === null || options === void 0 ? void 0 : options.maxValueLength) !== null && _d !== void 0 ? _d : 80;
|
|
27
|
+
}
|
|
28
|
+
// ── Scope channel (fires first, during stage execution) ───────────────
|
|
29
|
+
onRead(event) {
|
|
30
|
+
if (!event.key)
|
|
31
|
+
return;
|
|
32
|
+
this.bufferOp(event.stageName, {
|
|
33
|
+
type: 'read',
|
|
34
|
+
key: event.key,
|
|
35
|
+
valueSummary: summarizeValue(event.value, this.maxValueLength),
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
onWrite(event) {
|
|
39
|
+
this.bufferOp(event.stageName, {
|
|
40
|
+
type: 'write',
|
|
41
|
+
key: event.key,
|
|
42
|
+
valueSummary: summarizeValue(event.value, this.maxValueLength),
|
|
43
|
+
operation: event.operation,
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
// ── Flow channel (fires after stage execution) ────────────────────────
|
|
47
|
+
onStageExecuted(event) {
|
|
48
|
+
this.stageCounter++;
|
|
49
|
+
const text = this.isFirstStage
|
|
50
|
+
? event.description
|
|
51
|
+
? `The process began: ${event.description}.`
|
|
52
|
+
: `The process began with ${event.stageName}.`
|
|
53
|
+
: event.description
|
|
54
|
+
? `Next step: ${event.description}.`
|
|
55
|
+
: `Next, it moved on to ${event.stageName}.`;
|
|
56
|
+
this.isFirstStage = false;
|
|
57
|
+
this.entries.push({
|
|
58
|
+
type: 'stage',
|
|
59
|
+
text: `Stage ${this.stageCounter}: ${text}`,
|
|
60
|
+
depth: 0,
|
|
61
|
+
stageName: event.stageName,
|
|
62
|
+
});
|
|
63
|
+
this.flushOps(event.stageName);
|
|
64
|
+
}
|
|
65
|
+
onDecision(event) {
|
|
66
|
+
// Emit the decider stage entry (deciders don't fire onStageExecuted)
|
|
67
|
+
this.stageCounter++;
|
|
68
|
+
const stageText = this.isFirstStage
|
|
69
|
+
? event.description
|
|
70
|
+
? `The process began: ${event.description}.`
|
|
71
|
+
: `The process began with ${event.decider}.`
|
|
72
|
+
: event.description
|
|
73
|
+
? `Next step: ${event.description}.`
|
|
74
|
+
: `Next, it moved on to ${event.decider}.`;
|
|
75
|
+
this.isFirstStage = false;
|
|
76
|
+
this.entries.push({
|
|
77
|
+
type: 'stage',
|
|
78
|
+
text: `Stage ${this.stageCounter}: ${stageText}`,
|
|
79
|
+
depth: 0,
|
|
80
|
+
stageName: event.decider,
|
|
81
|
+
});
|
|
82
|
+
this.flushOps(event.decider);
|
|
83
|
+
// Emit the condition entry
|
|
84
|
+
const branchName = event.chosen;
|
|
85
|
+
let conditionText;
|
|
86
|
+
if (event.description && event.rationale) {
|
|
87
|
+
conditionText = `It ${event.description}: ${event.rationale}, so it chose ${branchName}.`;
|
|
88
|
+
}
|
|
89
|
+
else if (event.description) {
|
|
90
|
+
conditionText = `It ${event.description} and chose ${branchName}.`;
|
|
91
|
+
}
|
|
92
|
+
else if (event.rationale) {
|
|
93
|
+
conditionText = `A decision was made: ${event.rationale}, so the path taken was ${branchName}.`;
|
|
94
|
+
}
|
|
95
|
+
else {
|
|
96
|
+
conditionText = `A decision was made, and the path taken was ${branchName}.`;
|
|
97
|
+
}
|
|
98
|
+
this.entries.push({
|
|
99
|
+
type: 'condition',
|
|
100
|
+
text: `[Condition]: ${conditionText}`,
|
|
101
|
+
depth: 0,
|
|
102
|
+
stageName: event.decider,
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
onNext() {
|
|
106
|
+
// No-op. onStageExecuted already has the description for the next stage.
|
|
107
|
+
// For deciders (no onStageExecuted), onDecision handles the announcement.
|
|
108
|
+
}
|
|
109
|
+
onFork(event) {
|
|
110
|
+
const names = event.children.join(', ');
|
|
111
|
+
this.entries.push({
|
|
112
|
+
type: 'fork',
|
|
113
|
+
text: `[Parallel]: ${event.children.length} paths were executed in parallel: ${names}.`,
|
|
114
|
+
depth: 0,
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
onSelected(event) {
|
|
118
|
+
const names = event.selected.join(', ');
|
|
119
|
+
this.entries.push({
|
|
120
|
+
type: 'fork',
|
|
121
|
+
text: `[Selected]: ${event.selected.length} of ${event.total} paths were selected: ${names}.`,
|
|
122
|
+
depth: 0,
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
onSubflowEntry(event) {
|
|
126
|
+
const text = event.description
|
|
127
|
+
? `Entering the ${event.name} subflow: ${event.description}.`
|
|
128
|
+
: `Entering the ${event.name} subflow.`;
|
|
129
|
+
this.entries.push({ type: 'subflow', text, depth: 0 });
|
|
130
|
+
}
|
|
131
|
+
onSubflowExit(event) {
|
|
132
|
+
this.entries.push({
|
|
133
|
+
type: 'subflow',
|
|
134
|
+
text: `Exiting the ${event.name} subflow.`,
|
|
135
|
+
depth: 0,
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
onLoop(event) {
|
|
139
|
+
const text = event.description
|
|
140
|
+
? `On pass ${event.iteration}: ${event.description} again.`
|
|
141
|
+
: `On pass ${event.iteration} through ${event.target}.`;
|
|
142
|
+
this.entries.push({ type: 'loop', text, depth: 0 });
|
|
143
|
+
}
|
|
144
|
+
onBreak(event) {
|
|
145
|
+
this.entries.push({
|
|
146
|
+
type: 'break',
|
|
147
|
+
text: `Execution stopped at ${event.stageName}.`,
|
|
148
|
+
depth: 0,
|
|
149
|
+
stageName: event.stageName,
|
|
150
|
+
});
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Handles errors from both channels:
|
|
154
|
+
* - FlowRecorder.onError (FlowErrorEvent with message + structuredError)
|
|
155
|
+
* - Recorder.onError (ErrorEvent from scope system — ignored for narrative)
|
|
156
|
+
*/
|
|
157
|
+
onError(event) {
|
|
158
|
+
var _a, _b;
|
|
159
|
+
// Only handle flow errors (which have `message` and `structuredError`)
|
|
160
|
+
if (typeof event.message !== 'string')
|
|
161
|
+
return;
|
|
162
|
+
const flowEvent = event;
|
|
163
|
+
let text = `An error occurred at ${flowEvent.stageName}: ${flowEvent.message}.`;
|
|
164
|
+
if ((_b = (_a = flowEvent.structuredError) === null || _a === void 0 ? void 0 : _a.issues) === null || _b === void 0 ? void 0 : _b.length) {
|
|
165
|
+
const details = flowEvent.structuredError.issues
|
|
166
|
+
.map((issue) => {
|
|
167
|
+
const path = issue.path.length > 0 ? issue.path.join('.') : '(root)';
|
|
168
|
+
return `${path}: ${issue.message}`;
|
|
169
|
+
})
|
|
170
|
+
.join('; ');
|
|
171
|
+
text += ` Validation issues: ${details}.`;
|
|
172
|
+
}
|
|
173
|
+
this.entries.push({
|
|
174
|
+
type: 'error',
|
|
175
|
+
text: `[Error]: ${text}`,
|
|
176
|
+
depth: 0,
|
|
177
|
+
stageName: flowEvent.stageName,
|
|
178
|
+
});
|
|
179
|
+
}
|
|
180
|
+
// ── Output ────────────────────────────────────────────────────────────
|
|
181
|
+
/** Returns structured entries for programmatic consumption. */
|
|
182
|
+
getEntries() {
|
|
183
|
+
return [...this.entries];
|
|
184
|
+
}
|
|
185
|
+
/** Returns formatted narrative lines (same output as CombinedNarrativeBuilder.build). */
|
|
186
|
+
getNarrative(indent = ' ') {
|
|
187
|
+
return this.entries.map((entry) => `${indent.repeat(entry.depth)}${entry.text}`);
|
|
188
|
+
}
|
|
189
|
+
/** Clears all state. Called automatically before each run. */
|
|
190
|
+
clear() {
|
|
191
|
+
this.entries = [];
|
|
192
|
+
this.pendingOps.clear();
|
|
193
|
+
this.stageCounter = 0;
|
|
194
|
+
this.isFirstStage = true;
|
|
195
|
+
}
|
|
196
|
+
// ── Private helpers ───────────────────────────────────────────────────
|
|
197
|
+
bufferOp(stageName, op) {
|
|
198
|
+
let ops = this.pendingOps.get(stageName);
|
|
199
|
+
if (!ops) {
|
|
200
|
+
ops = [];
|
|
201
|
+
this.pendingOps.set(stageName, ops);
|
|
202
|
+
}
|
|
203
|
+
ops.push({ ...op, stepNumber: ops.length + 1 });
|
|
204
|
+
}
|
|
205
|
+
flushOps(stageName) {
|
|
206
|
+
const ops = this.pendingOps.get(stageName);
|
|
207
|
+
if (!ops || ops.length === 0)
|
|
208
|
+
return;
|
|
209
|
+
for (const op of ops) {
|
|
210
|
+
const stepPrefix = this.includeStepNumbers ? `Step ${op.stepNumber}: ` : '';
|
|
211
|
+
let text;
|
|
212
|
+
if (op.type === 'read') {
|
|
213
|
+
text =
|
|
214
|
+
this.includeValues && op.valueSummary
|
|
215
|
+
? `${stepPrefix}Read ${op.key} = ${op.valueSummary}`
|
|
216
|
+
: `${stepPrefix}Read ${op.key}`;
|
|
217
|
+
}
|
|
218
|
+
else if (op.operation === 'delete') {
|
|
219
|
+
text = `${stepPrefix}Delete ${op.key}`;
|
|
220
|
+
}
|
|
221
|
+
else if (op.operation === 'update') {
|
|
222
|
+
text = this.includeValues
|
|
223
|
+
? `${stepPrefix}Update ${op.key} = ${op.valueSummary}`
|
|
224
|
+
: `${stepPrefix}Update ${op.key}`;
|
|
225
|
+
}
|
|
226
|
+
else {
|
|
227
|
+
text = this.includeValues
|
|
228
|
+
? `${stepPrefix}Write ${op.key} = ${op.valueSummary}`
|
|
229
|
+
: `${stepPrefix}Write ${op.key}`;
|
|
230
|
+
}
|
|
231
|
+
this.entries.push({
|
|
232
|
+
type: 'step',
|
|
233
|
+
text,
|
|
234
|
+
depth: 1,
|
|
235
|
+
stageName,
|
|
236
|
+
stepNumber: op.stepNumber,
|
|
237
|
+
});
|
|
238
|
+
}
|
|
239
|
+
this.pendingOps.delete(stageName);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
// ── Value summarizer (same logic as NarrativeRecorder) ─────────────────────
|
|
243
|
+
function summarizeValue(value, maxLen) {
|
|
244
|
+
if (value === undefined)
|
|
245
|
+
return 'undefined';
|
|
246
|
+
if (value === null)
|
|
247
|
+
return 'null';
|
|
248
|
+
if (typeof value === 'string') {
|
|
249
|
+
return value.length <= maxLen ? `"${value}"` : `"${value.slice(0, maxLen - 3)}..."`;
|
|
250
|
+
}
|
|
251
|
+
if (typeof value === 'number' || typeof value === 'boolean')
|
|
252
|
+
return String(value);
|
|
253
|
+
if (Array.isArray(value)) {
|
|
254
|
+
return value.length === 0 ? '[]' : `(${value.length} item${value.length > 1 ? 's' : ''})`;
|
|
255
|
+
}
|
|
256
|
+
if (typeof value === 'object') {
|
|
257
|
+
const keys = Object.keys(value);
|
|
258
|
+
if (keys.length === 0)
|
|
259
|
+
return '{}';
|
|
260
|
+
const preview = keys.slice(0, 4).join(', ');
|
|
261
|
+
const suffix = keys.length > 4 ? `, ... (${keys.length} keys)` : '';
|
|
262
|
+
const result = `{${preview}${suffix}}`;
|
|
263
|
+
return result.length <= maxLen ? result : `{${keys.length} keys}`;
|
|
264
|
+
}
|
|
265
|
+
return String(value);
|
|
266
|
+
}
|
|
267
|
+
//# 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 type { ReadEvent, Recorder, WriteEvent } from '../../scope/types';\nimport type { CombinedNarrativeEntry } from './CombinedNarrativeBuilder';\nimport type {\n  FlowBreakEvent,\n  FlowDecisionEvent,\n  FlowErrorEvent,\n  FlowForkEvent,\n  FlowLoopEvent,\n  FlowRecorder,\n  FlowSelectedEvent,\n  FlowStageEvent,\n  FlowSubflowEvent,\n} from './types';\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  private pendingOps = new Map<string, BufferedOp[]>();\n  private stageCounter = 0;\n  private isFirstStage = true;\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    this.stageCounter++;\n    const text = this.isFirstStage\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    this.isFirstStage = false;\n\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${this.stageCounter}: ${text}`,\n      depth: 0,\n      stageName: event.stageName,\n    });\n    this.flushOps(event.stageName);\n  }\n\n  onDecision(event: FlowDecisionEvent): void {\n    // Emit the decider stage entry (deciders don't fire onStageExecuted)\n    this.stageCounter++;\n    const stageText = this.isFirstStage\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    this.isFirstStage = false;\n\n    this.entries.push({\n      type: 'stage',\n      text: `Stage ${this.stageCounter}: ${stageText}`,\n      depth: 0,\n      stageName: event.decider,\n    });\n    this.flushOps(event.decider);\n\n    // Emit the condition entry\n    const branchName = event.chosen;\n    let conditionText: string;\n    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    });\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]: ${event.children.length} paths were executed in parallel: ${names}.`,\n      depth: 0,\n    });\n  }\n\n  onSelected(event: FlowSelectedEvent): void {\n    const names = event.selected.join(', ');\n    this.entries.push({\n      type: 'fork',\n      text: `[Selected]: ${event.selected.length} of ${event.total} paths were selected: ${names}.`,\n      depth: 0,\n    });\n  }\n\n  onSubflowEntry(event: FlowSubflowEvent): void {\n    const text = event.description\n      ? `Entering the ${event.name} subflow: ${event.description}.`\n      : `Entering the ${event.name} subflow.`;\n    this.entries.push({ type: 'subflow', text, depth: 0 });\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    });\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({ type: 'loop', text, depth: 0 });\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    });\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    });\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  /** Clears all state. Called automatically before each run. */\n  clear(): void {\n    this.entries = [];\n    this.pendingOps.clear();\n    this.stageCounter = 0;\n    this.isFirstStage = true;\n  }\n\n  // ── Private helpers ───────────────────────────────────────────────────\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): 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        stepNumber: op.stepNumber,\n      });\n    }\n\n    this.pendingOps.delete(stageName);\n  }\n}\n\n// ── Value summarizer (same logic as NarrativeRecorder) ─────────────────────\n\nfunction summarizeValue(value: unknown, maxLen: number): string {\n  if (value === undefined) return 'undefined';\n  if (value === null) return 'null';\n  if (typeof value === 'string') {\n    return value.length <= maxLen ? `\"${value}\"` : `\"${value.slice(0, maxLen - 3)}...\"`;\n  }\n  if (typeof value === 'number' || typeof value === 'boolean') return String(value);\n  if (Array.isArray(value)) {\n    return value.length === 0 ? '[]' : `(${value.length} item${value.length > 1 ? 's' : ''})`;\n  }\n  if (typeof value === 'object') {\n    const keys = Object.keys(value as Record<string, unknown>);\n    if (keys.length === 0) return '{}';\n    const preview = keys.slice(0, 4).join(', ');\n    const suffix = keys.length > 4 ? `, ... (${keys.length} keys)` : '';\n    const result = `{${preview}${suffix}}`;\n    return result.length <= maxLen ? result : `{${keys.length} keys}`;\n  }\n  return String(value);\n}\n"]}
|