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