@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,189 @@
1
+ import z from "zod"
2
+ import { NamedError } from "@/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,450 @@
1
+ import { MessageV2 } from "./message-v2"
2
+ import { Log } from "@/util/log"
3
+ import { Identifier } from "@/id/id"
4
+ import { Session } from "."
5
+ import { Agent } from "@/agent/agent"
6
+ import { Snapshot } from "@/snapshot"
7
+ import { SessionSummary } from "./summary"
8
+ import { Bus } from "@/bus"
9
+ import { SessionRetry } from "./retry"
10
+ import { SessionStatus } from "./status"
11
+ import { SessionFallback } from "./fallback"
12
+ import { Provider } from "@/provider/provider"
13
+ import { Plugin } from "@/plugin"
14
+ import { LLM } from "./llm"
15
+ import { Config } from "@/config/config"
16
+ import { SessionCompaction } from "./compaction"
17
+ import { PermissionNext } from "@/permission/next"
18
+ import { Question } from "@/question"
19
+
20
+ export namespace SessionProcessor {
21
+ const DOOM_LOOP_THRESHOLD = 3
22
+ const log = Log.create({ service: "session.processor" })
23
+
24
+ export type Info = Awaited<ReturnType<typeof create>>
25
+ export type Result = Awaited<ReturnType<Info["process"]>>
26
+
27
+ export function create(input: {
28
+ assistantMessage: MessageV2.Assistant
29
+ sessionID: string
30
+ model: Provider.Model
31
+ abort: AbortSignal
32
+ }) {
33
+ const toolcalls: Record<string, MessageV2.ToolPart> = {}
34
+ let snapshot: string | undefined
35
+ let blocked = false
36
+ let attempt = 0
37
+ let needsCompaction = false
38
+
39
+ const result = {
40
+ get message() {
41
+ return input.assistantMessage
42
+ },
43
+ partFromToolCall(toolCallID: string) {
44
+ return toolcalls[toolCallID]
45
+ },
46
+ async process(streamInput: LLM.StreamInput) {
47
+ log.info("process")
48
+ needsCompaction = false
49
+ const shouldBreak = (await Config.get()).experimental?.continue_loop_on_deny !== true
50
+ while (true) {
51
+ try {
52
+ let currentText: MessageV2.TextPart | undefined
53
+ let reasoningMap: Record<string, MessageV2.ReasoningPart> = {}
54
+ const stream = await LLM.stream(streamInput)
55
+
56
+ for await (const value of stream.fullStream) {
57
+ input.abort.throwIfAborted()
58
+ switch (value.type) {
59
+ case "start":
60
+ SessionStatus.set(input.sessionID, { type: "busy" })
61
+ break
62
+
63
+ case "reasoning-start":
64
+ if (value.id in reasoningMap) {
65
+ continue
66
+ }
67
+ reasoningMap[value.id] = {
68
+ id: Identifier.ascending("part"),
69
+ messageID: input.assistantMessage.id,
70
+ sessionID: input.assistantMessage.sessionID,
71
+ type: "reasoning",
72
+ text: "",
73
+ time: {
74
+ start: Date.now(),
75
+ },
76
+ metadata: value.providerMetadata,
77
+ }
78
+ break
79
+
80
+ case "reasoning-delta":
81
+ if (value.id in reasoningMap) {
82
+ const part = reasoningMap[value.id]
83
+ part.text += value.text
84
+ if (value.providerMetadata) part.metadata = value.providerMetadata
85
+ if (part.text) await Session.updatePart({ part, delta: value.text })
86
+ }
87
+ break
88
+
89
+ case "reasoning-end":
90
+ if (value.id in reasoningMap) {
91
+ const part = reasoningMap[value.id]
92
+ part.text = part.text.trimEnd()
93
+
94
+ part.time = {
95
+ ...part.time,
96
+ end: Date.now(),
97
+ }
98
+ if (value.providerMetadata) part.metadata = value.providerMetadata
99
+ await Session.updatePart(part)
100
+ delete reasoningMap[value.id]
101
+ }
102
+ break
103
+
104
+ case "tool-input-start":
105
+ const part = await Session.updatePart({
106
+ id: toolcalls[value.id]?.id ?? Identifier.ascending("part"),
107
+ messageID: input.assistantMessage.id,
108
+ sessionID: input.assistantMessage.sessionID,
109
+ type: "tool",
110
+ tool: value.toolName,
111
+ callID: value.id,
112
+ state: {
113
+ status: "pending",
114
+ input: {},
115
+ raw: "",
116
+ },
117
+ })
118
+ toolcalls[value.id] = part as MessageV2.ToolPart
119
+ break
120
+
121
+ case "tool-input-delta":
122
+ break
123
+
124
+ case "tool-input-end":
125
+ break
126
+
127
+ case "tool-call": {
128
+ const match = toolcalls[value.toolCallId]
129
+ if (match) {
130
+ const part = await Session.updatePart({
131
+ ...match,
132
+ tool: value.toolName,
133
+ state: {
134
+ status: "running",
135
+ input: value.input,
136
+ time: {
137
+ start: Date.now(),
138
+ },
139
+ },
140
+ metadata: value.providerMetadata,
141
+ })
142
+ toolcalls[value.toolCallId] = part as MessageV2.ToolPart
143
+
144
+ const parts = await MessageV2.parts(input.assistantMessage.id)
145
+ const lastThree = parts.slice(-DOOM_LOOP_THRESHOLD)
146
+
147
+ if (
148
+ lastThree.length === DOOM_LOOP_THRESHOLD &&
149
+ lastThree.every(
150
+ (p) =>
151
+ p.type === "tool" &&
152
+ p.tool === value.toolName &&
153
+ p.state.status !== "pending" &&
154
+ JSON.stringify(p.state.input) === JSON.stringify(value.input),
155
+ )
156
+ ) {
157
+ const agent = await Agent.get(input.assistantMessage.agent)
158
+ await PermissionNext.ask({
159
+ permission: "doom_loop",
160
+ patterns: [value.toolName],
161
+ sessionID: input.assistantMessage.sessionID,
162
+ metadata: {
163
+ tool: value.toolName,
164
+ input: value.input,
165
+ },
166
+ always: [value.toolName],
167
+ ruleset: agent.permission,
168
+ })
169
+ }
170
+ }
171
+ break
172
+ }
173
+ case "tool-result": {
174
+ const match = toolcalls[value.toolCallId]
175
+ if (match && match.state.status === "running") {
176
+ await Session.updatePart({
177
+ ...match,
178
+ state: {
179
+ status: "completed",
180
+ input: value.input ?? match.state.input,
181
+ output: value.output.output,
182
+ metadata: value.output.metadata,
183
+ title: value.output.title,
184
+ time: {
185
+ start: match.state.time.start,
186
+ end: Date.now(),
187
+ },
188
+ attachments: value.output.attachments,
189
+ },
190
+ })
191
+
192
+ delete toolcalls[value.toolCallId]
193
+ }
194
+ break
195
+ }
196
+
197
+ case "tool-error": {
198
+ const match = toolcalls[value.toolCallId]
199
+ if (match && match.state.status === "running") {
200
+ await Session.updatePart({
201
+ ...match,
202
+ state: {
203
+ status: "error",
204
+ input: value.input ?? match.state.input,
205
+ error: (value.error as any).toString(),
206
+ time: {
207
+ start: match.state.time.start,
208
+ end: Date.now(),
209
+ },
210
+ },
211
+ })
212
+
213
+ if (
214
+ value.error instanceof PermissionNext.RejectedError ||
215
+ value.error instanceof Question.RejectedError
216
+ ) {
217
+ blocked = shouldBreak
218
+ }
219
+ delete toolcalls[value.toolCallId]
220
+ }
221
+ break
222
+ }
223
+ case "error":
224
+ throw value.error
225
+
226
+ case "start-step":
227
+ snapshot = await Snapshot.track()
228
+ await Session.updatePart({
229
+ id: Identifier.ascending("part"),
230
+ messageID: input.assistantMessage.id,
231
+ sessionID: input.sessionID,
232
+ snapshot,
233
+ type: "step-start",
234
+ })
235
+ break
236
+
237
+ case "finish-step":
238
+ const usage = Session.getUsage({
239
+ model: input.model,
240
+ usage: value.usage,
241
+ metadata: value.providerMetadata,
242
+ })
243
+ input.assistantMessage.finish = value.finishReason
244
+ input.assistantMessage.cost += usage.cost
245
+ input.assistantMessage.tokens = usage.tokens
246
+ await Session.updatePart({
247
+ id: Identifier.ascending("part"),
248
+ reason: value.finishReason,
249
+ snapshot: await Snapshot.track(),
250
+ messageID: input.assistantMessage.id,
251
+ sessionID: input.assistantMessage.sessionID,
252
+ type: "step-finish",
253
+ tokens: usage.tokens,
254
+ cost: usage.cost,
255
+ })
256
+ await Session.updateMessage(input.assistantMessage)
257
+ if (snapshot) {
258
+ const patch = await Snapshot.patch(snapshot)
259
+ if (patch.files.length) {
260
+ await Session.updatePart({
261
+ id: Identifier.ascending("part"),
262
+ messageID: input.assistantMessage.id,
263
+ sessionID: input.sessionID,
264
+ type: "patch",
265
+ hash: patch.hash,
266
+ files: patch.files,
267
+ })
268
+ }
269
+ snapshot = undefined
270
+ }
271
+ SessionSummary.summarize({
272
+ sessionID: input.sessionID,
273
+ messageID: input.assistantMessage.parentID,
274
+ })
275
+ if (await SessionCompaction.isOverflow({ tokens: usage.tokens, model: input.model })) {
276
+ needsCompaction = true
277
+ }
278
+ break
279
+
280
+ case "text-start":
281
+ currentText = {
282
+ id: Identifier.ascending("part"),
283
+ messageID: input.assistantMessage.id,
284
+ sessionID: input.assistantMessage.sessionID,
285
+ type: "text",
286
+ text: "",
287
+ time: {
288
+ start: Date.now(),
289
+ },
290
+ metadata: value.providerMetadata,
291
+ }
292
+ break
293
+
294
+ case "text-delta":
295
+ if (currentText) {
296
+ currentText.text += value.text
297
+ if (value.providerMetadata) currentText.metadata = value.providerMetadata
298
+ if (currentText.text)
299
+ await Session.updatePart({
300
+ part: currentText,
301
+ delta: value.text,
302
+ })
303
+ }
304
+ break
305
+
306
+ case "text-end":
307
+ if (currentText) {
308
+ currentText.text = currentText.text.trimEnd()
309
+ const textOutput = await Plugin.trigger(
310
+ "experimental.text.complete",
311
+ {
312
+ sessionID: input.sessionID,
313
+ messageID: input.assistantMessage.id,
314
+ partID: currentText.id,
315
+ },
316
+ { text: currentText.text },
317
+ )
318
+ currentText.text = textOutput.text
319
+ currentText.time = {
320
+ start: Date.now(),
321
+ end: Date.now(),
322
+ }
323
+ if (value.providerMetadata) currentText.metadata = value.providerMetadata
324
+ await Session.updatePart(currentText)
325
+ }
326
+ currentText = undefined
327
+ break
328
+
329
+ case "finish":
330
+ break
331
+
332
+ default:
333
+ log.info("unhandled", {
334
+ ...value,
335
+ })
336
+ continue
337
+ }
338
+ if (needsCompaction) break
339
+ }
340
+ } catch (e: any) {
341
+ log.error("process", {
342
+ error: e,
343
+ stack: JSON.stringify(e.stack),
344
+ })
345
+ const error = MessageV2.fromError(e, { providerID: input.model.providerID })
346
+ const retry = SessionRetry.retryable(error)
347
+ if (retry !== undefined) {
348
+ // Mark current provider as rate-limited
349
+ const headers = error.name === "APIError" ? error.data.responseHeaders : undefined
350
+ SessionFallback.markRateLimited(input.model.providerID, retry, headers)
351
+
352
+ // Try to find an alternative provider
353
+ const alternative = await SessionFallback.getNextAvailable(input.model.providerID, {
354
+ toolcall: input.model.capabilities.toolcall,
355
+ attachment: input.model.capabilities.attachment,
356
+ })
357
+
358
+ if (alternative) {
359
+ // Switch to alternative provider
360
+ SessionStatus.set(input.sessionID, {
361
+ type: "switching",
362
+ from: input.model.providerID,
363
+ to: alternative.providerID,
364
+ reason: retry,
365
+ })
366
+
367
+ // Get the full model info for the alternative
368
+ const newModel = await Provider.getModel(alternative.providerID, alternative.modelID)
369
+ input.model = newModel
370
+ log.info("Switched to alternative provider", {
371
+ from: input.model.providerID,
372
+ to: alternative.providerID,
373
+ model: alternative.modelID,
374
+ })
375
+ continue
376
+ }
377
+
378
+ // All providers exhausted - sleep until earliest reset
379
+ const resetTime = SessionFallback.getEarliestResetTime()
380
+ const delay = resetTime ? Math.max(resetTime - Date.now(), 1000) : SessionRetry.delay(attempt, error.name === "APIError" ? error : undefined)
381
+
382
+ attempt++
383
+ SessionStatus.set(input.sessionID, {
384
+ type: "sleeping",
385
+ reason: "All providers rate-limited",
386
+ resetAt: Date.now() + delay,
387
+ providers: SessionFallback.getRateLimitedProviders().map((p) => ({
388
+ providerID: p.providerID,
389
+ resetAt: p.resetAt,
390
+ })),
391
+ })
392
+
393
+ log.info("All providers rate-limited, sleeping", {
394
+ delay,
395
+ resetAt: new Date(Date.now() + delay).toISOString(),
396
+ })
397
+
398
+ await SessionRetry.sleep(delay, input.abort).catch(() => {})
399
+ SessionFallback.clearExpiredLimits()
400
+ continue
401
+ }
402
+ input.assistantMessage.error = error
403
+ Bus.publish(Session.Event.Error, {
404
+ sessionID: input.assistantMessage.sessionID,
405
+ error: input.assistantMessage.error,
406
+ })
407
+ }
408
+ if (snapshot) {
409
+ const patch = await Snapshot.patch(snapshot)
410
+ if (patch.files.length) {
411
+ await Session.updatePart({
412
+ id: Identifier.ascending("part"),
413
+ messageID: input.assistantMessage.id,
414
+ sessionID: input.sessionID,
415
+ type: "patch",
416
+ hash: patch.hash,
417
+ files: patch.files,
418
+ })
419
+ }
420
+ snapshot = undefined
421
+ }
422
+ const p = await MessageV2.parts(input.assistantMessage.id)
423
+ for (const part of p) {
424
+ if (part.type === "tool" && part.state.status !== "completed" && part.state.status !== "error") {
425
+ await Session.updatePart({
426
+ ...part,
427
+ state: {
428
+ ...part.state,
429
+ status: "error",
430
+ error: "Tool execution aborted",
431
+ time: {
432
+ start: Date.now(),
433
+ end: Date.now(),
434
+ },
435
+ },
436
+ })
437
+ }
438
+ }
439
+ input.assistantMessage.time.completed = Date.now()
440
+ await Session.updateMessage(input.assistantMessage)
441
+ if (needsCompaction) return "compact"
442
+ if (blocked) return "stop"
443
+ if (input.assistantMessage.error) return "stop"
444
+ return "continue"
445
+ }
446
+ },
447
+ }
448
+ return result
449
+ }
450
+ }