@aexol/spectral 0.7.1 → 0.7.5

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 (219) hide show
  1. package/CHANGELOG.md +5 -0
  2. package/dist/agent/agents.js +1 -1
  3. package/dist/agent/index.js +199 -184
  4. package/dist/commands/serve.js +0 -3
  5. package/dist/designer/data/systems/renault/DESIGN.md +1 -1
  6. package/dist/designer/philosophies.js +668 -0
  7. package/dist/mcp/sampling-handler.js +1 -1
  8. package/dist/memory/commands/status.js +1 -1
  9. package/dist/memory/compaction.js +2 -2
  10. package/dist/memory/config.js +1 -1
  11. package/dist/memory/debug-log.js +1 -1
  12. package/dist/memory/hooks/compaction-hook.js +29 -0
  13. package/dist/memory/index.js +2 -0
  14. package/dist/memory/observer.js +2 -2
  15. package/dist/memory/project-observations-store.js +14 -0
  16. package/dist/memory/tokens.js +1 -1
  17. package/dist/memory/tools/read-project-observations.js +82 -0
  18. package/dist/memory/tools/recall-observation.js +2 -2
  19. package/dist/pi/agent-core/agent-loop.js +501 -0
  20. package/dist/pi/agent-core/agent.js +401 -0
  21. package/dist/pi/agent-core/harness/agent-harness.js +899 -0
  22. package/dist/pi/agent-core/harness/compaction/branch-summarization.js +173 -0
  23. package/dist/pi/agent-core/harness/compaction/compaction.js +532 -0
  24. package/dist/pi/agent-core/harness/compaction/utils.js +130 -0
  25. package/dist/pi/agent-core/harness/env/nodejs.js +485 -0
  26. package/dist/pi/agent-core/harness/messages.js +101 -0
  27. package/dist/pi/agent-core/harness/prompt-templates.js +229 -0
  28. package/dist/pi/agent-core/harness/session/jsonl-repo.js +100 -0
  29. package/dist/pi/agent-core/harness/session/jsonl-storage.js +230 -0
  30. package/dist/pi/agent-core/harness/session/memory-repo.js +41 -0
  31. package/dist/pi/agent-core/harness/session/memory-storage.js +113 -0
  32. package/dist/pi/agent-core/harness/session/repo-utils.js +38 -0
  33. package/dist/pi/agent-core/harness/session/session.js +196 -0
  34. package/dist/pi/agent-core/harness/session/uuid.js +49 -0
  35. package/dist/pi/agent-core/harness/skills.js +310 -0
  36. package/dist/pi/agent-core/harness/system-prompt.js +29 -0
  37. package/dist/pi/agent-core/harness/types.js +93 -0
  38. package/dist/pi/agent-core/harness/utils/shell-output.js +125 -0
  39. package/dist/pi/agent-core/harness/utils/truncate.js +289 -0
  40. package/dist/pi/agent-core/index.js +24 -0
  41. package/dist/pi/agent-core/node.js +2 -0
  42. package/dist/pi/agent-core/proxy.js +277 -0
  43. package/dist/pi/agent-core/types.js +1 -0
  44. package/dist/pi/ai/api-registry.js +43 -0
  45. package/dist/pi/ai/cli.js +120 -0
  46. package/dist/pi/ai/env-api-keys.js +169 -0
  47. package/dist/pi/ai/image-models.generated.js +441 -0
  48. package/dist/pi/ai/image-models.js +22 -0
  49. package/dist/pi/ai/images-api-registry.js +21 -0
  50. package/dist/pi/ai/images.js +13 -0
  51. package/dist/pi/ai/index.js +18 -0
  52. package/dist/pi/ai/models.generated.js +16220 -0
  53. package/dist/pi/ai/models.js +70 -0
  54. package/dist/pi/ai/oauth.js +1 -0
  55. package/dist/pi/ai/providers/anthropic.js +945 -0
  56. package/dist/pi/ai/providers/faux.js +367 -0
  57. package/dist/pi/ai/providers/github-copilot-headers.js +28 -0
  58. package/dist/pi/ai/providers/openai-completions.js +945 -0
  59. package/dist/pi/ai/providers/openai-prompt-cache.js +9 -0
  60. package/dist/pi/ai/providers/register-builtins.js +97 -0
  61. package/dist/pi/ai/providers/simple-options.js +40 -0
  62. package/dist/pi/ai/providers/transform-messages.js +183 -0
  63. package/dist/pi/ai/session-resources.js +21 -0
  64. package/dist/pi/ai/stream.js +26 -0
  65. package/dist/pi/ai/types.js +1 -0
  66. package/dist/pi/ai/utils/diagnostics.js +24 -0
  67. package/dist/pi/ai/utils/event-stream.js +80 -0
  68. package/dist/pi/ai/utils/hash.js +13 -0
  69. package/dist/pi/ai/utils/headers.js +7 -0
  70. package/dist/pi/ai/utils/json-parse.js +112 -0
  71. package/dist/pi/ai/utils/node-http-proxy.js +96 -0
  72. package/dist/pi/ai/utils/oauth/anthropic.js +334 -0
  73. package/dist/pi/ai/utils/oauth/device-code.js +54 -0
  74. package/dist/pi/ai/utils/oauth/github-copilot.js +270 -0
  75. package/dist/pi/ai/utils/oauth/index.js +121 -0
  76. package/dist/pi/ai/utils/oauth/oauth-page.js +104 -0
  77. package/dist/pi/ai/utils/oauth/openai-codex.js +384 -0
  78. package/dist/pi/ai/utils/oauth/pkce.js +30 -0
  79. package/dist/pi/ai/utils/oauth/types.js +1 -0
  80. package/dist/pi/ai/utils/overflow.js +150 -0
  81. package/dist/pi/ai/utils/sanitize-unicode.js +25 -0
  82. package/dist/pi/ai/utils/typebox-helpers.js +20 -0
  83. package/dist/pi/ai/utils/validation.js +280 -0
  84. package/dist/pi/coding-agent/bun/cli.js +7 -0
  85. package/dist/pi/coding-agent/bun/restore-sandbox-env.js +31 -0
  86. package/dist/pi/coding-agent/cli/args.js +340 -0
  87. package/dist/pi/coding-agent/cli/file-processor.js +82 -0
  88. package/dist/pi/coding-agent/cli/initial-message.js +21 -0
  89. package/dist/pi/coding-agent/cli.js +17 -0
  90. package/dist/pi/coding-agent/config.js +414 -0
  91. package/dist/pi/coding-agent/core/agent-session-runtime.js +299 -0
  92. package/dist/pi/coding-agent/core/agent-session-services.js +117 -0
  93. package/dist/pi/coding-agent/core/agent-session.js +2498 -0
  94. package/dist/pi/coding-agent/core/auth-guidance.js +20 -0
  95. package/dist/pi/coding-agent/core/auth-storage.js +441 -0
  96. package/dist/pi/coding-agent/core/bash-executor.js +110 -0
  97. package/dist/pi/coding-agent/core/compaction/branch-summarization.js +242 -0
  98. package/dist/pi/coding-agent/core/compaction/compaction.js +624 -0
  99. package/dist/pi/coding-agent/core/compaction/index.js +6 -0
  100. package/dist/pi/coding-agent/core/compaction/utils.js +152 -0
  101. package/dist/pi/coding-agent/core/defaults.js +1 -0
  102. package/dist/pi/coding-agent/core/diagnostics.js +1 -0
  103. package/dist/pi/coding-agent/core/event-bus.js +24 -0
  104. package/dist/pi/coding-agent/core/exec.js +74 -0
  105. package/dist/pi/coding-agent/core/export-html/ansi-to-html.js +248 -0
  106. package/dist/pi/coding-agent/core/export-html/index.js +225 -0
  107. package/dist/pi/coding-agent/core/export-html/tool-renderer.js +107 -0
  108. package/dist/pi/coding-agent/core/extensions/index.js +8 -0
  109. package/dist/pi/coding-agent/core/extensions/loader.js +485 -0
  110. package/dist/pi/coding-agent/core/extensions/runner.js +824 -0
  111. package/dist/pi/coding-agent/core/extensions/types.js +44 -0
  112. package/dist/pi/coding-agent/core/extensions/wrapper.js +21 -0
  113. package/dist/pi/coding-agent/core/footer-data-provider.js +309 -0
  114. package/dist/pi/coding-agent/core/http-dispatcher.js +47 -0
  115. package/dist/pi/coding-agent/core/index.js +11 -0
  116. package/dist/pi/coding-agent/core/keybindings.js +294 -0
  117. package/dist/pi/coding-agent/core/messages.js +122 -0
  118. package/dist/pi/coding-agent/core/model-registry.js +728 -0
  119. package/dist/pi/coding-agent/core/model-resolver.js +494 -0
  120. package/dist/pi/coding-agent/core/output-guard.js +58 -0
  121. package/dist/pi/coding-agent/core/package-manager.js +2020 -0
  122. package/dist/pi/coding-agent/core/prompt-templates.js +237 -0
  123. package/dist/pi/coding-agent/core/provider-display-names.js +32 -0
  124. package/dist/pi/coding-agent/core/resolve-config-value.js +125 -0
  125. package/dist/pi/coding-agent/core/resource-loader.js +733 -0
  126. package/dist/pi/coding-agent/core/sdk.js +282 -0
  127. package/dist/pi/coding-agent/core/session-cwd.js +37 -0
  128. package/dist/pi/coding-agent/core/session-manager.js +1146 -0
  129. package/dist/pi/coding-agent/core/settings-manager.js +794 -0
  130. package/dist/pi/coding-agent/core/skills.js +386 -0
  131. package/dist/pi/coding-agent/core/slash-commands.js +24 -0
  132. package/dist/pi/coding-agent/core/source-info.js +18 -0
  133. package/dist/pi/coding-agent/core/system-prompt.js +122 -0
  134. package/dist/pi/coding-agent/core/telemetry.js +8 -0
  135. package/dist/pi/coding-agent/core/timings.js +30 -0
  136. package/dist/pi/coding-agent/core/tools/bash.js +341 -0
  137. package/dist/pi/coding-agent/core/tools/edit-diff.js +344 -0
  138. package/dist/pi/coding-agent/core/tools/edit.js +324 -0
  139. package/dist/pi/coding-agent/core/tools/file-mutation-queue.js +36 -0
  140. package/dist/pi/coding-agent/core/tools/find.js +297 -0
  141. package/dist/pi/coding-agent/core/tools/grep.js +303 -0
  142. package/dist/pi/coding-agent/core/tools/index.js +111 -0
  143. package/dist/pi/coding-agent/core/tools/ls.js +168 -0
  144. package/dist/pi/coding-agent/core/tools/output-accumulator.js +183 -0
  145. package/dist/pi/coding-agent/core/tools/path-utils.js +61 -0
  146. package/dist/pi/coding-agent/core/tools/read.js +288 -0
  147. package/dist/pi/coding-agent/core/tools/render-utils.js +48 -0
  148. package/dist/pi/coding-agent/core/tools/tool-definition-wrapper.js +33 -0
  149. package/dist/pi/coding-agent/core/tools/truncate.js +214 -0
  150. package/dist/pi/coding-agent/core/tools/write.js +212 -0
  151. package/dist/pi/coding-agent/index.js +41 -0
  152. package/dist/pi/coding-agent/main.js +5 -0
  153. package/dist/pi/coding-agent/migrations.js +280 -0
  154. package/dist/pi/coding-agent/modes/index.js +7 -0
  155. package/dist/pi/coding-agent/modes/interactive/components/diff.js +132 -0
  156. package/dist/pi/coding-agent/modes/interactive/components/keybinding-hints.js +35 -0
  157. package/dist/pi/coding-agent/modes/interactive/components/visual-truncate.js +32 -0
  158. package/dist/pi/coding-agent/modes/interactive/interactive-mode.js +3 -0
  159. package/dist/pi/coding-agent/modes/interactive/theme/theme.js +1023 -0
  160. package/dist/pi/coding-agent/modes/print-mode.js +130 -0
  161. package/dist/pi/coding-agent/modes/rpc/jsonl.js +48 -0
  162. package/dist/pi/coding-agent/modes/rpc/rpc-client.js +409 -0
  163. package/dist/pi/coding-agent/modes/rpc/rpc-mode.js +600 -0
  164. package/dist/pi/coding-agent/modes/rpc/rpc-types.js +7 -0
  165. package/dist/pi/coding-agent/utils/ansi.js +51 -0
  166. package/dist/pi/coding-agent/utils/changelog.js +86 -0
  167. package/dist/pi/coding-agent/utils/child-process.js +87 -0
  168. package/dist/pi/coding-agent/utils/clipboard-image.js +244 -0
  169. package/dist/pi/coding-agent/utils/clipboard-native.js +13 -0
  170. package/dist/pi/coding-agent/utils/clipboard.js +116 -0
  171. package/dist/pi/coding-agent/utils/exif-orientation.js +157 -0
  172. package/dist/pi/coding-agent/utils/frontmatter.js +25 -0
  173. package/dist/pi/coding-agent/utils/fs-watch.js +24 -0
  174. package/dist/pi/coding-agent/utils/git.js +162 -0
  175. package/dist/pi/coding-agent/utils/html.js +39 -0
  176. package/dist/pi/coding-agent/utils/image-convert.js +38 -0
  177. package/dist/pi/coding-agent/utils/image-resize.js +136 -0
  178. package/dist/pi/coding-agent/utils/mime.js +68 -0
  179. package/dist/pi/coding-agent/utils/paths.js +91 -0
  180. package/dist/pi/coding-agent/utils/photon.js +120 -0
  181. package/dist/pi/coding-agent/utils/pi-user-agent.js +4 -0
  182. package/dist/pi/coding-agent/utils/shell.js +194 -0
  183. package/dist/pi/coding-agent/utils/sleep.js +16 -0
  184. package/dist/pi/coding-agent/utils/syntax-highlight.js +117 -0
  185. package/dist/pi/coding-agent/utils/tools-manager.js +327 -0
  186. package/dist/pi/coding-agent/utils/version-check.js +81 -0
  187. package/dist/pi/coding-agent/utils/windows-self-update.js +76 -0
  188. package/dist/pi/tui/autocomplete.js +631 -0
  189. package/dist/pi/tui/components/box.js +103 -0
  190. package/dist/pi/tui/components/cancellable-loader.js +34 -0
  191. package/dist/pi/tui/components/editor.js +1915 -0
  192. package/dist/pi/tui/components/image.js +88 -0
  193. package/dist/pi/tui/components/input.js +425 -0
  194. package/dist/pi/tui/components/loader.js +68 -0
  195. package/dist/pi/tui/components/markdown.js +633 -0
  196. package/dist/pi/tui/components/select-list.js +158 -0
  197. package/dist/pi/tui/components/settings-list.js +184 -0
  198. package/dist/pi/tui/components/spacer.js +22 -0
  199. package/dist/pi/tui/components/text.js +88 -0
  200. package/dist/pi/tui/components/truncated-text.js +50 -0
  201. package/dist/pi/tui/editor-component.js +1 -0
  202. package/dist/pi/tui/fuzzy.js +109 -0
  203. package/dist/pi/tui/index.js +31 -0
  204. package/dist/pi/tui/keybindings.js +173 -0
  205. package/dist/pi/tui/keys.js +1172 -0
  206. package/dist/pi/tui/kill-ring.js +43 -0
  207. package/dist/pi/tui/stdin-buffer.js +360 -0
  208. package/dist/pi/tui/terminal-image.js +335 -0
  209. package/dist/pi/tui/terminal.js +324 -0
  210. package/dist/pi/tui/tui.js +1076 -0
  211. package/dist/pi/tui/undo-stack.js +24 -0
  212. package/dist/pi/tui/utils.js +1016 -0
  213. package/dist/relay/dispatcher.js +30 -0
  214. package/dist/server/handlers/queue.js +52 -0
  215. package/dist/server/pi-bridge.js +9 -1
  216. package/dist/server/session-stream.js +76 -111
  217. package/dist/server/storage.js +154 -2
  218. package/dist/server/title-generator.js +14 -153
  219. package/package.json +24 -6
