cerebras-cli 1.0.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 (314) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +10 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +4 -0
  6. package/package.json +128 -0
  7. package/parsers-config.ts +239 -0
  8. package/script/build.ts +151 -0
  9. package/script/postinstall.mjs +122 -0
  10. package/script/publish.ts +256 -0
  11. package/script/schema.ts +47 -0
  12. package/src/acp/README.md +164 -0
  13. package/src/acp/agent.ts +812 -0
  14. package/src/acp/session.ts +70 -0
  15. package/src/acp/types.ts +22 -0
  16. package/src/agent/agent.ts +310 -0
  17. package/src/agent/generate.txt +75 -0
  18. package/src/auth/index.ts +70 -0
  19. package/src/bun/index.ts +152 -0
  20. package/src/bus/global.ts +10 -0
  21. package/src/bus/index.ts +142 -0
  22. package/src/cli/bootstrap.ts +17 -0
  23. package/src/cli/cmd/acp.ts +88 -0
  24. package/src/cli/cmd/agent.ts +165 -0
  25. package/src/cli/cmd/auth.ts +369 -0
  26. package/src/cli/cmd/cmd.ts +7 -0
  27. package/src/cli/cmd/debug/config.ts +15 -0
  28. package/src/cli/cmd/debug/file.ts +91 -0
  29. package/src/cli/cmd/debug/index.ts +41 -0
  30. package/src/cli/cmd/debug/lsp.ts +47 -0
  31. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  32. package/src/cli/cmd/debug/scrap.ts +15 -0
  33. package/src/cli/cmd/debug/snapshot.ts +48 -0
  34. package/src/cli/cmd/export.ts +88 -0
  35. package/src/cli/cmd/generate.ts +38 -0
  36. package/src/cli/cmd/github.ts +1200 -0
  37. package/src/cli/cmd/import.ts +98 -0
  38. package/src/cli/cmd/mcp.ts +400 -0
  39. package/src/cli/cmd/models.ts +77 -0
  40. package/src/cli/cmd/pr.ts +112 -0
  41. package/src/cli/cmd/run.ts +342 -0
  42. package/src/cli/cmd/serve.ts +31 -0
  43. package/src/cli/cmd/session.ts +106 -0
  44. package/src/cli/cmd/stats.ts +298 -0
  45. package/src/cli/cmd/tui/app.tsx +732 -0
  46. package/src/cli/cmd/tui/attach.ts +25 -0
  47. package/src/cli/cmd/tui/component/border.tsx +21 -0
  48. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  49. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  50. package/src/cli/cmd/tui/component/dialog-feedback.tsx +160 -0
  51. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  52. package/src/cli/cmd/tui/component/dialog-model.tsx +223 -0
  53. package/src/cli/cmd/tui/component/dialog-notification.tsx +78 -0
  54. package/src/cli/cmd/tui/component/dialog-provider.tsx +222 -0
  55. package/src/cli/cmd/tui/component/dialog-session-list.tsx +97 -0
  56. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  57. package/src/cli/cmd/tui/component/dialog-status.tsx +114 -0
  58. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  59. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  60. package/src/cli/cmd/tui/component/logo.tsx +37 -0
  61. package/src/cli/cmd/tui/component/notification-banner.tsx +58 -0
  62. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +530 -0
  63. package/src/cli/cmd/tui/component/prompt/history.tsx +107 -0
  64. package/src/cli/cmd/tui/component/prompt/index.tsx +931 -0
  65. package/src/cli/cmd/tui/context/args.tsx +14 -0
  66. package/src/cli/cmd/tui/context/directory.ts +12 -0
  67. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  68. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  69. package/src/cli/cmd/tui/context/keybind.tsx +111 -0
  70. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  71. package/src/cli/cmd/tui/context/local.tsx +339 -0
  72. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  73. package/src/cli/cmd/tui/context/route.tsx +45 -0
  74. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  75. package/src/cli/cmd/tui/context/sync.tsx +374 -0
  76. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  77. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  78. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  79. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  80. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  81. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  82. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  83. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  84. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  85. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  86. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  87. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  88. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  89. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  90. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  91. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  92. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  93. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  94. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  95. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  96. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  97. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  98. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  99. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  100. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  101. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  102. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  103. package/src/cli/cmd/tui/context/theme.tsx +1077 -0
  104. package/src/cli/cmd/tui/event.ts +39 -0
  105. package/src/cli/cmd/tui/routes/home.tsx +104 -0
  106. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +93 -0
  107. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +37 -0
  108. package/src/cli/cmd/tui/routes/session/footer.tsx +76 -0
  109. package/src/cli/cmd/tui/routes/session/header.tsx +183 -0
  110. package/src/cli/cmd/tui/routes/session/index.tsx +1703 -0
  111. package/src/cli/cmd/tui/routes/session/sidebar.tsx +586 -0
  112. package/src/cli/cmd/tui/spawn.ts +60 -0
  113. package/src/cli/cmd/tui/thread.ts +120 -0
  114. package/src/cli/cmd/tui/ui/dialog-alert.tsx +55 -0
  115. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +81 -0
  116. package/src/cli/cmd/tui/ui/dialog-help.tsx +36 -0
  117. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +75 -0
  118. package/src/cli/cmd/tui/ui/dialog-select.tsx +317 -0
  119. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  120. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  121. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  122. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  123. package/src/cli/cmd/tui/util/editor.ts +32 -0
  124. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  125. package/src/cli/cmd/tui/worker.ts +63 -0
  126. package/src/cli/cmd/uninstall.ts +344 -0
  127. package/src/cli/cmd/upgrade.ts +67 -0
  128. package/src/cli/cmd/web.ts +84 -0
  129. package/src/cli/error.ts +55 -0
  130. package/src/cli/ui.ts +84 -0
  131. package/src/cli/upgrade.ts +25 -0
  132. package/src/command/index.ts +79 -0
  133. package/src/command/template/initialize.txt +10 -0
  134. package/src/command/template/review.txt +73 -0
  135. package/src/config/config.ts +886 -0
  136. package/src/config/markdown.ts +41 -0
  137. package/src/env/index.ts +26 -0
  138. package/src/file/fzf.ts +124 -0
  139. package/src/file/ignore.ts +83 -0
  140. package/src/file/index.ts +326 -0
  141. package/src/file/ripgrep.ts +391 -0
  142. package/src/file/time.ts +38 -0
  143. package/src/file/watcher.ts +89 -0
  144. package/src/flag/flag.ts +28 -0
  145. package/src/format/formatter.ts +277 -0
  146. package/src/format/index.ts +137 -0
  147. package/src/global/index.ts +52 -0
  148. package/src/id/id.ts +73 -0
  149. package/src/ide/index.ts +75 -0
  150. package/src/index.ts +158 -0
  151. package/src/installation/index.ts +194 -0
  152. package/src/lsp/client.ts +215 -0
  153. package/src/lsp/index.ts +370 -0
  154. package/src/lsp/language.ts +111 -0
  155. package/src/lsp/server.ts +1327 -0
  156. package/src/mcp/auth.ts +82 -0
  157. package/src/mcp/index.ts +576 -0
  158. package/src/mcp/oauth-callback.ts +203 -0
  159. package/src/mcp/oauth-provider.ts +132 -0
  160. package/src/notification/index.ts +101 -0
  161. package/src/patch/index.ts +622 -0
  162. package/src/permission/index.ts +198 -0
  163. package/src/plugin/index.ts +95 -0
  164. package/src/project/bootstrap.ts +31 -0
  165. package/src/project/instance.ts +68 -0
  166. package/src/project/project.ts +133 -0
  167. package/src/project/state.ts +65 -0
  168. package/src/project/vcs.ts +77 -0
  169. package/src/provider/auth.ts +143 -0
  170. package/src/provider/models-macro.ts +11 -0
  171. package/src/provider/models.ts +93 -0
  172. package/src/provider/provider.ts +996 -0
  173. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  174. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  175. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  176. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  177. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +27 -0
  178. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  179. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  180. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  181. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  182. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  183. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  184. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  185. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  186. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  187. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  190. package/src/provider/transform.ts +406 -0
  191. package/src/pty/index.ts +226 -0
  192. package/src/ratelimit/index.ts +185 -0
  193. package/src/server/error.ts +36 -0
  194. package/src/server/project.ts +50 -0
  195. package/src/server/server.ts +2463 -0
  196. package/src/server/tui.ts +71 -0
  197. package/src/session/compaction.ts +257 -0
  198. package/src/session/index.ts +470 -0
  199. package/src/session/message-v2.ts +641 -0
  200. package/src/session/message.ts +189 -0
  201. package/src/session/processor.ts +443 -0
  202. package/src/session/prompt/anthropic-20250930.txt +166 -0
  203. package/src/session/prompt/anthropic.txt +105 -0
  204. package/src/session/prompt/anthropic_spoof.txt +1 -0
  205. package/src/session/prompt/beast.txt +147 -0
  206. package/src/session/prompt/build-switch.txt +5 -0
  207. package/src/session/prompt/codex.txt +318 -0
  208. package/src/session/prompt/compaction.txt +12 -0
  209. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  210. package/src/session/prompt/gemini.txt +155 -0
  211. package/src/session/prompt/max-steps.txt +16 -0
  212. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  213. package/src/session/prompt/plan.txt +26 -0
  214. package/src/session/prompt/polaris.txt +107 -0
  215. package/src/session/prompt/qwen.txt +109 -0
  216. package/src/session/prompt/summarize.txt +4 -0
  217. package/src/session/prompt/title.txt +36 -0
  218. package/src/session/prompt.ts +1541 -0
  219. package/src/session/retry.ts +82 -0
  220. package/src/session/revert.ts +108 -0
  221. package/src/session/status.ts +75 -0
  222. package/src/session/summary.ts +203 -0
  223. package/src/session/system.ts +148 -0
  224. package/src/session/todo.ts +36 -0
  225. package/src/share/share-next.ts +195 -0
  226. package/src/share/share.ts +87 -0
  227. package/src/snapshot/index.ts +197 -0
  228. package/src/storage/storage.ts +226 -0
  229. package/src/telemetry/index.ts +232 -0
  230. package/src/tool/bash.ts +365 -0
  231. package/src/tool/bash.txt +128 -0
  232. package/src/tool/batch.ts +173 -0
  233. package/src/tool/batch.txt +28 -0
  234. package/src/tool/codesearch.ts +138 -0
  235. package/src/tool/codesearch.txt +12 -0
  236. package/src/tool/edit.ts +674 -0
  237. package/src/tool/edit.txt +10 -0
  238. package/src/tool/glob.ts +65 -0
  239. package/src/tool/glob.txt +6 -0
  240. package/src/tool/grep.ts +120 -0
  241. package/src/tool/grep.txt +8 -0
  242. package/src/tool/invalid.ts +17 -0
  243. package/src/tool/ls.ts +110 -0
  244. package/src/tool/ls.txt +1 -0
  245. package/src/tool/lsp-diagnostics.ts +26 -0
  246. package/src/tool/lsp-diagnostics.txt +1 -0
  247. package/src/tool/lsp-hover.ts +31 -0
  248. package/src/tool/lsp-hover.txt +1 -0
  249. package/src/tool/multiedit.ts +46 -0
  250. package/src/tool/multiedit.txt +41 -0
  251. package/src/tool/patch.ts +233 -0
  252. package/src/tool/patch.txt +1 -0
  253. package/src/tool/read.ts +217 -0
  254. package/src/tool/read.txt +12 -0
  255. package/src/tool/registry.ts +148 -0
  256. package/src/tool/task.ts +135 -0
  257. package/src/tool/task.txt +60 -0
  258. package/src/tool/todo.ts +39 -0
  259. package/src/tool/todoread.txt +14 -0
  260. package/src/tool/todowrite.txt +167 -0
  261. package/src/tool/tool.ts +66 -0
  262. package/src/tool/webfetch.ts +187 -0
  263. package/src/tool/webfetch.txt +14 -0
  264. package/src/tool/websearch.ts +150 -0
  265. package/src/tool/websearch.txt +11 -0
  266. package/src/tool/write.ts +99 -0
  267. package/src/tool/write.txt +8 -0
  268. package/src/types/shims.d.ts +3 -0
  269. package/src/util/color.ts +19 -0
  270. package/src/util/context.ts +25 -0
  271. package/src/util/defer.ts +12 -0
  272. package/src/util/eventloop.ts +20 -0
  273. package/src/util/filesystem.ts +69 -0
  274. package/src/util/fn.ts +11 -0
  275. package/src/util/iife.ts +3 -0
  276. package/src/util/keybind.ts +79 -0
  277. package/src/util/lazy.ts +11 -0
  278. package/src/util/locale.ts +81 -0
  279. package/src/util/lock.ts +98 -0
  280. package/src/util/log.ts +177 -0
  281. package/src/util/queue.ts +32 -0
  282. package/src/util/rpc.ts +42 -0
  283. package/src/util/scrap.ts +10 -0
  284. package/src/util/signal.ts +12 -0
  285. package/src/util/timeout.ts +14 -0
  286. package/src/util/token.ts +7 -0
  287. package/src/util/wildcard.ts +54 -0
  288. package/sst-env.d.ts +9 -0
  289. package/test/bun.test.ts +53 -0
  290. package/test/config/agent-color.test.ts +66 -0
  291. package/test/config/config.test.ts +503 -0
  292. package/test/config/markdown.test.ts +89 -0
  293. package/test/file/ignore.test.ts +10 -0
  294. package/test/fixture/fixture.ts +28 -0
  295. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  296. package/test/ide/ide.test.ts +82 -0
  297. package/test/keybind.test.ts +317 -0
  298. package/test/lsp/client.test.ts +95 -0
  299. package/test/patch/patch.test.ts +348 -0
  300. package/test/preload.ts +38 -0
  301. package/test/project/project.test.ts +42 -0
  302. package/test/provider/provider.test.ts +1809 -0
  303. package/test/provider/transform.test.ts +305 -0
  304. package/test/session/retry.test.ts +61 -0
  305. package/test/session/session.test.ts +71 -0
  306. package/test/snapshot/snapshot.test.ts +939 -0
  307. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  308. package/test/tool/bash.test.ts +55 -0
  309. package/test/tool/patch.test.ts +259 -0
  310. package/test/util/iife.test.ts +36 -0
  311. package/test/util/lazy.test.ts +50 -0
  312. package/test/util/timeout.test.ts +21 -0
  313. package/test/util/wildcard.test.ts +55 -0
  314. package/tsconfig.json +17 -0
