chad-code 1.3.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 (338) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/README.npm.md +64 -0
  5. package/bin/chad-code +84 -0
  6. package/bunfig.toml +7 -0
  7. package/eslint.config.js +29 -0
  8. package/package.json +107 -0
  9. package/parsers-config.ts +253 -0
  10. package/script/build.ts +167 -0
  11. package/script/postinstall.mjs +122 -0
  12. package/script/publish-registries.ts +187 -0
  13. package/script/publish.ts +93 -0
  14. package/script/schema.ts +47 -0
  15. package/src/acp/README.md +164 -0
  16. package/src/acp/agent.ts +1086 -0
  17. package/src/acp/session.ts +101 -0
  18. package/src/acp/types.ts +22 -0
  19. package/src/agent/agent.ts +253 -0
  20. package/src/agent/generate.txt +75 -0
  21. package/src/agent/prompt/compaction.txt +12 -0
  22. package/src/agent/prompt/explore.txt +18 -0
  23. package/src/agent/prompt/summary.txt +11 -0
  24. package/src/agent/prompt/title.txt +36 -0
  25. package/src/auth/index.ts +70 -0
  26. package/src/bun/index.ts +130 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +69 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +132 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +28 -0
  36. package/src/cli/cmd/debug/config.ts +15 -0
  37. package/src/cli/cmd/debug/file.ts +91 -0
  38. package/src/cli/cmd/debug/index.ts +45 -0
  39. package/src/cli/cmd/debug/lsp.ts +48 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  41. package/src/cli/cmd/debug/scrap.ts +15 -0
  42. package/src/cli/cmd/debug/skill.ts +15 -0
  43. package/src/cli/cmd/debug/snapshot.ts +48 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +32 -0
  47. package/src/cli/cmd/import.ts +98 -0
  48. package/src/cli/cmd/mcp.ts +670 -0
  49. package/src/cli/cmd/models.ts +42 -0
  50. package/src/cli/cmd/pr.ts +112 -0
  51. package/src/cli/cmd/run.ts +374 -0
  52. package/src/cli/cmd/serve.ts +16 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +402 -0
  55. package/src/cli/cmd/tui/app.tsx +705 -0
  56. package/src/cli/cmd/tui/attach.ts +32 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +232 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +228 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +115 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +43 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +654 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1078 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  76. package/src/cli/cmd/tui/component/tips.ts +92 -0
  77. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  78. package/src/cli/cmd/tui/context/args.tsx +14 -0
  79. package/src/cli/cmd/tui/context/directory.ts +13 -0
  80. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  81. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  82. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  83. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  84. package/src/cli/cmd/tui/context/local.tsx +392 -0
  85. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  86. package/src/cli/cmd/tui/context/route.tsx +46 -0
  87. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  88. package/src/cli/cmd/tui/context/sync.tsx +384 -0
  89. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  90. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  91. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  94. package/src/cli/cmd/tui/context/theme/chad.json +245 -0
  95. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  96. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  97. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  98. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  99. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  100. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  101. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  102. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  103. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  104. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  105. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  107. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  108. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  111. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  112. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  113. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  114. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  115. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  116. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  117. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  118. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  119. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  120. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  121. package/src/cli/cmd/tui/context/theme.tsx +1137 -0
  122. package/src/cli/cmd/tui/event.ts +46 -0
  123. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  128. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  129. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  130. package/src/cli/cmd/tui/routes/session/index.tsx +1814 -0
  131. package/src/cli/cmd/tui/routes/session/permission.tsx +416 -0
  132. package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
  133. package/src/cli/cmd/tui/spawn.ts +48 -0
  134. package/src/cli/cmd/tui/thread.ts +111 -0
  135. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  136. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  137. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  138. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  139. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  140. package/src/cli/cmd/tui/ui/dialog-select.tsx +345 -0
  141. package/src/cli/cmd/tui/ui/dialog.tsx +171 -0
  142. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  143. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  144. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  145. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  146. package/src/cli/cmd/tui/util/editor.ts +32 -0
  147. package/src/cli/cmd/tui/util/signal.ts +7 -0
  148. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  149. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  150. package/src/cli/cmd/tui/worker.ts +68 -0
  151. package/src/cli/cmd/uninstall.ts +344 -0
  152. package/src/cli/cmd/upgrade.ts +67 -0
  153. package/src/cli/cmd/web.ts +73 -0
  154. package/src/cli/error.ts +56 -0
  155. package/src/cli/network.ts +53 -0
  156. package/src/cli/ui.ts +87 -0
  157. package/src/cli/upgrade.ts +25 -0
  158. package/src/command/index.ts +131 -0
  159. package/src/command/template/initialize.txt +10 -0
  160. package/src/command/template/review.txt +97 -0
  161. package/src/config/config.ts +1124 -0
  162. package/src/config/markdown.ts +41 -0
  163. package/src/env/index.ts +26 -0
  164. package/src/file/ignore.ts +83 -0
  165. package/src/file/index.ts +411 -0
  166. package/src/file/ripgrep.ts +402 -0
  167. package/src/file/time.ts +64 -0
  168. package/src/file/watcher.ts +117 -0
  169. package/src/flag/flag.ts +52 -0
  170. package/src/format/formatter.ts +359 -0
  171. package/src/format/index.ts +137 -0
  172. package/src/global/index.ts +55 -0
  173. package/src/id/id.ts +73 -0
  174. package/src/ide/index.ts +77 -0
  175. package/src/index.ts +159 -0
  176. package/src/installation/index.ts +198 -0
  177. package/src/lsp/client.ts +252 -0
  178. package/src/lsp/index.ts +485 -0
  179. package/src/lsp/language.ts +119 -0
  180. package/src/lsp/server.ts +2023 -0
  181. package/src/mcp/auth.ts +135 -0
  182. package/src/mcp/index.ts +874 -0
  183. package/src/mcp/oauth-callback.ts +200 -0
  184. package/src/mcp/oauth-provider.ts +154 -0
  185. package/src/patch/index.ts +622 -0
  186. package/src/permission/arity.ts +163 -0
  187. package/src/permission/index.ts +210 -0
  188. package/src/permission/next.ts +268 -0
  189. package/src/plugin/index.ts +106 -0
  190. package/src/project/bootstrap.ts +31 -0
  191. package/src/project/instance.ts +78 -0
  192. package/src/project/project.ts +263 -0
  193. package/src/project/state.ts +65 -0
  194. package/src/project/vcs.ts +76 -0
  195. package/src/provider/auth.ts +143 -0
  196. package/src/provider/models-macro.ts +4 -0
  197. package/src/provider/models.ts +77 -0
  198. package/src/provider/provider.ts +516 -0
  199. package/src/provider/transform.ts +114 -0
  200. package/src/pty/index.ts +212 -0
  201. package/src/server/error.ts +36 -0
  202. package/src/server/mdns.ts +57 -0
  203. package/src/server/project.ts +79 -0
  204. package/src/server/server.ts +2866 -0
  205. package/src/server/tui.ts +71 -0
  206. package/src/session/compaction.ts +225 -0
  207. package/src/session/index.ts +469 -0
  208. package/src/session/llm.ts +213 -0
  209. package/src/session/message-v2.ts +742 -0
  210. package/src/session/message.ts +189 -0
  211. package/src/session/processor.ts +402 -0
  212. package/src/session/prompt/anthropic-20250930.txt +166 -0
  213. package/src/session/prompt/anthropic.txt +105 -0
  214. package/src/session/prompt/anthropic_spoof.txt +1 -0
  215. package/src/session/prompt/beast.txt +147 -0
  216. package/src/session/prompt/build-switch.txt +5 -0
  217. package/src/session/prompt/codex.txt +318 -0
  218. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  219. package/src/session/prompt/gemini.txt +155 -0
  220. package/src/session/prompt/max-steps.txt +16 -0
  221. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  222. package/src/session/prompt/plan.txt +26 -0
  223. package/src/session/prompt/qwen.txt +109 -0
  224. package/src/session/prompt.ts +1621 -0
  225. package/src/session/retry.ts +90 -0
  226. package/src/session/revert.ts +108 -0
  227. package/src/session/status.ts +76 -0
  228. package/src/session/summary.ts +194 -0
  229. package/src/session/system.ts +108 -0
  230. package/src/session/todo.ts +37 -0
  231. package/src/share/share-next.ts +194 -0
  232. package/src/share/share.ts +23 -0
  233. package/src/shell/shell.ts +67 -0
  234. package/src/skill/index.ts +1 -0
  235. package/src/skill/skill.ts +124 -0
  236. package/src/snapshot/index.ts +197 -0
  237. package/src/storage/storage.ts +226 -0
  238. package/src/tool/bash.ts +262 -0
  239. package/src/tool/bash.txt +116 -0
  240. package/src/tool/batch.ts +175 -0
  241. package/src/tool/batch.txt +24 -0
  242. package/src/tool/codesearch.ts +132 -0
  243. package/src/tool/codesearch.txt +12 -0
  244. package/src/tool/edit.ts +655 -0
  245. package/src/tool/edit.txt +10 -0
  246. package/src/tool/glob.ts +75 -0
  247. package/src/tool/glob.txt +6 -0
  248. package/src/tool/grep.ts +132 -0
  249. package/src/tool/grep.txt +8 -0
  250. package/src/tool/invalid.ts +17 -0
  251. package/src/tool/ls.ts +119 -0
  252. package/src/tool/ls.txt +1 -0
  253. package/src/tool/lsp.ts +94 -0
  254. package/src/tool/lsp.txt +19 -0
  255. package/src/tool/multiedit.ts +46 -0
  256. package/src/tool/multiedit.txt +41 -0
  257. package/src/tool/patch.ts +210 -0
  258. package/src/tool/patch.txt +1 -0
  259. package/src/tool/read.ts +191 -0
  260. package/src/tool/read.txt +12 -0
  261. package/src/tool/registry.ts +137 -0
  262. package/src/tool/skill.ts +77 -0
  263. package/src/tool/task.ts +167 -0
  264. package/src/tool/task.txt +60 -0
  265. package/src/tool/todo.ts +53 -0
  266. package/src/tool/todoread.txt +14 -0
  267. package/src/tool/todowrite.txt +167 -0
  268. package/src/tool/tool.ts +73 -0
  269. package/src/tool/webfetch.ts +182 -0
  270. package/src/tool/webfetch.txt +13 -0
  271. package/src/tool/websearch.ts +144 -0
  272. package/src/tool/websearch.txt +11 -0
  273. package/src/tool/write.ts +84 -0
  274. package/src/tool/write.txt +8 -0
  275. package/src/util/archive.ts +16 -0
  276. package/src/util/color.ts +19 -0
  277. package/src/util/context.ts +25 -0
  278. package/src/util/defer.ts +12 -0
  279. package/src/util/eventloop.ts +20 -0
  280. package/src/util/filesystem.ts +83 -0
  281. package/src/util/fn.ts +11 -0
  282. package/src/util/iife.ts +3 -0
  283. package/src/util/keybind.ts +102 -0
  284. package/src/util/lazy.ts +18 -0
  285. package/src/util/locale.ts +81 -0
  286. package/src/util/lock.ts +98 -0
  287. package/src/util/log.ts +180 -0
  288. package/src/util/queue.ts +32 -0
  289. package/src/util/rpc.ts +42 -0
  290. package/src/util/scrap.ts +10 -0
  291. package/src/util/signal.ts +12 -0
  292. package/src/util/timeout.ts +14 -0
  293. package/src/util/token.ts +7 -0
  294. package/src/util/wildcard.ts +54 -0
  295. package/src/worktree/index.ts +217 -0
  296. package/sst-env.d.ts +9 -0
  297. package/test/agent/agent.test.ts +448 -0
  298. package/test/bun.test.ts +53 -0
  299. package/test/cli/github-action.test.ts +129 -0
  300. package/test/cli/github-remote.test.ts +80 -0
  301. package/test/cli/tui/transcript.test.ts +297 -0
  302. package/test/config/agent-color.test.ts +66 -0
  303. package/test/config/config.test.ts +870 -0
  304. package/test/config/markdown.test.ts +89 -0
  305. package/test/file/ignore.test.ts +10 -0
  306. package/test/file/path-traversal.test.ts +115 -0
  307. package/test/fixture/fixture.ts +45 -0
  308. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  309. package/test/ide/ide.test.ts +82 -0
  310. package/test/keybind.test.ts +421 -0
  311. package/test/lsp/client.test.ts +95 -0
  312. package/test/mcp/headers.test.ts +153 -0
  313. package/test/patch/patch.test.ts +348 -0
  314. package/test/permission/arity.test.ts +33 -0
  315. package/test/permission/next.test.ts +652 -0
  316. package/test/preload.ts +63 -0
  317. package/test/project/project.test.ts +120 -0
  318. package/test/provider/amazon-bedrock.test.ts +236 -0
  319. package/test/provider/provider.test.ts +2127 -0
  320. package/test/provider/transform.test.ts +980 -0
  321. package/test/server/session-select.test.ts +78 -0
  322. package/test/session/compaction.test.ts +251 -0
  323. package/test/session/message-v2.test.ts +570 -0
  324. package/test/session/retry.test.ts +131 -0
  325. package/test/session/revert-compact.test.ts +285 -0
  326. package/test/session/session.test.ts +71 -0
  327. package/test/skill/skill.test.ts +185 -0
  328. package/test/snapshot/snapshot.test.ts +939 -0
  329. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  330. package/test/tool/bash.test.ts +232 -0
  331. package/test/tool/grep.test.ts +109 -0
  332. package/test/tool/patch.test.ts +261 -0
  333. package/test/tool/read.test.ts +167 -0
  334. package/test/util/iife.test.ts +36 -0
  335. package/test/util/lazy.test.ts +50 -0
  336. package/test/util/timeout.test.ts +21 -0
  337. package/test/util/wildcard.test.ts +55 -0
  338. package/tsconfig.json +16 -0
