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.
Files changed (25) hide show
  1. package/dist/esm/lib/engine/narrative/CombinedNarrativeBuilder.js +43 -14
  2. package/dist/esm/lib/engine/narrative/CombinedNarrativeRecorder.js +267 -0
  3. package/dist/esm/lib/engine/narrative/NarrativeFlowRecorder.js +25 -1
  4. package/dist/esm/lib/engine/narrative/index.js +2 -1
  5. package/dist/esm/lib/runner/FlowChartExecutor.js +33 -16
  6. package/dist/esm/lib/scope/recorders/NarrativeRecorder.js +6 -1
  7. package/dist/esm/types/lib/engine/narrative/CombinedNarrativeBuilder.d.ts +15 -2
  8. package/dist/esm/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +63 -0
  9. package/dist/esm/types/lib/engine/narrative/NarrativeFlowRecorder.d.ts +11 -0
  10. package/dist/esm/types/lib/engine/narrative/index.d.ts +2 -0
  11. package/dist/esm/types/lib/runner/FlowChartExecutor.d.ts +8 -1
  12. package/dist/esm/types/lib/scope/recorders/NarrativeRecorder.d.ts +5 -0
  13. package/dist/lib/engine/narrative/CombinedNarrativeBuilder.js +43 -14
  14. package/dist/lib/engine/narrative/CombinedNarrativeRecorder.js +271 -0
  15. package/dist/lib/engine/narrative/NarrativeFlowRecorder.js +25 -1
  16. package/dist/lib/engine/narrative/index.js +4 -2
  17. package/dist/lib/runner/FlowChartExecutor.js +33 -16
  18. package/dist/lib/scope/recorders/NarrativeRecorder.js +6 -1
  19. package/dist/types/lib/engine/narrative/CombinedNarrativeBuilder.d.ts +15 -2
  20. package/dist/types/lib/engine/narrative/CombinedNarrativeRecorder.d.ts +63 -0
  21. package/dist/types/lib/engine/narrative/NarrativeFlowRecorder.d.ts +11 -0
  22. package/dist/types/lib/engine/narrative/index.d.ts +2 -0
  23. package/dist/types/lib/runner/FlowChartExecutor.d.ts +8 -1
  24. package/dist/types/lib/scope/recorders/NarrativeRecorder.d.ts +5 -0
  25. 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
- buildEntries(flowSentences, recorder) {
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 (const sentence of flowSentences) {
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: parsed.stageName,
54
+ stageName: resolvedStageName,
38
55
  });
39
- if (parsed.stageName) {
40
- const data = this.findStageData(parsed.stageName, stageData);
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 || _a === void 0 ? void 0 : _a.trim() };
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: (_b = match === null || match === void 0 ? void 0 : match[1]) === null || _b === void 0 ? void 0 : _b.trim() };
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: (_c = match === null || match === void 0 ? void 0 : match[1]) === null || _c === void 0 ? void 0 : _c.trim() };
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: (_d = match === null || match === void 0 ? void 0 : match[1]) === null || _d === void 0 ? void 0 : _d.trim() };
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"]}