@@ -0,0 +1,189 @@
1
+ import z from "zod"
2
+ import { NamedError } from "@opencode-ai/util/error"
3
+
4
+ export namespace Message {
5
+ export const OutputLengthError = NamedError.create("MessageOutputLengthError", z.object({}))
6
+ export const AuthError = NamedError.create(
7
+ "ProviderAuthError",
8
+ z.object({
9
+ providerID: z.string(),
10
+ message: z.string(),
11
+ }),
12
+ )
13
+
14
+ export const ToolCall = z
15
+ .object({
16
+ state: z.literal("call"),
17
+ step: z.number().optional(),
18
+ toolCallId: z.string(),
19
+ toolName: z.string(),
20
+ args: z.custom<Required<unknown>>(),
21
+ })
22
+ .meta({
23
+ ref: "ToolCall",
24
+ })
25
+ export type ToolCall = z.infer<typeof ToolCall>
26
+
27
+ export const ToolPartialCall = z
28
+ .object({
29
+ state: z.literal("partial-call"),
30
+ step: z.number().optional(),
31
+ toolCallId: z.string(),
32
+ toolName: z.string(),
33
+ args: z.custom<Required<unknown>>(),
34
+ })
35
+ .meta({
36
+ ref: "ToolPartialCall",
37
+ })
38
+ export type ToolPartialCall = z.infer<typeof ToolPartialCall>
39
+
40
+ export const ToolResult = z
41
+ .object({
42
+ state: z.literal("result"),
43
+ step: z.number().optional(),
44
+ toolCallId: z.string(),
45
+ toolName: z.string(),
46
+ args: z.custom<Required<unknown>>(),
47
+ result: z.string(),
48
+ })
49
+ .meta({
50
+ ref: "ToolResult",
51
+ })
52
+ export type ToolResult = z.infer<typeof ToolResult>
53
+
54
+ export const ToolInvocation = z.discriminatedUnion("state", [ToolCall, ToolPartialCall, ToolResult]).meta({
55
+ ref: "ToolInvocation",
56
+ })
57
+ export type ToolInvocation = z.infer<typeof ToolInvocation>
58
+
59
+ export const TextPart = z
60
+ .object({
61
+ type: z.literal("text"),
62
+ text: z.string(),
63
+ })
64
+ .meta({
65
+ ref: "TextPart",
66
+ })
67
+ export type TextPart = z.infer<typeof TextPart>
68
+
69
+ export const ReasoningPart = z
70
+ .object({
71
+ type: z.literal("reasoning"),
72
+ text: z.string(),
73
+ providerMetadata: z.record(z.string(), z.any()).optional(),
74
+ })
75
+ .meta({
76
+ ref: "ReasoningPart",
77
+ })
78
+ export type ReasoningPart = z.infer<typeof ReasoningPart>
79
+
80
+ export const ToolInvocationPart = z
81
+ .object({
82
+ type: z.literal("tool-invocation"),
83
+ toolInvocation: ToolInvocation,
84
+ })
85
+ .meta({
86
+ ref: "ToolInvocationPart",
87
+ })
88
+ export type ToolInvocationPart = z.infer<typeof ToolInvocationPart>
89
+
90
+ export const SourceUrlPart = z
91
+ .object({
92
+ type: z.literal("source-url"),
93
+ sourceId: z.string(),
94
+ url: z.string(),
95
+ title: z.string().optional(),
96
+ providerMetadata: z.record(z.string(), z.any()).optional(),
97
+ })
98
+ .meta({
99
+ ref: "SourceUrlPart",
100
+ })
101
+ export type SourceUrlPart = z.infer<typeof SourceUrlPart>
102
+
103
+ export const FilePart = z
104
+ .object({
105
+ type: z.literal("file"),
106
+ mediaType: z.string(),
107
+ filename: z.string().optional(),
108
+ url: z.string(),
109
+ })
110
+ .meta({
111
+ ref: "FilePart",
112
+ })
113
+ export type FilePart = z.infer<typeof FilePart>
114
+
115
+ export const StepStartPart = z
116
+ .object({
117
+ type: z.literal("step-start"),
118
+ })
119
+ .meta({
120
+ ref: "StepStartPart",
121
+ })
122
+ export type StepStartPart = z.infer<typeof StepStartPart>
123
+
124
+ export const MessagePart = z
125
+ .discriminatedUnion("type", [TextPart, ReasoningPart, ToolInvocationPart, SourceUrlPart, FilePart, StepStartPart])
126
+ .meta({
127
+ ref: "MessagePart",
128
+ })
129
+ export type MessagePart = z.infer<typeof MessagePart>
130
+
131
+ export const Info = z
132
+ .object({
133
+ id: z.string(),
134
+ role: z.enum(["user", "assistant"]),
135
+ parts: z.array(MessagePart),
136
+ metadata: z
137
+ .object({
138
+ time: z.object({
139
+ created: z.number(),
140
+ completed: z.number().optional(),
141
+ }),
142
+ error: z
143
+ .discriminatedUnion("name", [AuthError.Schema, NamedError.Unknown.Schema, OutputLengthError.Schema])
144
+ .optional(),
145
+ sessionID: z.string(),
146
+ tool: z.record(
147
+ z.string(),
148
+ z
149
+ .object({
150
+ title: z.string(),
151
+ snapshot: z.string().optional(),
152
+ time: z.object({
153
+ start: z.number(),
154
+ end: z.number(),
155
+ }),
156
+ })
157
+ .catchall(z.any()),
158
+ ),
159
+ assistant: z
160
+ .object({
161
+ system: z.string().array(),
162
+ modelID: z.string(),
163
+ providerID: z.string(),
164
+ path: z.object({
165
+ cwd: z.string(),
166
+ root: z.string(),
167
+ }),
168
+ cost: z.number(),
169
+ summary: z.boolean().optional(),
170
+ tokens: z.object({
171
+ input: z.number(),
172
+ output: z.number(),
173
+ reasoning: z.number(),
174
+ cache: z.object({
175
+ read: z.number(),
176
+ write: z.number(),
177
+ }),
178
+ }),
179
+ })
180
+ .optional(),
181
+ snapshot: z.string().optional(),
182
+ })
183
+ .meta({ ref: "MessageMetadata" }),
184
+ })
185
+ .meta({
186
+ ref: "Message",
187
+ })
188
+ export type Info = z.infer<typeof Info>
189
+ }
@@ -0,0 +1,443 @@
1
+ import { MessageV2 } from "./message-v2"
2
+ import { streamText } from "ai"
3
+ import { Log } from "@/util/log"
4
+ import { Identifier } from "@/id/id"
5
+ import { Session } from "."
6
+ import { Agent } from "@/agent/agent"
7
+ import { Permission } from "@/permission"
8
+ import { Snapshot } from "@/snapshot"
9
+ import { SessionSummary } from "./summary"
10
+ import { Bus } from "@/bus"
11
+ import { SessionRetry } from "./retry"
12
+ import { SessionStatus } from "./status"
13
+ import { Plugin } from "@/plugin"
14
+ import type { Provider } from "@/provider/provider"
15
+ import { Telemetry } from "@/telemetry"
16
+
17
+ export namespace SessionProcessor {
18
+ const DOOM_LOOP_THRESHOLD = 3
19
+ const log = Log.create({ service: "session.processor" })
20
+
21
+ export type Info = Awaited<ReturnType<typeof create>>
22
+ export type Result = Awaited<ReturnType<Info["process"]>>
23
+
24
+ export type StreamInput = Parameters<typeof streamText>[0]
25
+
26
+ export type TBD = {
27
+ model: {
28
+ modelID: string
29
+ providerID: string
30
+ }
31
+ }
32
+
33
+ export function create(input: {
34
+ assistantMessage: MessageV2.Assistant
35
+ sessionID: string
36
+ model: Provider.Model
37
+ abort: AbortSignal
38
+ conversationTurns?: number
39
+ }) {
40
+ const toolcalls: Record<string, MessageV2.ToolPart> = {}
41
+ let snapshot: string | undefined
42
+ let blocked = false
43
+ let attempt = 0
44
+
45
+ const result = {
46
+ get message() {
47
+ return input.assistantMessage
48
+ },
49
+ partFromToolCall(toolCallID: string) {
50
+ return toolcalls[toolCallID]
51
+ },
52
+ async process(streamInput: StreamInput) {
53
+ log.info("process")
54
+ while (true) {
55
+ try {
56
+ let currentText: MessageV2.TextPart | undefined
57
+ let reasoningMap: Record<string, MessageV2.ReasoningPart> = {}
58
+ const stream = streamText(streamInput)
59
+
60
+ for await (const value of stream.fullStream) {
61
+ input.abort.throwIfAborted()
62
+ switch (value.type) {
63
+ case "start":
64
+ SessionStatus.set(input.sessionID, { type: "busy" })
65
+ break
66
+
67
+ case "reasoning-start":
68
+ if (value.id in reasoningMap) {
69
+ continue
70
+ }
71
+ reasoningMap[value.id] = {
72
+ id: Identifier.ascending("part"),
73
+ messageID: input.assistantMessage.id,
74
+ sessionID: input.assistantMessage.sessionID,
75
+ type: "reasoning",
76
+ text: "",
77
+ time: {
78
+ start: Date.now(),
79
+ },
80
+ metadata: value.providerMetadata,
81
+ }
82
+ break
83
+
84
+ case "reasoning-delta":
85
+ if (value.id in reasoningMap) {
86
+ const part = reasoningMap[value.id]
87
+ part.text += value.text
88
+ if (value.providerMetadata) part.metadata = value.providerMetadata
89
+ if (part.text) await Session.updatePart({ part, delta: value.text })
90
+ }
91
+ break
92
+
93
+ case "reasoning-end":
94
+ if (value.id in reasoningMap) {
95
+ const part = reasoningMap[value.id]
96
+ part.text = part.text.trimEnd()
97
+
98
+ part.time = {
99
+ ...part.time,
100
+ end: Date.now(),
101
+ }
102
+ if (value.providerMetadata) part.metadata = value.providerMetadata
103
+ await Session.updatePart(part)
104
+ delete reasoningMap[value.id]
105
+ }
106
+ break
107
+
108
+ case "tool-input-start":
109
+ const part = await Session.updatePart({
110
+ id: toolcalls[value.id]?.id ?? Identifier.ascending("part"),
111
+ messageID: input.assistantMessage.id,
112
+ sessionID: input.assistantMessage.sessionID,
113
+ type: "tool",
114
+ tool: value.toolName,
115
+ callID: value.id,
116
+ state: {
117
+ status: "pending",
118
+ input: {},
119
+ raw: "",
120
+ },
121
+ })
122
+ toolcalls[value.id] = part as MessageV2.ToolPart
123
+ break
124
+
125
+ case "tool-input-delta":
126
+ break
127
+
128
+ case "tool-input-end":
129
+ break
130
+
131
+ case "tool-call": {
132
+ const match = toolcalls[value.toolCallId]
133
+ if (match) {
134
+ const part = await Session.updatePart({
135
+ ...match,
136
+ tool: value.toolName,
137
+ state: {
138
+ status: "running",
139
+ input: value.input,
140
+ time: {
141
+ start: Date.now(),
142
+ },
143
+ },
144
+ metadata: value.providerMetadata,
145
+ })
146
+ toolcalls[value.toolCallId] = part as MessageV2.ToolPart
147
+
148
+ const parts = await MessageV2.parts(input.assistantMessage.id)
149
+ const lastThree = parts.slice(-DOOM_LOOP_THRESHOLD)
150
+
151
+ if (
152
+ lastThree.length === DOOM_LOOP_THRESHOLD &&
153
+ lastThree.every(
154
+ (p) =>
155
+ p.type === "tool" &&
156
+ p.tool === value.toolName &&
157
+ p.state.status !== "pending" &&
158
+ JSON.stringify(p.state.input) === JSON.stringify(value.input),
159
+ )
160
+ ) {
161
+ const permission = await Agent.get(input.assistantMessage.mode).then((x) => x.permission)
162
+ if (permission.doom_loop === "ask") {
163
+ await Permission.ask({
164
+ type: "doom_loop",
165
+ pattern: value.toolName,
166
+ sessionID: input.assistantMessage.sessionID,
167
+ messageID: input.assistantMessage.id,
168
+ callID: value.toolCallId,
169
+ title: `Possible doom loop: "${value.toolName}" called ${DOOM_LOOP_THRESHOLD} times with identical arguments`,
170
+ metadata: {
171
+ tool: value.toolName,
172
+ input: value.input,
173
+ },
174
+ })
175
+ } else if (permission.doom_loop === "deny") {
176
+ throw new Permission.RejectedError(
177
+ input.assistantMessage.sessionID,
178
+ "doom_loop",
179
+ value.toolCallId,
180
+ {
181
+ tool: value.toolName,
182
+ input: value.input,
183
+ },
184
+ `You seem to be stuck in a doom loop, please stop repeating the same action`,
185
+ )
186
+ }
187
+ }
188
+ }
189
+ break
190
+ }
191
+ case "tool-result": {
192
+ const match = toolcalls[value.toolCallId]
193
+ if (match && match.state.status === "running") {
194
+ await Session.updatePart({
195
+ ...match,
196
+ state: {
197
+ status: "completed",
198
+ input: value.input,
199
+ output: value.output.output,
200
+ metadata: value.output.metadata,
201
+ title: value.output.title,
202
+ time: {
203
+ start: match.state.time.start,
204
+ end: Date.now(),
205
+ },
206
+ attachments: value.output.attachments,
207
+ },
208
+ })
209
+
210
+ delete toolcalls[value.toolCallId]
211
+ }
212
+ break
213
+ }
214
+
215
+ case "tool-error": {
216
+ const match = toolcalls[value.toolCallId]
217
+ if (match && match.state.status === "running") {
218
+ await Session.updatePart({
219
+ ...match,
220
+ state: {
221
+ status: "error",
222
+ input: value.input,
223
+ error: (value.error as any).toString(),
224
+ metadata: value.error instanceof Permission.RejectedError ? value.error.metadata : undefined,
225
+ time: {
226
+ start: match.state.time.start,
227
+ end: Date.now(),
228
+ },
229
+ },
230
+ })
231
+
232
+ if (value.error instanceof Permission.RejectedError) {
233
+ blocked = true
234
+ }
235
+ delete toolcalls[value.toolCallId]
236
+ }
237
+ break
238
+ }
239
+ case "error":
240
+ throw value.error
241
+
242
+ case "start-step":
243
+ snapshot = await Snapshot.track()
244
+ await Session.updatePart({
245
+ id: Identifier.ascending("part"),
246
+ messageID: input.assistantMessage.id,
247
+ sessionID: input.sessionID,
248
+ snapshot,
249
+ type: "step-start",
250
+ })
251
+ break
252
+
253
+ case "finish-step":
254
+ const usage = Session.getUsage({
255
+ model: input.model,
256
+ usage: value.usage,
257
+ metadata: value.providerMetadata,
258
+ })
259
+ input.assistantMessage.finish = value.finishReason
260
+ input.assistantMessage.cost += usage.cost
261
+ input.assistantMessage.tokens = usage.tokens
262
+ await Session.updatePart({
263
+ id: Identifier.ascending("part"),
264
+ reason: value.finishReason,
265
+ snapshot: await Snapshot.track(),
266
+ messageID: input.assistantMessage.id,
267
+ sessionID: input.assistantMessage.sessionID,
268
+ type: "step-finish",
269
+ tokens: usage.tokens,
270
+ cost: usage.cost,
271
+ })
272
+ await Session.updateMessage(input.assistantMessage)
273
+
274
+ // Track telemetry for this step with session-level totals
275
+ // Compute cumulative session totals (like the sidebar displays)
276
+ const allMessages = await Session.messages({ sessionID: input.sessionID })
277
+ let sessionTotalCachedTokens = 0
278
+ let sessionTotalPromptTokens = 0
279
+ let sessionTotalOutputTokens = 0
280
+ for (const msg of allMessages) {
281
+ if (msg.info.role === "assistant") {
282
+ const tokens = msg.info.tokens
283
+ sessionTotalCachedTokens += tokens.cache.read
284
+ sessionTotalPromptTokens += tokens.input + tokens.cache.read
285
+ sessionTotalOutputTokens += tokens.output
286
+ }
287
+ }
288
+
289
+ Telemetry.trackFromUsage({
290
+ sessionID: input.sessionID,
291
+ providerID: input.model.providerID,
292
+ modelID: input.model.id,
293
+ tokens: {
294
+ input: usage.tokens.input,
295
+ output: usage.tokens.output,
296
+ reasoning: usage.tokens.reasoning,
297
+ cache: { read: usage.tokens.cache.read },
298
+ },
299
+ conversationTurns: input.conversationTurns ?? 0,
300
+ finishReason: value.finishReason,
301
+ sessionTotals: {
302
+ cachedTokens: sessionTotalCachedTokens,
303
+ promptTokens: sessionTotalPromptTokens,
304
+ outputTokens: sessionTotalOutputTokens,
305
+ },
306
+ })
307
+
308
+ if (snapshot) {
309
+ const patch = await Snapshot.patch(snapshot)
310
+ if (patch.files.length) {
311
+ await Session.updatePart({
312
+ id: Identifier.ascending("part"),
313
+ messageID: input.assistantMessage.id,
314
+ sessionID: input.sessionID,
315
+ type: "patch",
316
+ hash: patch.hash,
317
+ files: patch.files,
318
+ })
319
+ }
320
+ snapshot = undefined
321
+ }
322
+ SessionSummary.summarize({
323
+ sessionID: input.sessionID,
324
+ messageID: input.assistantMessage.parentID,
325
+ })
326
+ break
327
+
328
+ case "text-start":
329
+ currentText = {
330
+ id: Identifier.ascending("part"),
331
+ messageID: input.assistantMessage.id,
332
+ sessionID: input.assistantMessage.sessionID,
333
+ type: "text",
334
+ text: "",
335
+ time: {
336
+ start: Date.now(),
337
+ },
338
+ metadata: value.providerMetadata,
339
+ }
340
+ break
341
+
342
+ case "text-delta":
343
+ if (currentText) {
344
+ currentText.text += value.text
345
+ if (value.providerMetadata) currentText.metadata = value.providerMetadata
346
+ if (currentText.text)
347
+ await Session.updatePart({
348
+ part: currentText,
349
+ delta: value.text,
350
+ })
351
+ }
352
+ break
353
+
354
+ case "text-end":
355
+ if (currentText) {
356
+ currentText.text = currentText.text.trimEnd()
357
+ const textOutput = await Plugin.trigger(
358
+ "experimental.text.complete",
359
+ {
360
+ sessionID: input.sessionID,
361
+ messageID: input.assistantMessage.id,
362
+ partID: currentText.id,
363
+ },
364
+ { text: currentText.text },
365
+ )
366
+ currentText.text = textOutput.text
367
+ currentText.time = {
368
+ start: Date.now(),
369
+ end: Date.now(),
370
+ }
371
+ if (value.providerMetadata) currentText.metadata = value.providerMetadata
372
+ await Session.updatePart(currentText)
373
+ }
374
+ currentText = undefined
375
+ break
376
+
377
+ case "finish":
378
+ break
379
+
380
+ default:
381
+ log.info("unhandled", {
382
+ ...value,
383
+ })
384
+ continue
385
+ }
386
+ }
387
+ } catch (e: any) {
388
+ log.error("process", {
389
+ error: e,
390
+ stack: JSON.stringify(e.stack),
391
+ })
392
+ const error = MessageV2.fromError(e, { providerID: input.model.providerID })
393
+ const retry = SessionRetry.retryable(error)
394
+ if (retry !== undefined) {
395
+ attempt++
396
+ const delay = SessionRetry.delay(attempt, error.name === "APIError" ? error : undefined)
397
+ if (delay !== undefined) {
398
+ const seconds = Math.max(1, Math.ceil(delay / 1000))
399
+ const message = `Rate limit hit, retrying in ${seconds}s`
400
+ SessionStatus.set(input.sessionID, {
401
+ type: "retry",
402
+ attempt,
403
+ message,
404
+ next: Date.now() + delay,
405
+ })
406
+ await SessionRetry.sleep(delay, input.abort).catch(() => {})
407
+ continue
408
+ }
409
+ }
410
+ input.assistantMessage.error = error
411
+ Bus.publish(Session.Event.Error, {
412
+ sessionID: input.assistantMessage.sessionID,
413
+ error: input.assistantMessage.error,
414
+ })
415
+ }
416
+ const p = await MessageV2.parts(input.assistantMessage.id)
417
+ for (const part of p) {
418
+ if (part.type === "tool" && part.state.status !== "completed" && part.state.status !== "error") {
419
+ await Session.updatePart({
420
+ ...part,
421
+ state: {
422
+ ...part.state,
423
+ status: "error",
424
+ error: "Tool execution aborted",
425
+ time: {
426
+ start: Date.now(),
427
+ end: Date.now(),
428
+ },
429
+ },
430
+ })
431
+ }
432
+ }
433
+ input.assistantMessage.time.completed = Date.now()
434
+ await Session.updateMessage(input.assistantMessage)
435
+ if (blocked) return "stop"
436
+ if (input.assistantMessage.error) return "stop"
437
+ return "continue"
438
+ }
439
+ },
440
+ }
441
+ return result
442
+ }
443
+ }