@oh-my-pi/pi-coding-agent 1.337.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (224) hide show
  1. package/CHANGELOG.md +1228 -0
  2. package/README.md +1041 -0
  3. package/docs/compaction.md +403 -0
  4. package/docs/custom-tools.md +541 -0
  5. package/docs/extension-loading.md +1004 -0
  6. package/docs/hooks.md +867 -0
  7. package/docs/rpc.md +1040 -0
  8. package/docs/sdk.md +994 -0
  9. package/docs/session-tree-plan.md +441 -0
  10. package/docs/session.md +240 -0
  11. package/docs/skills.md +290 -0
  12. package/docs/theme.md +637 -0
  13. package/docs/tree.md +197 -0
  14. package/docs/tui.md +341 -0
  15. package/examples/README.md +21 -0
  16. package/examples/custom-tools/README.md +124 -0
  17. package/examples/custom-tools/hello/index.ts +20 -0
  18. package/examples/custom-tools/question/index.ts +84 -0
  19. package/examples/custom-tools/subagent/README.md +172 -0
  20. package/examples/custom-tools/subagent/agents/planner.md +37 -0
  21. package/examples/custom-tools/subagent/agents/reviewer.md +35 -0
  22. package/examples/custom-tools/subagent/agents/scout.md +50 -0
  23. package/examples/custom-tools/subagent/agents/worker.md +24 -0
  24. package/examples/custom-tools/subagent/agents.ts +156 -0
  25. package/examples/custom-tools/subagent/commands/implement-and-review.md +10 -0
  26. package/examples/custom-tools/subagent/commands/implement.md +10 -0
  27. package/examples/custom-tools/subagent/commands/scout-and-plan.md +9 -0
  28. package/examples/custom-tools/subagent/index.ts +1002 -0
  29. package/examples/custom-tools/todo/index.ts +212 -0
  30. package/examples/hooks/README.md +56 -0
  31. package/examples/hooks/auto-commit-on-exit.ts +49 -0
  32. package/examples/hooks/confirm-destructive.ts +59 -0
  33. package/examples/hooks/custom-compaction.ts +116 -0
  34. package/examples/hooks/dirty-repo-guard.ts +52 -0
  35. package/examples/hooks/file-trigger.ts +41 -0
  36. package/examples/hooks/git-checkpoint.ts +53 -0
  37. package/examples/hooks/handoff.ts +150 -0
  38. package/examples/hooks/permission-gate.ts +34 -0
  39. package/examples/hooks/protected-paths.ts +30 -0
  40. package/examples/hooks/qna.ts +119 -0
  41. package/examples/hooks/snake.ts +343 -0
  42. package/examples/hooks/status-line.ts +40 -0
  43. package/examples/sdk/01-minimal.ts +22 -0
  44. package/examples/sdk/02-custom-model.ts +49 -0
  45. package/examples/sdk/03-custom-prompt.ts +44 -0
  46. package/examples/sdk/04-skills.ts +44 -0
  47. package/examples/sdk/05-tools.ts +90 -0
  48. package/examples/sdk/06-hooks.ts +61 -0
  49. package/examples/sdk/07-context-files.ts +36 -0
  50. package/examples/sdk/08-slash-commands.ts +42 -0
  51. package/examples/sdk/09-api-keys-and-oauth.ts +55 -0
  52. package/examples/sdk/10-settings.ts +38 -0
  53. package/examples/sdk/11-sessions.ts +48 -0
  54. package/examples/sdk/12-full-control.ts +95 -0
  55. package/examples/sdk/README.md +154 -0
  56. package/package.json +81 -0
  57. package/src/cli/args.ts +246 -0
  58. package/src/cli/file-processor.ts +72 -0
  59. package/src/cli/list-models.ts +104 -0
  60. package/src/cli/plugin-cli.ts +650 -0
  61. package/src/cli/session-picker.ts +41 -0
  62. package/src/cli.ts +10 -0
  63. package/src/commands/init.md +20 -0
  64. package/src/config.ts +159 -0
  65. package/src/core/agent-session.ts +1900 -0
  66. package/src/core/auth-storage.ts +236 -0
  67. package/src/core/bash-executor.ts +196 -0
  68. package/src/core/compaction/branch-summarization.ts +343 -0
  69. package/src/core/compaction/compaction.ts +742 -0
  70. package/src/core/compaction/index.ts +7 -0
  71. package/src/core/compaction/utils.ts +154 -0
  72. package/src/core/custom-tools/index.ts +21 -0
  73. package/src/core/custom-tools/loader.ts +248 -0
  74. package/src/core/custom-tools/types.ts +169 -0
  75. package/src/core/custom-tools/wrapper.ts +28 -0
  76. package/src/core/exec.ts +129 -0
  77. package/src/core/export-html/index.ts +211 -0
  78. package/src/core/export-html/template.css +781 -0
  79. package/src/core/export-html/template.html +54 -0
  80. package/src/core/export-html/template.js +1185 -0
  81. package/src/core/export-html/vendor/highlight.min.js +1213 -0
  82. package/src/core/export-html/vendor/marked.min.js +6 -0
  83. package/src/core/hooks/index.ts +16 -0
  84. package/src/core/hooks/loader.ts +312 -0
  85. package/src/core/hooks/runner.ts +434 -0
  86. package/src/core/hooks/tool-wrapper.ts +99 -0
  87. package/src/core/hooks/types.ts +773 -0
  88. package/src/core/index.ts +52 -0
  89. package/src/core/mcp/client.ts +158 -0
  90. package/src/core/mcp/config.ts +154 -0
  91. package/src/core/mcp/index.ts +45 -0
  92. package/src/core/mcp/loader.ts +68 -0
  93. package/src/core/mcp/manager.ts +181 -0
  94. package/src/core/mcp/tool-bridge.ts +148 -0
  95. package/src/core/mcp/transports/http.ts +316 -0
  96. package/src/core/mcp/transports/index.ts +6 -0
  97. package/src/core/mcp/transports/stdio.ts +252 -0
  98. package/src/core/mcp/types.ts +220 -0
  99. package/src/core/messages.ts +189 -0
  100. package/src/core/model-registry.ts +317 -0
  101. package/src/core/model-resolver.ts +393 -0
  102. package/src/core/plugins/doctor.ts +59 -0
  103. package/src/core/plugins/index.ts +38 -0
  104. package/src/core/plugins/installer.ts +189 -0
  105. package/src/core/plugins/loader.ts +338 -0
  106. package/src/core/plugins/manager.ts +672 -0
  107. package/src/core/plugins/parser.ts +105 -0
  108. package/src/core/plugins/paths.ts +32 -0
  109. package/src/core/plugins/types.ts +190 -0
  110. package/src/core/sdk.ts +760 -0
  111. package/src/core/session-manager.ts +1128 -0
  112. package/src/core/settings-manager.ts +443 -0
  113. package/src/core/skills.ts +437 -0
  114. package/src/core/slash-commands.ts +248 -0
  115. package/src/core/system-prompt.ts +439 -0
  116. package/src/core/timings.ts +25 -0
  117. package/src/core/tools/ask.ts +211 -0
  118. package/src/core/tools/bash-interceptor.ts +120 -0
  119. package/src/core/tools/bash.ts +250 -0
  120. package/src/core/tools/context.ts +32 -0
  121. package/src/core/tools/edit-diff.ts +475 -0
  122. package/src/core/tools/edit.ts +208 -0
  123. package/src/core/tools/exa/company.ts +59 -0
  124. package/src/core/tools/exa/index.ts +64 -0
  125. package/src/core/tools/exa/linkedin.ts +59 -0
  126. package/src/core/tools/exa/logger.ts +56 -0
  127. package/src/core/tools/exa/mcp-client.ts +368 -0
  128. package/src/core/tools/exa/render.ts +196 -0
  129. package/src/core/tools/exa/researcher.ts +90 -0
  130. package/src/core/tools/exa/search.ts +337 -0
  131. package/src/core/tools/exa/types.ts +168 -0
  132. package/src/core/tools/exa/websets.ts +248 -0
  133. package/src/core/tools/find.ts +261 -0
  134. package/src/core/tools/grep.ts +555 -0
  135. package/src/core/tools/index.ts +202 -0
  136. package/src/core/tools/ls.ts +140 -0
  137. package/src/core/tools/lsp/client.ts +605 -0
  138. package/src/core/tools/lsp/config.ts +147 -0
  139. package/src/core/tools/lsp/edits.ts +101 -0
  140. package/src/core/tools/lsp/index.ts +804 -0
  141. package/src/core/tools/lsp/render.ts +447 -0
  142. package/src/core/tools/lsp/rust-analyzer.ts +145 -0
  143. package/src/core/tools/lsp/types.ts +463 -0
  144. package/src/core/tools/lsp/utils.ts +486 -0
  145. package/src/core/tools/notebook.ts +229 -0
  146. package/src/core/tools/path-utils.ts +61 -0
  147. package/src/core/tools/read.ts +240 -0
  148. package/src/core/tools/renderers.ts +540 -0
  149. package/src/core/tools/task/agents.ts +153 -0
  150. package/src/core/tools/task/artifacts.ts +114 -0
  151. package/src/core/tools/task/bundled-agents/browser.md +71 -0
  152. package/src/core/tools/task/bundled-agents/explore.md +82 -0
  153. package/src/core/tools/task/bundled-agents/plan.md +54 -0
  154. package/src/core/tools/task/bundled-agents/reviewer.md +59 -0
  155. package/src/core/tools/task/bundled-agents/task.md +53 -0
  156. package/src/core/tools/task/bundled-commands/architect-plan.md +10 -0
  157. package/src/core/tools/task/bundled-commands/implement-with-critic.md +11 -0
  158. package/src/core/tools/task/bundled-commands/implement.md +11 -0
  159. package/src/core/tools/task/commands.ts +213 -0
  160. package/src/core/tools/task/discovery.ts +208 -0
  161. package/src/core/tools/task/executor.ts +367 -0
  162. package/src/core/tools/task/index.ts +388 -0
  163. package/src/core/tools/task/model-resolver.ts +115 -0
  164. package/src/core/tools/task/parallel.ts +38 -0
  165. package/src/core/tools/task/render.ts +232 -0
  166. package/src/core/tools/task/types.ts +99 -0
  167. package/src/core/tools/truncate.ts +265 -0
  168. package/src/core/tools/web-fetch.ts +2370 -0
  169. package/src/core/tools/web-search/auth.ts +193 -0
  170. package/src/core/tools/web-search/index.ts +537 -0
  171. package/src/core/tools/web-search/providers/anthropic.ts +198 -0
  172. package/src/core/tools/web-search/providers/exa.ts +302 -0
  173. package/src/core/tools/web-search/providers/perplexity.ts +195 -0
  174. package/src/core/tools/web-search/render.ts +182 -0
  175. package/src/core/tools/web-search/types.ts +180 -0
  176. package/src/core/tools/write.ts +99 -0
  177. package/src/index.ts +176 -0
  178. package/src/main.ts +464 -0
  179. package/src/migrations.ts +135 -0
  180. package/src/modes/index.ts +43 -0
  181. package/src/modes/interactive/components/armin.ts +382 -0
  182. package/src/modes/interactive/components/assistant-message.ts +86 -0
  183. package/src/modes/interactive/components/bash-execution.ts +196 -0
  184. package/src/modes/interactive/components/bordered-loader.ts +41 -0
  185. package/src/modes/interactive/components/branch-summary-message.ts +42 -0
  186. package/src/modes/interactive/components/compaction-summary-message.ts +45 -0
  187. package/src/modes/interactive/components/custom-editor.ts +122 -0
  188. package/src/modes/interactive/components/diff.ts +147 -0
  189. package/src/modes/interactive/components/dynamic-border.ts +25 -0
  190. package/src/modes/interactive/components/footer.ts +381 -0
  191. package/src/modes/interactive/components/hook-editor.ts +117 -0
  192. package/src/modes/interactive/components/hook-input.ts +64 -0
  193. package/src/modes/interactive/components/hook-message.ts +96 -0
  194. package/src/modes/interactive/components/hook-selector.ts +91 -0
  195. package/src/modes/interactive/components/model-selector.ts +247 -0
  196. package/src/modes/interactive/components/oauth-selector.ts +120 -0
  197. package/src/modes/interactive/components/plugin-settings.ts +479 -0
  198. package/src/modes/interactive/components/queue-mode-selector.ts +56 -0
  199. package/src/modes/interactive/components/session-selector.ts +204 -0
  200. package/src/modes/interactive/components/settings-selector.ts +453 -0
  201. package/src/modes/interactive/components/show-images-selector.ts +45 -0
  202. package/src/modes/interactive/components/theme-selector.ts +62 -0
  203. package/src/modes/interactive/components/thinking-selector.ts +64 -0
  204. package/src/modes/interactive/components/tool-execution.ts +675 -0
  205. package/src/modes/interactive/components/tree-selector.ts +866 -0
  206. package/src/modes/interactive/components/user-message-selector.ts +159 -0
  207. package/src/modes/interactive/components/user-message.ts +18 -0
  208. package/src/modes/interactive/components/visual-truncate.ts +50 -0
  209. package/src/modes/interactive/components/welcome.ts +183 -0
  210. package/src/modes/interactive/interactive-mode.ts +2516 -0
  211. package/src/modes/interactive/theme/dark.json +101 -0
  212. package/src/modes/interactive/theme/light.json +98 -0
  213. package/src/modes/interactive/theme/theme-schema.json +308 -0
  214. package/src/modes/interactive/theme/theme.ts +998 -0
  215. package/src/modes/print-mode.ts +128 -0
  216. package/src/modes/rpc/rpc-client.ts +527 -0
  217. package/src/modes/rpc/rpc-mode.ts +483 -0
  218. package/src/modes/rpc/rpc-types.ts +203 -0
  219. package/src/utils/changelog.ts +99 -0
  220. package/src/utils/clipboard.ts +265 -0
  221. package/src/utils/fuzzy.ts +108 -0
  222. package/src/utils/mime.ts +30 -0
  223. package/src/utils/shell.ts +276 -0
  224. package/src/utils/tools-manager.ts +274 -0