@@ -0,0 +1,501 @@
1
+ /**
2
+ * Agent loop that works with AgentMessage throughout.
3
+ * Transforms to Message[] only at the LLM call boundary.
4
+ */
5
+ import { EventStream, streamSimple, validateToolArguments, } from "../ai/index.js";
6
+ /**
7
+ * Start an agent loop with a new prompt message.
8
+ * The prompt is added to the context and events are emitted for it.
9
+ */
10
+ export function agentLoop(prompts, context, config, signal, streamFn) {
11
+ const stream = createAgentStream();
12
+ void runAgentLoop(prompts, context, config, async (event) => {
13
+ stream.push(event);
14
+ }, signal, streamFn).then((messages) => {
15
+ stream.end(messages);
16
+ });
17
+ return stream;
18
+ }
19
+ /**
20
+ * Continue an agent loop from the current context without adding a new message.
21
+ * Used for retries - context already has user message or tool results.
22
+ *
23
+ * **Important:** The last message in context must convert to a `user` or `toolResult` message
24
+ * via `convertToLlm`. If it doesn't, the LLM provider will reject the request.
25
+ * This cannot be validated here since `convertToLlm` is only called once per turn.
26
+ */
27
+ export function agentLoopContinue(context, config, signal, streamFn) {
28
+ if (context.messages.length === 0) {
29
+ throw new Error("Cannot continue: no messages in context");
30
+ }
31
+ if (context.messages[context.messages.length - 1].role === "assistant") {
32
+ throw new Error("Cannot continue from message role: assistant");
33
+ }
34
+ const stream = createAgentStream();
35
+ void runAgentLoopContinue(context, config, async (event) => {
36
+ stream.push(event);
37
+ }, signal, streamFn).then((messages) => {
38
+ stream.end(messages);
39
+ });
40
+ return stream;
41
+ }
42
+ export async function runAgentLoop(prompts, context, config, emit, signal, streamFn) {
43
+ const newMessages = [...prompts];
44
+ const currentContext = {
45
+ ...context,
46
+ messages: [...context.messages, ...prompts],
47
+ };
48
+ await emit({ type: "agent_start" });
49
+ await emit({ type: "turn_start" });
50
+ for (const prompt of prompts) {
51
+ await emit({ type: "message_start", message: prompt });
52
+ await emit({ type: "message_end", message: prompt });
53
+ }
54
+ await runLoop(currentContext, newMessages, config, signal, emit, streamFn);
55
+ return newMessages;
56
+ }
57
+ export async function runAgentLoopContinue(context, config, emit, signal, streamFn) {
58
+ if (context.messages.length === 0) {
59
+ throw new Error("Cannot continue: no messages in context");
60
+ }
61
+ if (context.messages[context.messages.length - 1].role === "assistant") {
62
+ throw new Error("Cannot continue from message role: assistant");
63
+ }
64
+ const newMessages = [];
65
+ const currentContext = { ...context };
66
+ await emit({ type: "agent_start" });
67
+ await emit({ type: "turn_start" });
68
+ await runLoop(currentContext, newMessages, config, signal, emit, streamFn);
69
+ return newMessages;
70
+ }
71
+ function createAgentStream() {
72
+ return new EventStream((event) => event.type === "agent_end", (event) => (event.type === "agent_end" ? event.messages : []));
73
+ }
74
+ /**
75
+ * Main loop logic shared by agentLoop and agentLoopContinue.
76
+ */
77
+ async function runLoop(initialContext, newMessages, initialConfig, signal, emit, streamFn) {
78
+ let currentContext = initialContext;
79
+ let config = initialConfig;
80
+ let firstTurn = true;
81
+ // Check for steering messages at start (user may have typed while waiting)
82
+ let pendingMessages = (await config.getSteeringMessages?.()) || [];
83
+ // Outer loop: continues when queued follow-up messages arrive after agent would stop
84
+ while (true) {
85
+ let hasMoreToolCalls = true;
86
+ // Inner loop: process tool calls and steering messages
87
+ while (hasMoreToolCalls || pendingMessages.length > 0) {
88
+ if (!firstTurn) {
89
+ await emit({ type: "turn_start" });
90
+ }
91
+ else {
92
+ firstTurn = false;
93
+ }
94
+ // Process pending messages (inject before next assistant response)
95
+ if (pendingMessages.length > 0) {
96
+ for (const message of pendingMessages) {
97
+ await emit({ type: "message_start", message });
98
+ await emit({ type: "message_end", message });
99
+ currentContext.messages.push(message);
100
+ newMessages.push(message);
101
+ }
102
+ pendingMessages = [];
103
+ }
104
+ // Stream assistant response
105
+ const message = await streamAssistantResponse(currentContext, config, signal, emit, streamFn);
106
+ newMessages.push(message);
107
+ if (message.stopReason === "error" || message.stopReason === "aborted") {
108
+ await emit({ type: "turn_end", message, toolResults: [] });
109
+ await emit({ type: "agent_end", messages: newMessages });
110
+ return;
111
+ }
112
+ // Check for tool calls
113
+ const toolCalls = message.content.filter((c) => c.type === "toolCall");
114
+ const toolResults = [];
115
+ hasMoreToolCalls = false;
116
+ if (toolCalls.length > 0) {
117
+ const executedToolBatch = await executeToolCalls(currentContext, message, config, signal, emit);
118
+ toolResults.push(...executedToolBatch.messages);
119
+ hasMoreToolCalls = !executedToolBatch.terminate;
120
+ for (const result of toolResults) {
121
+ currentContext.messages.push(result);
122
+ newMessages.push(result);
123
+ }
124
+ }
125
+ await emit({ type: "turn_end", message, toolResults });
126
+ const nextTurnContext = {
127
+ message,
128
+ toolResults,
129
+ context: currentContext,
130
+ newMessages,
131
+ };
132
+ const nextTurnSnapshot = await config.prepareNextTurn?.(nextTurnContext);
133
+ if (nextTurnSnapshot) {
134
+ currentContext = nextTurnSnapshot.context ?? currentContext;
135
+ config = {
136
+ ...config,
137
+ model: nextTurnSnapshot.model ?? config.model,
138
+ reasoning: nextTurnSnapshot.thinkingLevel === undefined
139
+ ? config.reasoning
140
+ : nextTurnSnapshot.thinkingLevel === "off"
141
+ ? undefined
142
+ : nextTurnSnapshot.thinkingLevel,
143
+ };
144
+ }
145
+ if (await config.shouldStopAfterTurn?.({
146
+ message,
147
+ toolResults,
148
+ context: currentContext,
149
+ newMessages,
150
+ })) {
151
+ await emit({ type: "agent_end", messages: newMessages });
152
+ return;
153
+ }
154
+ pendingMessages = (await config.getSteeringMessages?.()) || [];
155
+ }
156
+ // Agent would stop here. Check for follow-up messages.
157
+ const followUpMessages = (await config.getFollowUpMessages?.()) || [];
158
+ if (followUpMessages.length > 0) {
159
+ // Set as pending so inner loop processes them
160
+ pendingMessages = followUpMessages;
161
+ continue;
162
+ }
163
+ // No more messages, exit
164
+ break;
165
+ }
166
+ await emit({ type: "agent_end", messages: newMessages });
167
+ }
168
+ /**
169
+ * Stream an assistant response from the LLM.
170
+ * This is where AgentMessage[] gets transformed to Message[] for the LLM.
171
+ */
172
+ async function streamAssistantResponse(context, config, signal, emit, streamFn) {
173
+ // Apply context transform if configured (AgentMessage[] → AgentMessage[])
174
+ let messages = context.messages;
175
+ if (config.transformContext) {
176
+ messages = await config.transformContext(messages, signal);
177
+ }
178
+ // Convert to LLM-compatible messages (AgentMessage[] → Message[])
179
+ const llmMessages = await config.convertToLlm(messages);
180
+ // Build LLM context
181
+ const llmContext = {
182
+ systemPrompt: context.systemPrompt,
183
+ messages: llmMessages,
184
+ tools: context.tools,
185
+ };
186
+ const streamFunction = streamFn || streamSimple;
187
+ // Resolve API key (important for expiring tokens)
188
+ const resolvedApiKey = (config.getApiKey ? await config.getApiKey(config.model.provider) : undefined) || config.apiKey;
189
+ const response = await streamFunction(config.model, llmContext, {
190
+ ...config,
191
+ apiKey: resolvedApiKey,
192
+ signal,
193
+ });
194
+ let partialMessage = null;
195
+ let addedPartial = false;
196
+ for await (const event of response) {
197
+ switch (event.type) {
198
+ case "start":
199
+ partialMessage = event.partial;
200
+ context.messages.push(partialMessage);
201
+ addedPartial = true;
202
+ await emit({ type: "message_start", message: { ...partialMessage } });
203
+ break;
204
+ case "text_start":
205
+ case "text_delta":
206
+ case "text_end":
207
+ case "thinking_start":
208
+ case "thinking_delta":
209
+ case "thinking_end":
210
+ case "toolcall_start":
211
+ case "toolcall_delta":
212
+ case "toolcall_end":
213
+ if (partialMessage) {
214
+ partialMessage = event.partial;
215
+ context.messages[context.messages.length - 1] = partialMessage;
216
+ await emit({
217
+ type: "message_update",
218
+ assistantMessageEvent: event,
219
+ message: { ...partialMessage },
220
+ });
221
+ }
222
+ break;
223
+ case "done":
224
+ case "error": {
225
+ const finalMessage = await response.result();
226
+ if (addedPartial) {
227
+ context.messages[context.messages.length - 1] = finalMessage;
228
+ }
229
+ else {
230
+ context.messages.push(finalMessage);
231
+ }
232
+ if (!addedPartial) {
233
+ await emit({ type: "message_start", message: { ...finalMessage } });
234
+ }
235
+ await emit({ type: "message_end", message: finalMessage });
236
+ return finalMessage;
237
+ }
238
+ }
239
+ }
240
+ const finalMessage = await response.result();
241
+ if (addedPartial) {
242
+ context.messages[context.messages.length - 1] = finalMessage;
243
+ }
244
+ else {
245
+ context.messages.push(finalMessage);
246
+ await emit({ type: "message_start", message: { ...finalMessage } });
247
+ }
248
+ await emit({ type: "message_end", message: finalMessage });
249
+ return finalMessage;
250
+ }
251
+ /**
252
+ * Execute tool calls from an assistant message.
253
+ */
254
+ async function executeToolCalls(currentContext, assistantMessage, config, signal, emit) {
255
+ const toolCalls = assistantMessage.content.filter((c) => c.type === "toolCall");
256
+ const hasSequentialToolCall = toolCalls.some((tc) => currentContext.tools?.find((t) => t.name === tc.name)?.executionMode === "sequential");
257
+ if (config.toolExecution === "sequential" || hasSequentialToolCall) {
258
+ return executeToolCallsSequential(currentContext, assistantMessage, toolCalls, config, signal, emit);
259
+ }
260
+ return executeToolCallsParallel(currentContext, assistantMessage, toolCalls, config, signal, emit);
261
+ }
262
+ async function executeToolCallsSequential(currentContext, assistantMessage, toolCalls, config, signal, emit) {
263
+ const finalizedCalls = [];
264
+ const messages = [];
265
+ for (const toolCall of toolCalls) {
266
+ await emit({
267
+ type: "tool_execution_start",
268
+ toolCallId: toolCall.id,
269
+ toolName: toolCall.name,
270
+ args: toolCall.arguments,
271
+ });
272
+ const preparation = await prepareToolCall(currentContext, assistantMessage, toolCall, config, signal);
273
+ let finalized;
274
+ if (preparation.kind === "immediate") {
275
+ finalized = {
276
+ toolCall,
277
+ result: preparation.result,
278
+ isError: preparation.isError,
279
+ };
280
+ }
281
+ else {
282
+ const executed = await executePreparedToolCall(preparation, signal, emit);
283
+ finalized = await finalizeExecutedToolCall(currentContext, assistantMessage, preparation, executed, config, signal);
284
+ }
285
+ await emitToolExecutionEnd(finalized, emit);
286
+ const toolResultMessage = createToolResultMessage(finalized);
287
+ await emitToolResultMessage(toolResultMessage, emit);
288
+ finalizedCalls.push(finalized);
289
+ messages.push(toolResultMessage);
290
+ if (signal?.aborted) {
291
+ break;
292
+ }
293
+ }
294
+ return {
295
+ messages,
296
+ terminate: shouldTerminateToolBatch(finalizedCalls),
297
+ };
298
+ }
299
+ async function executeToolCallsParallel(currentContext, assistantMessage, toolCalls, config, signal, emit) {
300
+ const finalizedCalls = [];
301
+ for (const toolCall of toolCalls) {
302
+ await emit({
303
+ type: "tool_execution_start",
304
+ toolCallId: toolCall.id,
305
+ toolName: toolCall.name,
306
+ args: toolCall.arguments,
307
+ });
308
+ const preparation = await prepareToolCall(currentContext, assistantMessage, toolCall, config, signal);
309
+ if (preparation.kind === "immediate") {
310
+ const finalized = {
311
+ toolCall,
312
+ result: preparation.result,
313
+ isError: preparation.isError,
314
+ };
315
+ await emitToolExecutionEnd(finalized, emit);
316
+ finalizedCalls.push(finalized);
317
+ if (signal?.aborted) {
318
+ break;
319
+ }
320
+ continue;
321
+ }
322
+ finalizedCalls.push(async () => {
323
+ const executed = await executePreparedToolCall(preparation, signal, emit);
324
+ const finalized = await finalizeExecutedToolCall(currentContext, assistantMessage, preparation, executed, config, signal);
325
+ await emitToolExecutionEnd(finalized, emit);
326
+ return finalized;
327
+ });
328
+ if (signal?.aborted) {
329
+ break;
330
+ }
331
+ }
332
+ const orderedFinalizedCalls = await Promise.all(finalizedCalls.map((entry) => (typeof entry === "function" ? entry() : Promise.resolve(entry))));
333
+ const messages = [];
334
+ for (const finalized of orderedFinalizedCalls) {
335
+ const toolResultMessage = createToolResultMessage(finalized);
336
+ await emitToolResultMessage(toolResultMessage, emit);
337
+ messages.push(toolResultMessage);
338
+ }
339
+ return {
340
+ messages,
341
+ terminate: shouldTerminateToolBatch(orderedFinalizedCalls),
342
+ };
343
+ }
344
+ function shouldTerminateToolBatch(finalizedCalls) {
345
+ return finalizedCalls.length > 0 && finalizedCalls.every((finalized) => finalized.result.terminate === true);
346
+ }
347
+ function prepareToolCallArguments(tool, toolCall) {
348
+ if (!tool.prepareArguments) {
349
+ return toolCall;
350
+ }
351
+ const preparedArguments = tool.prepareArguments(toolCall.arguments);
352
+ if (preparedArguments === toolCall.arguments) {
353
+ return toolCall;
354
+ }
355
+ return {
356
+ ...toolCall,
357
+ arguments: preparedArguments,
358
+ };
359
+ }
360
+ async function prepareToolCall(currentContext, assistantMessage, toolCall, config, signal) {
361
+ const tool = currentContext.tools?.find((t) => t.name === toolCall.name);
362
+ if (!tool) {
363
+ return {
364
+ kind: "immediate",
365
+ result: createErrorToolResult(`Tool ${toolCall.name} not found`),
366
+ isError: true,
367
+ };
368
+ }
369
+ try {
370
+ const preparedToolCall = prepareToolCallArguments(tool, toolCall);
371
+ const validatedArgs = validateToolArguments(tool, preparedToolCall);
372
+ if (config.beforeToolCall) {
373
+ const beforeResult = await config.beforeToolCall({
374
+ assistantMessage,
375
+ toolCall,
376
+ args: validatedArgs,
377
+ context: currentContext,
378
+ }, signal);
379
+ if (signal?.aborted) {
380
+ return {
381
+ kind: "immediate",
382
+ result: createErrorToolResult("Operation aborted"),
383
+ isError: true,
384
+ };
385
+ }
386
+ if (beforeResult?.block) {
387
+ return {
388
+ kind: "immediate",
389
+ result: createErrorToolResult(beforeResult.reason || "Tool execution was blocked"),
390
+ isError: true,
391
+ };
392
+ }
393
+ }
394
+ if (signal?.aborted) {
395
+ return {
396
+ kind: "immediate",
397
+ result: createErrorToolResult("Operation aborted"),
398
+ isError: true,
399
+ };
400
+ }
401
+ return {
402
+ kind: "prepared",
403
+ toolCall,
404
+ tool,
405
+ args: validatedArgs,
406
+ };
407
+ }
408
+ catch (error) {
409
+ return {
410
+ kind: "immediate",
411
+ result: createErrorToolResult(error instanceof Error ? error.message : String(error)),
412
+ isError: true,
413
+ };
414
+ }
415
+ }
416
+ async function executePreparedToolCall(prepared, signal, emit) {
417
+ const updateEvents = [];
418
+ try {
419
+ const result = await prepared.tool.execute(prepared.toolCall.id, prepared.args, signal, (partialResult) => {
420
+ updateEvents.push(Promise.resolve(emit({
421
+ type: "tool_execution_update",
422
+ toolCallId: prepared.toolCall.id,
423
+ toolName: prepared.toolCall.name,
424
+ args: prepared.toolCall.arguments,
425
+ partialResult,
426
+ })));
427
+ });
428
+ await Promise.all(updateEvents);
429
+ return { result, isError: false };
430
+ }
431
+ catch (error) {
432
+ await Promise.all(updateEvents);
433
+ return {
434
+ result: createErrorToolResult(error instanceof Error ? error.message : String(error)),
435
+ isError: true,
436
+ };
437
+ }
438
+ }
439
+ async function finalizeExecutedToolCall(currentContext, assistantMessage, prepared, executed, config, signal) {
440
+ let result = executed.result;
441
+ let isError = executed.isError;
442
+ if (config.afterToolCall) {
443
+ try {
444
+ const afterResult = await config.afterToolCall({
445
+ assistantMessage,
446
+ toolCall: prepared.toolCall,
447
+ args: prepared.args,
448
+ result,
449
+ isError,
450
+ context: currentContext,
451
+ }, signal);
452
+ if (afterResult) {
453
+ result = {
454
+ content: afterResult.content ?? result.content,
455
+ details: afterResult.details ?? result.details,
456
+ terminate: afterResult.terminate ?? result.terminate,
457
+ };
458
+ isError = afterResult.isError ?? isError;
459
+ }
460
+ }
461
+ catch (error) {
462
+ result = createErrorToolResult(error instanceof Error ? error.message : String(error));
463
+ isError = true;
464
+ }
465
+ }
466
+ return {
467
+ toolCall: prepared.toolCall,
468
+ result,
469
+ isError,
470
+ };
471
+ }
472
+ function createErrorToolResult(message) {
473
+ return {
474
+ content: [{ type: "text", text: message }],
475
+ details: {},
476
+ };
477
+ }
478
+ async function emitToolExecutionEnd(finalized, emit) {
479
+ await emit({
480
+ type: "tool_execution_end",
481
+ toolCallId: finalized.toolCall.id,
482
+ toolName: finalized.toolCall.name,
483
+ result: finalized.result,
484
+ isError: finalized.isError,
485
+ });
486
+ }
487
+ function createToolResultMessage(finalized) {
488
+ return {
489
+ role: "toolResult",
490
+ toolCallId: finalized.toolCall.id,
491
+ toolName: finalized.toolCall.name,
492
+ content: finalized.result.content,
493
+ details: finalized.result.details,
494
+ isError: finalized.isError,
495
+ timestamp: Date.now(),
496
+ };
497
+ }
498
+ async function emitToolResultMessage(toolResultMessage, emit) {
499
+ await emit({ type: "message_start", message: toolResultMessage });
500
+ await emit({ type: "message_end", message: toolResultMessage });
501
+ }