@@ -0,0 +1,742 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import z from "zod"
3
+ import { NamedError } from "@opencode-ai/util/error"
4
+ import { APICallError, convertToModelMessages, LoadAPIKeyError, type ModelMessage, type UIMessage } from "ai"
5
+ import { Identifier } from "../id/id"
6
+ import { LSP } from "../lsp"
7
+ import { Snapshot } from "@/snapshot"
8
+ import { fn } from "@/util/fn"
9
+ import { Storage } from "@/storage/storage"
10
+ import { ProviderTransform } from "@/provider/transform"
11
+ import { STATUS_CODES } from "http"
12
+ import { iife } from "@/util/iife"
13
+ import { type SystemError } from "bun"
14
+
15
+ export namespace MessageV2 {
16
+ export const OutputLengthError = NamedError.create("MessageOutputLengthError", z.object({}))
17
+ export const AbortedError = NamedError.create("MessageAbortedError", z.object({ message: z.string() }))
18
+ export const AuthError = NamedError.create(
19
+ "ProviderAuthError",
20
+ z.object({
21
+ providerID: z.string(),
22
+ message: z.string(),
23
+ }),
24
+ )
25
+ export const APIError = NamedError.create(
26
+ "APIError",
27
+ z.object({
28
+ message: z.string(),
29
+ statusCode: z.number().optional(),
30
+ isRetryable: z.boolean(),
31
+ responseHeaders: z.record(z.string(), z.string()).optional(),
32
+ responseBody: z.string().optional(),
33
+ metadata: z.record(z.string(), z.string()).optional(),
34
+ }),
35
+ )
36
+ export type APIError = z.infer<typeof APIError.Schema>
37
+
38
+ const PartBase = z.object({
39
+ id: z.string(),
40
+ sessionID: z.string(),
41
+ messageID: z.string(),
42
+ })
43
+
44
+ export const SnapshotPart = PartBase.extend({
45
+ type: z.literal("snapshot"),
46
+ snapshot: z.string(),
47
+ }).meta({
48
+ ref: "SnapshotPart",
49
+ })
50
+ export type SnapshotPart = z.infer<typeof SnapshotPart>
51
+
52
+ export const PatchPart = PartBase.extend({
53
+ type: z.literal("patch"),
54
+ hash: z.string(),
55
+ files: z.string().array(),
56
+ }).meta({
57
+ ref: "PatchPart",
58
+ })
59
+ export type PatchPart = z.infer<typeof PatchPart>
60
+
61
+ export const TextPart = PartBase.extend({
62
+ type: z.literal("text"),
63
+ text: z.string(),
64
+ synthetic: z.boolean().optional(),
65
+ ignored: z.boolean().optional(),
66
+ time: z
67
+ .object({
68
+ start: z.number(),
69
+ end: z.number().optional(),
70
+ })
71
+ .optional(),
72
+ metadata: z.record(z.string(), z.any()).optional(),
73
+ }).meta({
74
+ ref: "TextPart",
75
+ })
76
+ export type TextPart = z.infer<typeof TextPart>
77
+
78
+ export const ReasoningPart = PartBase.extend({
79
+ type: z.literal("reasoning"),
80
+ text: z.string(),
81
+ metadata: z.record(z.string(), z.any()).optional(),
82
+ time: z.object({
83
+ start: z.number(),
84
+ end: z.number().optional(),
85
+ }),
86
+ }).meta({
87
+ ref: "ReasoningPart",
88
+ })
89
+ export type ReasoningPart = z.infer<typeof ReasoningPart>
90
+
91
+ const FilePartSourceBase = z.object({
92
+ text: z
93
+ .object({
94
+ value: z.string(),
95
+ start: z.number().int(),
96
+ end: z.number().int(),
97
+ })
98
+ .meta({
99
+ ref: "FilePartSourceText",
100
+ }),
101
+ })
102
+
103
+ export const FileSource = FilePartSourceBase.extend({
104
+ type: z.literal("file"),
105
+ path: z.string(),
106
+ }).meta({
107
+ ref: "FileSource",
108
+ })
109
+
110
+ export const SymbolSource = FilePartSourceBase.extend({
111
+ type: z.literal("symbol"),
112
+ path: z.string(),
113
+ range: LSP.Range,
114
+ name: z.string(),
115
+ kind: z.number().int(),
116
+ }).meta({
117
+ ref: "SymbolSource",
118
+ })
119
+
120
+ export const ResourceSource = FilePartSourceBase.extend({
121
+ type: z.literal("resource"),
122
+ clientName: z.string(),
123
+ uri: z.string(),
124
+ }).meta({
125
+ ref: "ResourceSource",
126
+ })
127
+
128
+ export const FilePartSource = z.discriminatedUnion("type", [FileSource, SymbolSource, ResourceSource]).meta({
129
+ ref: "FilePartSource",
130
+ })
131
+
132
+ export const FilePart = PartBase.extend({
133
+ type: z.literal("file"),
134
+ mime: z.string(),
135
+ filename: z.string().optional(),
136
+ url: z.string(),
137
+ source: FilePartSource.optional(),
138
+ }).meta({
139
+ ref: "FilePart",
140
+ })
141
+ export type FilePart = z.infer<typeof FilePart>
142
+
143
+ export const AgentPart = PartBase.extend({
144
+ type: z.literal("agent"),
145
+ name: z.string(),
146
+ source: z
147
+ .object({
148
+ value: z.string(),
149
+ start: z.number().int(),
150
+ end: z.number().int(),
151
+ })
152
+ .optional(),
153
+ }).meta({
154
+ ref: "AgentPart",
155
+ })
156
+ export type AgentPart = z.infer<typeof AgentPart>
157
+
158
+ export const CompactionPart = PartBase.extend({
159
+ type: z.literal("compaction"),
160
+ auto: z.boolean(),
161
+ }).meta({
162
+ ref: "CompactionPart",
163
+ })
164
+ export type CompactionPart = z.infer<typeof CompactionPart>
165
+
166
+ export const SubtaskPart = PartBase.extend({
167
+ type: z.literal("subtask"),
168
+ prompt: z.string(),
169
+ description: z.string(),
170
+ agent: z.string(),
171
+ command: z.string().optional(),
172
+ })
173
+ export type SubtaskPart = z.infer<typeof SubtaskPart>
174
+
175
+ export const RetryPart = PartBase.extend({
176
+ type: z.literal("retry"),
177
+ attempt: z.number(),
178
+ error: APIError.Schema,
179
+ time: z.object({
180
+ created: z.number(),
181
+ }),
182
+ }).meta({
183
+ ref: "RetryPart",
184
+ })
185
+ export type RetryPart = z.infer<typeof RetryPart>
186
+
187
+ export const StepStartPart = PartBase.extend({
188
+ type: z.literal("step-start"),
189
+ snapshot: z.string().optional(),
190
+ }).meta({
191
+ ref: "StepStartPart",
192
+ })
193
+ export type StepStartPart = z.infer<typeof StepStartPart>
194
+
195
+ export const StepFinishPart = PartBase.extend({
196
+ type: z.literal("step-finish"),
197
+ reason: z.string(),
198
+ snapshot: z.string().optional(),
199
+ cost: z.number(),
200
+ tokens: z.object({
201
+ input: z.number(),
202
+ output: z.number(),
203
+ reasoning: z.number(),
204
+ cache: z.object({
205
+ read: z.number(),
206
+ write: z.number(),
207
+ }),
208
+ }),
209
+ }).meta({
210
+ ref: "StepFinishPart",
211
+ })
212
+ export type StepFinishPart = z.infer<typeof StepFinishPart>
213
+
214
+ export const ToolStatePending = z
215
+ .object({
216
+ status: z.literal("pending"),
217
+ input: z.record(z.string(), z.any()),
218
+ raw: z.string(),
219
+ })
220
+ .meta({
221
+ ref: "ToolStatePending",
222
+ })
223
+
224
+ export type ToolStatePending = z.infer<typeof ToolStatePending>
225
+
226
+ export const ToolStateRunning = z
227
+ .object({
228
+ status: z.literal("running"),
229
+ input: z.record(z.string(), z.any()),
230
+ title: z.string().optional(),
231
+ metadata: z.record(z.string(), z.any()).optional(),
232
+ time: z.object({
233
+ start: z.number(),
234
+ }),
235
+ })
236
+ .meta({
237
+ ref: "ToolStateRunning",
238
+ })
239
+ export type ToolStateRunning = z.infer<typeof ToolStateRunning>
240
+
241
+ export const ToolStateCompleted = z
242
+ .object({
243
+ status: z.literal("completed"),
244
+ input: z.record(z.string(), z.any()),
245
+ output: z.string(),
246
+ title: z.string(),
247
+ metadata: z.record(z.string(), z.any()),
248
+ time: z.object({
249
+ start: z.number(),
250
+ end: z.number(),
251
+ compacted: z.number().optional(),
252
+ }),
253
+ attachments: FilePart.array().optional(),
254
+ })
255
+ .meta({
256
+ ref: "ToolStateCompleted",
257
+ })
258
+ export type ToolStateCompleted = z.infer<typeof ToolStateCompleted>
259
+
260
+ export const ToolStateError = z
261
+ .object({
262
+ status: z.literal("error"),
263
+ input: z.record(z.string(), z.any()),
264
+ error: z.string(),
265
+ metadata: z.record(z.string(), z.any()).optional(),
266
+ time: z.object({
267
+ start: z.number(),
268
+ end: z.number(),
269
+ }),
270
+ })
271
+ .meta({
272
+ ref: "ToolStateError",
273
+ })
274
+ export type ToolStateError = z.infer<typeof ToolStateError>
275
+
276
+ export const ToolState = z
277
+ .discriminatedUnion("status", [ToolStatePending, ToolStateRunning, ToolStateCompleted, ToolStateError])
278
+ .meta({
279
+ ref: "ToolState",
280
+ })
281
+
282
+ export const ToolPart = PartBase.extend({
283
+ type: z.literal("tool"),
284
+ callID: z.string(),
285
+ tool: z.string(),
286
+ state: ToolState,
287
+ metadata: z.record(z.string(), z.any()).optional(),
288
+ }).meta({
289
+ ref: "ToolPart",
290
+ })
291
+ export type ToolPart = z.infer<typeof ToolPart>
292
+
293
+ const Base = z.object({
294
+ id: z.string(),
295
+ sessionID: z.string(),
296
+ })
297
+
298
+ export const User = Base.extend({
299
+ role: z.literal("user"),
300
+ time: z.object({
301
+ created: z.number(),
302
+ }),
303
+ summary: z
304
+ .object({
305
+ title: z.string().optional(),
306
+ body: z.string().optional(),
307
+ diffs: Snapshot.FileDiff.array(),
308
+ })
309
+ .optional(),
310
+ agent: z.string(),
311
+ model: z.object({
312
+ providerID: z.string(),
313
+ modelID: z.string(),
314
+ }),
315
+ system: z.string().optional(),
316
+ tools: z.record(z.string(), z.boolean()).optional(),
317
+ variant: z.string().optional(),
318
+ }).meta({
319
+ ref: "UserMessage",
320
+ })
321
+ export type User = z.infer<typeof User>
322
+
323
+ export const Part = z
324
+ .discriminatedUnion("type", [
325
+ TextPart,
326
+ SubtaskPart,
327
+ ReasoningPart,
328
+ FilePart,
329
+ ToolPart,
330
+ StepStartPart,
331
+ StepFinishPart,
332
+ SnapshotPart,
333
+ PatchPart,
334
+ AgentPart,
335
+ RetryPart,
336
+ CompactionPart,
337
+ ])
338
+ .meta({
339
+ ref: "Part",
340
+ })
341
+ export type Part = z.infer<typeof Part>
342
+
343
+ export const Assistant = Base.extend({
344
+ role: z.literal("assistant"),
345
+ time: z.object({
346
+ created: z.number(),
347
+ completed: z.number().optional(),
348
+ }),
349
+ error: z
350
+ .discriminatedUnion("name", [
351
+ AuthError.Schema,
352
+ NamedError.Unknown.Schema,
353
+ OutputLengthError.Schema,
354
+ AbortedError.Schema,
355
+ APIError.Schema,
356
+ ])
357
+ .optional(),
358
+ parentID: z.string(),
359
+ modelID: z.string(),
360
+ providerID: z.string(),
361
+ /**
362
+ * @deprecated
363
+ */
364
+ mode: z.string(),
365
+ agent: z.string(),
366
+ path: z.object({
367
+ cwd: z.string(),
368
+ root: z.string(),
369
+ }),
370
+ summary: z.boolean().optional(),
371
+ cost: z.number(),
372
+ tokens: z.object({
373
+ input: z.number(),
374
+ output: z.number(),
375
+ reasoning: z.number(),
376
+ cache: z.object({
377
+ read: z.number(),
378
+ write: z.number(),
379
+ }),
380
+ }),
381
+ finish: z.string().optional(),
382
+ }).meta({
383
+ ref: "AssistantMessage",
384
+ })
385
+ export type Assistant = z.infer<typeof Assistant>
386
+
387
+ export const Info = z.discriminatedUnion("role", [User, Assistant]).meta({
388
+ ref: "Message",
389
+ })
390
+ export type Info = z.infer<typeof Info>
391
+
392
+ export const Event = {
393
+ Updated: BusEvent.define(
394
+ "message.updated",
395
+ z.object({
396
+ info: Info,
397
+ }),
398
+ ),
399
+ Removed: BusEvent.define(
400
+ "message.removed",
401
+ z.object({
402
+ sessionID: z.string(),
403
+ messageID: z.string(),
404
+ }),
405
+ ),
406
+ PartUpdated: BusEvent.define(
407
+ "message.part.updated",
408
+ z.object({
409
+ part: Part,
410
+ delta: z.string().optional(),
411
+ }),
412
+ ),
413
+ PartRemoved: BusEvent.define(
414
+ "message.part.removed",
415
+ z.object({
416
+ sessionID: z.string(),
417
+ messageID: z.string(),
418
+ partID: z.string(),
419
+ }),
420
+ ),
421
+ }
422
+
423
+ export const WithParts = z.object({
424
+ info: Info,
425
+ parts: z.array(Part),
426
+ })
427
+ export type WithParts = z.infer<typeof WithParts>
428
+
429
+ export function toModelMessage(input: WithParts[]): ModelMessage[] {
430
+ const result: UIMessage[] = []
431
+
432
+ for (const msg of input) {
433
+ if (msg.parts.length === 0) continue
434
+
435
+ if (msg.info.role === "user") {
436
+ const userMessage: UIMessage = {
437
+ id: msg.info.id,
438
+ role: "user",
439
+ parts: [],
440
+ }
441
+ result.push(userMessage)
442
+ for (const part of msg.parts) {
443
+ if (part.type === "text" && !part.ignored)
444
+ userMessage.parts.push({
445
+ type: "text",
446
+ text: part.text,
447
+ })
448
+ // text/plain and directory files are converted into text parts, ignore them
449
+ if (part.type === "file" && part.mime !== "text/plain" && part.mime !== "application/x-directory")
450
+ userMessage.parts.push({
451
+ type: "file",
452
+ url: part.url,
453
+ mediaType: part.mime,
454
+ filename: part.filename,
455
+ })
456
+
457
+ if (part.type === "compaction") {
458
+ userMessage.parts.push({
459
+ type: "text",
460
+ text: "What did we do so far?",
461
+ })
462
+ }
463
+ if (part.type === "subtask") {
464
+ userMessage.parts.push({
465
+ type: "text",
466
+ text: "The following tool was executed by the user",
467
+ })
468
+ }
469
+ }
470
+ }
471
+
472
+ if (msg.info.role === "assistant") {
473
+ if (
474
+ msg.info.error &&
475
+ !(
476
+ MessageV2.AbortedError.isInstance(msg.info.error) &&
477
+ msg.parts.some((part) => part.type !== "step-start" && part.type !== "reasoning")
478
+ )
479
+ ) {
480
+ continue
481
+ }
482
+ const assistantMessage: UIMessage = {
483
+ id: msg.info.id,
484
+ role: "assistant",
485
+ parts: [],
486
+ }
487
+ for (const part of msg.parts) {
488
+ if (part.type === "text")
489
+ assistantMessage.parts.push({
490
+ type: "text",
491
+ text: part.text,
492
+ providerMetadata: part.metadata,
493
+ })
494
+ if (part.type === "step-start")
495
+ assistantMessage.parts.push({
496
+ type: "step-start",
497
+ })
498
+ if (part.type === "tool") {
499
+ if (part.state.status === "completed") {
500
+ if (part.state.attachments?.length) {
501
+ result.push({
502
+ id: Identifier.ascending("message"),
503
+ role: "user",
504
+ parts: [
505
+ {
506
+ type: "text",
507
+ text: `Tool ${part.tool} returned an attachment:`,
508
+ },
509
+ ...part.state.attachments.map((attachment) => ({
510
+ type: "file" as const,
511
+ url: attachment.url,
512
+ mediaType: attachment.mime,
513
+ filename: attachment.filename,
514
+ })),
515
+ ],
516
+ })
517
+ }
518
+ // Ensure tool output is never empty (some APIs don't handle empty tool results)
519
+ let toolOutput = part.state.time.compacted ? "[Old tool result content cleared]" : part.state.output
520
+ if (!toolOutput || toolOutput === "") {
521
+ toolOutput = "Success"
522
+ }
523
+ assistantMessage.parts.push({
524
+ type: ("tool-" + part.tool) as `tool-${string}`,
525
+ state: "output-available",
526
+ toolCallId: part.callID,
527
+ input: part.state.input,
528
+ output: toolOutput,
529
+ callProviderMetadata: part.metadata,
530
+ })
531
+ }
532
+ if (part.state.status === "error")
533
+ assistantMessage.parts.push({
534
+ type: ("tool-" + part.tool) as `tool-${string}`,
535
+ state: "output-error",
536
+ toolCallId: part.callID,
537
+ input: part.state.input,
538
+ errorText: part.state.error,
539
+ callProviderMetadata: part.metadata,
540
+ })
541
+ }
542
+ if (part.type === "reasoning") {
543
+ assistantMessage.parts.push({
544
+ type: "reasoning",
545
+ text: part.text,
546
+ providerMetadata: part.metadata,
547
+ })
548
+ }
549
+ }
550
+ if (assistantMessage.parts.length > 0) {
551
+ result.push(assistantMessage)
552
+ }
553
+ }
554
+ }
555
+
556
+ // Filter out step-start parts as they can cause duplicate messages in some APIs
557
+ const filtered = result
558
+ .map((msg) => ({
559
+ ...msg,
560
+ parts: msg.parts.filter((part) => part.type !== "step-start"),
561
+ }))
562
+ .filter((msg) => msg.parts.length > 0)
563
+
564
+ // Merge consecutive assistant messages to avoid duplicates
565
+ const merged = mergeConsecutiveAssistantMessages(filtered)
566
+
567
+ return convertToModelMessages(merged)
568
+ }
569
+
570
+ // Merge consecutive assistant messages into one to prevent duplicates
571
+ function mergeConsecutiveAssistantMessages(messages: UIMessage[]): UIMessage[] {
572
+ const result: UIMessage[] = []
573
+
574
+ for (const msg of messages) {
575
+ const last = result[result.length - 1]
576
+
577
+ // If current and last are both assistant messages, merge them
578
+ if (last && last.role === "assistant" && msg.role === "assistant") {
579
+ // Merge parts, avoiding duplicate tool calls by toolCallId
580
+ const existingToolCallIds = new Set(
581
+ last.parts
582
+ .filter((p) => p.type.startsWith("tool-"))
583
+ .map((p) => (p as any).toolCallId),
584
+ )
585
+
586
+ for (const part of msg.parts) {
587
+ if (part.type.startsWith("tool-")) {
588
+ const toolCallId = (part as any).toolCallId
589
+ if (!existingToolCallIds.has(toolCallId)) {
590
+ last.parts.push(part)
591
+ existingToolCallIds.add(toolCallId)
592
+ }
593
+ } else if (part.type === "text") {
594
+ // Only add text if it's different from existing text parts
595
+ const existingTexts = last.parts
596
+ .filter((p) => p.type === "text")
597
+ .map((p) => (p as any).text)
598
+ if (!existingTexts.includes((part as any).text)) {
599
+ last.parts.push(part)
600
+ }
601
+ } else {
602
+ last.parts.push(part)
603
+ }
604
+ }
605
+ } else {
606
+ result.push({ ...msg, parts: [...msg.parts] })
607
+ }
608
+ }
609
+
610
+ return result
611
+ }
612
+
613
+ export const stream = fn(Identifier.schema("session"), async function* (sessionID) {
614
+ const list = await Array.fromAsync(await Storage.list(["message", sessionID]))
615
+ for (let i = list.length - 1; i >= 0; i--) {
616
+ yield await get({
617
+ sessionID,
618
+ messageID: list[i][2],
619
+ })
620
+ }
621
+ })
622
+
623
+ export const parts = fn(Identifier.schema("message"), async (messageID) => {
624
+ const result = [] as MessageV2.Part[]
625
+ for (const item of await Storage.list(["part", messageID])) {
626
+ const read = await Storage.read<MessageV2.Part>(item)
627
+ result.push(read)
628
+ }
629
+ result.sort((a, b) => (a.id > b.id ? 1 : -1))
630
+ return result
631
+ })
632
+
633
+ export const get = fn(
634
+ z.object({
635
+ sessionID: Identifier.schema("session"),
636
+ messageID: Identifier.schema("message"),
637
+ }),
638
+ async (input) => {
639
+ return {
640
+ info: await Storage.read<MessageV2.Info>(["message", input.sessionID, input.messageID]),
641
+ parts: await parts(input.messageID),
642
+ }
643
+ },
644
+ )
645
+
646
+ export async function filterCompacted(stream: AsyncIterable<MessageV2.WithParts>) {
647
+ const result = [] as MessageV2.WithParts[]
648
+ const completed = new Set<string>()
649
+ for await (const msg of stream) {
650
+ result.push(msg)
651
+ if (
652
+ msg.info.role === "user" &&
653
+ completed.has(msg.info.id) &&
654
+ msg.parts.some((part) => part.type === "compaction")
655
+ )
656
+ break
657
+ if (msg.info.role === "assistant" && msg.info.summary && msg.info.finish) completed.add(msg.info.parentID)
658
+ }
659
+ result.reverse()
660
+ return result
661
+ }
662
+
663
+ export function fromError(e: unknown, ctx: { providerID: string }) {
664
+ switch (true) {
665
+ case e instanceof DOMException && e.name === "AbortError":
666
+ return new MessageV2.AbortedError(
667
+ { message: e.message },
668
+ {
669
+ cause: e,
670
+ },
671
+ ).toObject()
672
+ case MessageV2.OutputLengthError.isInstance(e):
673
+ return e
674
+ case LoadAPIKeyError.isInstance(e):
675
+ return new MessageV2.AuthError(
676
+ {
677
+ providerID: ctx.providerID,
678
+ message: e.message,
679
+ },
680
+ { cause: e },
681
+ ).toObject()
682
+ case (e as SystemError)?.code === "ECONNRESET":
683
+ return new MessageV2.APIError(
684
+ {
685
+ message: "Connection reset by server",
686
+ isRetryable: true,
687
+ metadata: {
688
+ code: (e as SystemError).code ?? "",
689
+ syscall: (e as SystemError).syscall ?? "",
690
+ message: (e as SystemError).message ?? "",
691
+ },
692
+ },
693
+ { cause: e },
694
+ ).toObject()
695
+ case APICallError.isInstance(e):
696
+ const message = iife(() => {
697
+ let msg = e.message
698
+ if (msg === "") {
699
+ if (e.responseBody) return e.responseBody
700
+ if (e.statusCode) {
701
+ const err = STATUS_CODES[e.statusCode]
702
+ if (err) return err
703
+ }
704
+ return "Unknown error"
705
+ }
706
+ const transformed = ProviderTransform.error(ctx.providerID, e)
707
+ if (transformed !== msg) {
708
+ return transformed
709
+ }
710
+ if (!e.responseBody || (e.statusCode && msg !== STATUS_CODES[e.statusCode])) {
711
+ return msg
712
+ }
713
+
714
+ try {
715
+ const body = JSON.parse(e.responseBody)
716
+ // try to extract common error message fields
717
+ const errMsg = body.message || body.error || body.error?.message
718
+ if (errMsg && typeof errMsg === "string") {
719
+ return `${msg}: ${errMsg}`
720
+ }
721
+ } catch {}
722
+
723
+ return `${msg}: ${e.responseBody}`
724
+ }).trim()
725
+
726
+ return new MessageV2.APIError(
727
+ {
728
+ message,
729
+ statusCode: e.statusCode,
730
+ isRetryable: e.isRetryable,
731
+ responseHeaders: e.responseHeaders,
732
+ responseBody: e.responseBody,
733
+ },
734
+ { cause: e },
735
+ ).toObject()
736
+ case e instanceof Error:
737
+ return new NamedError.Unknown({ message: e.toString() }, { cause: e }).toObject()
738
+ default:
739
+ return new NamedError.Unknown({ message: JSON.stringify(e) }, { cause: e })
740
+ }
741
+ }
742
+ }