@@ -0,0 +1,343 @@
1
+ /**
2
+ * Branch summarization for tree navigation.
3
+ *
4
+ * When navigating to a different point in the session tree, this generates
5
+ * a summary of the branch being left so context isn't lost.
6
+ */
7
+
8
+ import type { AgentMessage } from "@oh-my-pi/pi-agent-core";
9
+ import type { Model } from "@oh-my-pi/pi-ai";
10
+ import { completeSimple } from "@oh-my-pi/pi-ai";
11
+ import {
12
+ convertToLlm,
13
+ createBranchSummaryMessage,
14
+ createCompactionSummaryMessage,
15
+ createHookMessage,
16
+ } from "../messages.js";
17
+ import type { ReadonlySessionManager, SessionEntry } from "../session-manager.js";
18
+ import { estimateTokens } from "./compaction.js";
19
+ import {
20
+ computeFileLists,
21
+ createFileOps,
22
+ extractFileOpsFromMessage,
23
+ type FileOperations,
24
+ formatFileOperations,
25
+ SUMMARIZATION_SYSTEM_PROMPT,
26
+ serializeConversation,
27
+ } from "./utils.js";
28
+
29
+ // ============================================================================
30
+ // Types
31
+ // ============================================================================
32
+
33
+ export interface BranchSummaryResult {
34
+ summary?: string;
35
+ readFiles?: string[];
36
+ modifiedFiles?: string[];
37
+ aborted?: boolean;
38
+ error?: string;
39
+ }
40
+
41
+ /** Details stored in BranchSummaryEntry.details for file tracking */
42
+ export interface BranchSummaryDetails {
43
+ readFiles: string[];
44
+ modifiedFiles: string[];
45
+ }
46
+
47
+ export type { FileOperations } from "./utils.js";
48
+
49
+ export interface BranchPreparation {
50
+ /** Messages extracted for summarization, in chronological order */
51
+ messages: AgentMessage[];
52
+ /** File operations extracted from tool calls */
53
+ fileOps: FileOperations;
54
+ /** Total estimated tokens in messages */
55
+ totalTokens: number;
56
+ }
57
+
58
+ export interface CollectEntriesResult {
59
+ /** Entries to summarize, in chronological order */
60
+ entries: SessionEntry[];
61
+ /** Common ancestor between old and new position, if any */
62
+ commonAncestorId: string | null;
63
+ }
64
+
65
+ export interface GenerateBranchSummaryOptions {
66
+ /** Model to use for summarization */
67
+ model: Model<any>;
68
+ /** API key for the model */
69
+ apiKey: string;
70
+ /** Abort signal for cancellation */
71
+ signal: AbortSignal;
72
+ /** Optional custom instructions for summarization */
73
+ customInstructions?: string;
74
+ /** Tokens reserved for prompt + LLM response (default 16384) */
75
+ reserveTokens?: number;
76
+ }
77
+
78
+ // ============================================================================
79
+ // Entry Collection
80
+ // ============================================================================
81
+
82
+ /**
83
+ * Collect entries that should be summarized when navigating from one position to another.
84
+ *
85
+ * Walks from oldLeafId back to the common ancestor with targetId, collecting entries
86
+ * along the way. Does NOT stop at compaction boundaries - those are included and their
87
+ * summaries become context.
88
+ *
89
+ * @param session - Session manager (read-only access)
90
+ * @param oldLeafId - Current position (where we're navigating from)
91
+ * @param targetId - Target position (where we're navigating to)
92
+ * @returns Entries to summarize and the common ancestor
93
+ */
94
+ export function collectEntriesForBranchSummary(
95
+ session: ReadonlySessionManager,
96
+ oldLeafId: string | null,
97
+ targetId: string,
98
+ ): CollectEntriesResult {
99
+ // If no old position, nothing to summarize
100
+ if (!oldLeafId) {
101
+ return { entries: [], commonAncestorId: null };
102
+ }
103
+
104
+ // Find common ancestor (deepest node that's on both paths)
105
+ const oldPath = new Set(session.getBranch(oldLeafId).map((e) => e.id));
106
+ const targetPath = session.getBranch(targetId);
107
+
108
+ // targetPath is root-first, so iterate backwards to find deepest common ancestor
109
+ let commonAncestorId: string | null = null;
110
+ for (let i = targetPath.length - 1; i >= 0; i--) {
111
+ if (oldPath.has(targetPath[i].id)) {
112
+ commonAncestorId = targetPath[i].id;
113
+ break;
114
+ }
115
+ }
116
+
117
+ // Collect entries from old leaf back to common ancestor
118
+ const entries: SessionEntry[] = [];
119
+ let current: string | null = oldLeafId;
120
+
121
+ while (current && current !== commonAncestorId) {
122
+ const entry = session.getEntry(current);
123
+ if (!entry) break;
124
+ entries.push(entry);
125
+ current = entry.parentId;
126
+ }
127
+
128
+ // Reverse to get chronological order
129
+ entries.reverse();
130
+
131
+ return { entries, commonAncestorId };
132
+ }
133
+
134
+ // ============================================================================
135
+ // Entry to Message Conversion
136
+ // ============================================================================
137
+
138
+ /**
139
+ * Extract AgentMessage from a session entry.
140
+ * Similar to getMessageFromEntry in compaction.ts but also handles compaction entries.
141
+ */
142
+ function getMessageFromEntry(entry: SessionEntry): AgentMessage | undefined {
143
+ switch (entry.type) {
144
+ case "message":
145
+ // Skip tool results - context is in assistant's tool call
146
+ if (entry.message.role === "toolResult") return undefined;
147
+ return entry.message;
148
+
149
+ case "custom_message":
150
+ return createHookMessage(entry.customType, entry.content, entry.display, entry.details, entry.timestamp);
151
+
152
+ case "branch_summary":
153
+ return createBranchSummaryMessage(entry.summary, entry.fromId, entry.timestamp);
154
+
155
+ case "compaction":
156
+ return createCompactionSummaryMessage(entry.summary, entry.tokensBefore, entry.timestamp);
157
+
158
+ // These don't contribute to conversation content
159
+ case "thinking_level_change":
160
+ case "model_change":
161
+ case "custom":
162
+ case "label":
163
+ return undefined;
164
+ }
165
+ }
166
+
167
+ /**
168
+ * Prepare entries for summarization with token budget.
169
+ *
170
+ * Walks entries from NEWEST to OLDEST, adding messages until we hit the token budget.
171
+ * This ensures we keep the most recent context when the branch is too long.
172
+ *
173
+ * Also collects file operations from:
174
+ * - Tool calls in assistant messages
175
+ * - Existing branch_summary entries' details (for cumulative tracking)
176
+ *
177
+ * @param entries - Entries in chronological order
178
+ * @param tokenBudget - Maximum tokens to include (0 = no limit)
179
+ */
180
+ export function prepareBranchEntries(entries: SessionEntry[], tokenBudget: number = 0): BranchPreparation {
181
+ const messages: AgentMessage[] = [];
182
+ const fileOps = createFileOps();
183
+ let totalTokens = 0;
184
+
185
+ // First pass: collect file ops from ALL entries (even if they don't fit in token budget)
186
+ // This ensures we capture cumulative file tracking from nested branch summaries
187
+ // Only extract from pi-generated summaries (fromHook !== true), not hook-generated ones
188
+ for (const entry of entries) {
189
+ if (entry.type === "branch_summary" && !entry.fromHook && entry.details) {
190
+ const details = entry.details as BranchSummaryDetails;
191
+ if (Array.isArray(details.readFiles)) {
192
+ for (const f of details.readFiles) fileOps.read.add(f);
193
+ }
194
+ if (Array.isArray(details.modifiedFiles)) {
195
+ // Modified files go into both edited and written for proper deduplication
196
+ for (const f of details.modifiedFiles) {
197
+ fileOps.edited.add(f);
198
+ }
199
+ }
200
+ }
201
+ }
202
+
203
+ // Second pass: walk from newest to oldest, adding messages until token budget
204
+ for (let i = entries.length - 1; i >= 0; i--) {
205
+ const entry = entries[i];
206
+ const message = getMessageFromEntry(entry);
207
+ if (!message) continue;
208
+
209
+ // Extract file ops from assistant messages (tool calls)
210
+ extractFileOpsFromMessage(message, fileOps);
211
+
212
+ const tokens = estimateTokens(message);
213
+
214
+ // Check budget before adding
215
+ if (tokenBudget > 0 && totalTokens + tokens > tokenBudget) {
216
+ // If this is a summary entry, try to fit it anyway as it's important context
217
+ if (entry.type === "compaction" || entry.type === "branch_summary") {
218
+ if (totalTokens < tokenBudget * 0.9) {
219
+ messages.unshift(message);
220
+ totalTokens += tokens;
221
+ }
222
+ }
223
+ // Stop - we've hit the budget
224
+ break;
225
+ }
226
+
227
+ messages.unshift(message);
228
+ totalTokens += tokens;
229
+ }
230
+
231
+ return { messages, fileOps, totalTokens };
232
+ }
233
+
234
+ // ============================================================================
235
+ // Summary Generation
236
+ // ============================================================================
237
+
238
+ const BRANCH_SUMMARY_PREAMBLE = `The user explored a different conversation branch before returning here.
239
+ Summary of that exploration:
240
+
241
+ `;
242
+
243
+ const BRANCH_SUMMARY_PROMPT = `Create a structured summary of this conversation branch for context when returning later.
244
+
245
+ Use this EXACT format:
246
+
247
+ ## Goal
248
+ [What was the user trying to accomplish in this branch?]
249
+
250
+ ## Constraints & Preferences
251
+ - [Any constraints, preferences, or requirements mentioned]
252
+ - [Or "(none)" if none were mentioned]
253
+
254
+ ## Progress
255
+ ### Done
256
+ - [x] [Completed tasks/changes]
257
+
258
+ ### In Progress
259
+ - [ ] [Work that was started but not finished]
260
+
261
+ ### Blocked
262
+ - [Issues preventing progress, if any]
263
+
264
+ ## Key Decisions
265
+ - **[Decision]**: [Brief rationale]
266
+
267
+ ## Next Steps
268
+ 1. [What should happen next to continue this work]
269
+
270
+ Keep each section concise. Preserve exact file paths, function names, and error messages.`;
271
+
272
+ /**
273
+ * Generate a summary of abandoned branch entries.
274
+ *
275
+ * @param entries - Session entries to summarize (chronological order)
276
+ * @param options - Generation options
277
+ */
278
+ export async function generateBranchSummary(
279
+ entries: SessionEntry[],
280
+ options: GenerateBranchSummaryOptions,
281
+ ): Promise<BranchSummaryResult> {
282
+ const { model, apiKey, signal, customInstructions, reserveTokens = 16384 } = options;
283
+
284
+ // Token budget = context window minus reserved space for prompt + response
285
+ const contextWindow = model.contextWindow || 128000;
286
+ const tokenBudget = contextWindow - reserveTokens;
287
+
288
+ const { messages, fileOps } = prepareBranchEntries(entries, tokenBudget);
289
+
290
+ if (messages.length === 0) {
291
+ return { summary: "No content to summarize" };
292
+ }
293
+
294
+ // Transform to LLM-compatible messages, then serialize to text
295
+ // Serialization prevents the model from treating it as a conversation to continue
296
+ const llmMessages = convertToLlm(messages);
297
+ const conversationText = serializeConversation(llmMessages);
298
+
299
+ // Build prompt
300
+ const instructions = customInstructions || BRANCH_SUMMARY_PROMPT;
301
+ const promptText = `<conversation>\n${conversationText}\n</conversation>\n\n${instructions}`;
302
+
303
+ const summarizationMessages = [
304
+ {
305
+ role: "user" as const,
306
+ content: [{ type: "text" as const, text: promptText }],
307
+ timestamp: Date.now(),
308
+ },
309
+ ];
310
+
311
+ // Call LLM for summarization
312
+ const response = await completeSimple(
313
+ model,
314
+ { systemPrompt: SUMMARIZATION_SYSTEM_PROMPT, messages: summarizationMessages },
315
+ { apiKey, signal, maxTokens: 2048 },
316
+ );
317
+
318
+ // Check if aborted or errored
319
+ if (response.stopReason === "aborted") {
320
+ return { aborted: true };
321
+ }
322
+ if (response.stopReason === "error") {
323
+ return { error: response.errorMessage || "Summarization failed" };
324
+ }
325
+
326
+ let summary = response.content
327
+ .filter((c): c is { type: "text"; text: string } => c.type === "text")
328
+ .map((c) => c.text)
329
+ .join("\n");
330
+
331
+ // Prepend preamble to provide context about the branch summary
332
+ summary = BRANCH_SUMMARY_PREAMBLE + summary;
333
+
334
+ // Compute file lists and append to summary
335
+ const { readFiles, modifiedFiles } = computeFileLists(fileOps);
336
+ summary += formatFileOperations(readFiles, modifiedFiles);
337
+
338
+ return {
339
+ summary: summary || "No summary generated",
340
+ readFiles,
341
+ modifiedFiles,
342
+ };
343
+ }