@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,99 @@
1
+ import type { NamedError } from "@/util/error"
2
+ import { MessageV2 } from "./message-v2"
3
+ import { iife } from "@/util/iife"
4
+
5
+ export namespace SessionRetry {
6
+ export const RETRY_INITIAL_DELAY = 2000
7
+ export const RETRY_BACKOFF_FACTOR = 2
8
+ export const RETRY_MAX_DELAY_NO_HEADERS = 30_000 // 30 seconds
9
+ export const RETRY_MAX_DELAY = 2_147_483_647 // max 32-bit signed integer for setTimeout
10
+
11
+ export async function sleep(ms: number, signal: AbortSignal): Promise<void> {
12
+ return new Promise((resolve, reject) => {
13
+ const abortHandler = () => {
14
+ clearTimeout(timeout)
15
+ reject(new DOMException("Aborted", "AbortError"))
16
+ }
17
+ const timeout = setTimeout(
18
+ () => {
19
+ signal.removeEventListener("abort", abortHandler)
20
+ resolve()
21
+ },
22
+ Math.min(ms, RETRY_MAX_DELAY),
23
+ )
24
+ signal.addEventListener("abort", abortHandler, { once: true })
25
+ })
26
+ }
27
+
28
+ export function delay(attempt: number, error?: MessageV2.APIError) {
29
+ if (error) {
30
+ const headers = error.data.responseHeaders
31
+ if (headers) {
32
+ const retryAfterMs = headers["retry-after-ms"]
33
+ if (retryAfterMs) {
34
+ const parsedMs = Number.parseFloat(retryAfterMs)
35
+ if (!Number.isNaN(parsedMs)) {
36
+ return parsedMs
37
+ }
38
+ }
39
+
40
+ const retryAfter = headers["retry-after"]
41
+ if (retryAfter) {
42
+ const parsedSeconds = Number.parseFloat(retryAfter)
43
+ if (!Number.isNaN(parsedSeconds)) {
44
+ // convert seconds to milliseconds
45
+ return Math.ceil(parsedSeconds * 1000)
46
+ }
47
+ // Try parsing as HTTP date format
48
+ const parsed = Date.parse(retryAfter) - Date.now()
49
+ if (!Number.isNaN(parsed) && parsed > 0) {
50
+ return Math.ceil(parsed)
51
+ }
52
+ }
53
+
54
+ return RETRY_INITIAL_DELAY * Math.pow(RETRY_BACKOFF_FACTOR, attempt - 1)
55
+ }
56
+ }
57
+
58
+ return Math.min(RETRY_INITIAL_DELAY * Math.pow(RETRY_BACKOFF_FACTOR, attempt - 1), RETRY_MAX_DELAY_NO_HEADERS)
59
+ }
60
+
61
+ export function retryable(error: ReturnType<NamedError["toObject"]>) {
62
+ if (MessageV2.APIError.isInstance(error)) {
63
+ if (!error.data.isRetryable) return undefined
64
+ return error.data.message.includes("Overloaded") ? "Provider is overloaded" : error.data.message
65
+ }
66
+
67
+ const json = iife(() => {
68
+ try {
69
+ if (typeof error.data?.message === "string") {
70
+ const parsed = JSON.parse(error.data.message)
71
+ return parsed
72
+ }
73
+
74
+ return JSON.parse(error.data.message)
75
+ } catch {
76
+ return undefined
77
+ }
78
+ })
79
+ if (!json || typeof json !== "object") return undefined
80
+ const code = typeof json.code === "string" ? json.code : ""
81
+
82
+ if (json.type === "error" && json.error?.type === "too_many_requests") {
83
+ return "Too Many Requests"
84
+ }
85
+ if (code.includes("exhausted") || code.includes("unavailable")) {
86
+ return "Provider is overloaded"
87
+ }
88
+ if (json.type === "error" && json.error?.code?.includes("rate_limit")) {
89
+ return "Rate Limited"
90
+ }
91
+ if (
92
+ json.error?.message?.includes("no_kv_space") ||
93
+ (json.type === "error" && json.error?.type === "server_error") ||
94
+ !!json.error
95
+ ) {
96
+ return "Provider Server Error"
97
+ }
98
+ }
99
+ }
@@ -0,0 +1,121 @@
1
+ import z from "zod"
2
+ import { Identifier } from "../id/id"
3
+ import { Snapshot } from "../snapshot"
4
+ import { MessageV2 } from "./message-v2"
5
+ import { Session } from "."
6
+ import { Log } from "../util/log"
7
+ import { splitWhen } from "remeda"
8
+ import { Storage } from "../storage/storage"
9
+ import { Bus } from "../bus"
10
+ import { SessionPrompt } from "./prompt"
11
+ import { SessionSummary } from "./summary"
12
+
13
+ export namespace SessionRevert {
14
+ const log = Log.create({ service: "session.revert" })
15
+
16
+ export const RevertInput = z.object({
17
+ sessionID: Identifier.schema("session"),
18
+ messageID: Identifier.schema("message"),
19
+ partID: Identifier.schema("part").optional(),
20
+ })
21
+ export type RevertInput = z.infer<typeof RevertInput>
22
+
23
+ export async function revert(input: RevertInput) {
24
+ SessionPrompt.assertNotBusy(input.sessionID)
25
+ const all = await Session.messages({ sessionID: input.sessionID })
26
+ let lastUser: MessageV2.User | undefined
27
+ const session = await Session.get(input.sessionID)
28
+
29
+ let revert: Session.Info["revert"]
30
+ const patches: Snapshot.Patch[] = []
31
+ for (const msg of all) {
32
+ if (msg.info.role === "user") lastUser = msg.info
33
+ const remaining = []
34
+ for (const part of msg.parts) {
35
+ if (revert) {
36
+ if (part.type === "patch") {
37
+ patches.push(part)
38
+ }
39
+ continue
40
+ }
41
+
42
+ if (!revert) {
43
+ if ((msg.info.id === input.messageID && !input.partID) || part.id === input.partID) {
44
+ // if no useful parts left in message, same as reverting whole message
45
+ const partID = remaining.some((item) => ["text", "tool"].includes(item.type)) ? input.partID : undefined
46
+ revert = {
47
+ messageID: !partID && lastUser ? lastUser.id : msg.info.id,
48
+ partID,
49
+ }
50
+ }
51
+ remaining.push(part)
52
+ }
53
+ }
54
+ }
55
+
56
+ if (revert) {
57
+ const session = await Session.get(input.sessionID)
58
+ revert.snapshot = session.revert?.snapshot ?? (await Snapshot.track())
59
+ await Snapshot.revert(patches)
60
+ if (revert.snapshot) revert.diff = await Snapshot.diff(revert.snapshot)
61
+ const rangeMessages = all.filter((msg) => msg.info.id >= revert!.messageID)
62
+ const diffs = await SessionSummary.computeDiff({ messages: rangeMessages })
63
+ await Storage.write(["session_diff", input.sessionID], diffs)
64
+ Bus.publish(Session.Event.Diff, {
65
+ sessionID: input.sessionID,
66
+ diff: diffs,
67
+ })
68
+ return Session.update(input.sessionID, (draft) => {
69
+ draft.revert = revert
70
+ draft.summary = {
71
+ additions: diffs.reduce((sum, x) => sum + x.additions, 0),
72
+ deletions: diffs.reduce((sum, x) => sum + x.deletions, 0),
73
+ files: diffs.length,
74
+ }
75
+ })
76
+ }
77
+ return session
78
+ }
79
+
80
+ export async function unrevert(input: { sessionID: string }) {
81
+ log.info("unreverting", input)
82
+ SessionPrompt.assertNotBusy(input.sessionID)
83
+ const session = await Session.get(input.sessionID)
84
+ if (!session.revert) return session
85
+ if (session.revert.snapshot) await Snapshot.restore(session.revert.snapshot)
86
+ const next = await Session.update(input.sessionID, (draft) => {
87
+ draft.revert = undefined
88
+ })
89
+ return next
90
+ }
91
+
92
+ export async function cleanup(session: Session.Info) {
93
+ if (!session.revert) return
94
+ const sessionID = session.id
95
+ let msgs = await Session.messages({ sessionID })
96
+ const messageID = session.revert.messageID
97
+ const [preserve, remove] = splitWhen(msgs, (x) => x.info.id === messageID)
98
+ msgs = preserve
99
+ for (const msg of remove) {
100
+ await Storage.remove(["message", sessionID, msg.info.id])
101
+ await Bus.publish(MessageV2.Event.Removed, { sessionID: sessionID, messageID: msg.info.id })
102
+ }
103
+ const last = preserve.at(-1)
104
+ if (session.revert.partID && last) {
105
+ const partID = session.revert.partID
106
+ const [preserveParts, removeParts] = splitWhen(last.parts, (x) => x.id === partID)
107
+ last.parts = preserveParts
108
+ for (const part of removeParts) {
109
+ await Storage.remove(["part", last.info.id, part.id])
110
+ await Bus.publish(MessageV2.Event.PartRemoved, {
111
+ sessionID: sessionID,
112
+ messageID: last.info.id,
113
+ partID: part.id,
114
+ })
115
+ }
116
+ }
117
+ await Session.update(sessionID, (draft) => {
118
+ draft.revert = undefined
119
+ })
120
+ }
121
+ }
@@ -0,0 +1,100 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import { Instance } from "@/project/instance"
4
+ import z from "zod"
5
+
6
+ export namespace SessionStatus {
7
+ export const Info = z
8
+ .union([
9
+ z.object({
10
+ type: z.literal("idle"),
11
+ }),
12
+ z.object({
13
+ type: z.literal("retry"),
14
+ attempt: z.number(),
15
+ message: z.string(),
16
+ next: z.number(),
17
+ }),
18
+ z.object({
19
+ type: z.literal("busy"),
20
+ }),
21
+ z.object({
22
+ type: z.literal("bloxy"),
23
+ taskIndex: z.number(),
24
+ totalTasks: z.number(),
25
+ currentTask: z.string(),
26
+ status: z.enum(["executing", "validating", "advancing", "paused"]),
27
+ }),
28
+ z.object({
29
+ type: z.literal("switching"),
30
+ from: z.string(),
31
+ to: z.string(),
32
+ reason: z.string(),
33
+ }),
34
+ z.object({
35
+ type: z.literal("sleeping"),
36
+ reason: z.string(),
37
+ resetAt: z.number(),
38
+ providers: z.array(
39
+ z.object({
40
+ providerID: z.string(),
41
+ resetAt: z.number(),
42
+ }),
43
+ ),
44
+ }),
45
+ ])
46
+ .meta({
47
+ ref: "SessionStatus",
48
+ })
49
+ export type Info = z.infer<typeof Info>
50
+
51
+ export const Event = {
52
+ Status: BusEvent.define(
53
+ "session.status",
54
+ z.object({
55
+ sessionID: z.string(),
56
+ status: Info,
57
+ }),
58
+ ),
59
+ // deprecated
60
+ Idle: BusEvent.define(
61
+ "session.idle",
62
+ z.object({
63
+ sessionID: z.string(),
64
+ }),
65
+ ),
66
+ }
67
+
68
+ const state = Instance.state(() => {
69
+ const data: Record<string, Info> = {}
70
+ return data
71
+ })
72
+
73
+ export function get(sessionID: string) {
74
+ return (
75
+ state()[sessionID] ?? {
76
+ type: "idle",
77
+ }
78
+ )
79
+ }
80
+
81
+ export function list() {
82
+ return state()
83
+ }
84
+
85
+ export function set(sessionID: string, status: Info) {
86
+ Bus.publish(Event.Status, {
87
+ sessionID,
88
+ status,
89
+ })
90
+ if (status.type === "idle") {
91
+ // deprecated
92
+ Bus.publish(Event.Idle, {
93
+ sessionID,
94
+ })
95
+ delete state()[sessionID]
96
+ return
97
+ }
98
+ state()[sessionID] = status
99
+ }
100
+ }
@@ -0,0 +1,217 @@
1
+ import { Provider } from "@/provider/provider"
2
+
3
+ import { fn } from "@/util/fn"
4
+ import z from "zod"
5
+ import { Session } from "."
6
+
7
+ import { MessageV2 } from "./message-v2"
8
+ import { Identifier } from "@/id/id"
9
+ import { Snapshot } from "@/snapshot"
10
+
11
+ import { Log } from "@/util/log"
12
+ import path from "path"
13
+ import { Instance } from "@/project/instance"
14
+ import { Storage } from "@/storage/storage"
15
+ import { Bus } from "@/bus"
16
+
17
+ import { LLM } from "./llm"
18
+ import { Agent } from "@/agent/agent"
19
+
20
+ export namespace SessionSummary {
21
+ const log = Log.create({ service: "session.summary" })
22
+
23
+ function unquoteGitPath(input: string) {
24
+ if (!input.startsWith('"')) return input
25
+ if (!input.endsWith('"')) return input
26
+ const body = input.slice(1, -1)
27
+ const bytes: number[] = []
28
+
29
+ for (let i = 0; i < body.length; i++) {
30
+ const char = body[i]!
31
+ if (char !== "\\") {
32
+ bytes.push(char.charCodeAt(0))
33
+ continue
34
+ }
35
+
36
+ const next = body[i + 1]
37
+ if (!next) {
38
+ bytes.push("\\".charCodeAt(0))
39
+ continue
40
+ }
41
+
42
+ if (next >= "0" && next <= "7") {
43
+ const chunk = body.slice(i + 1, i + 4)
44
+ const match = chunk.match(/^[0-7]{1,3}/)
45
+ if (!match) {
46
+ bytes.push(next.charCodeAt(0))
47
+ i++
48
+ continue
49
+ }
50
+ bytes.push(parseInt(match[0], 8))
51
+ i += match[0].length
52
+ continue
53
+ }
54
+
55
+ const escaped =
56
+ next === "n"
57
+ ? "\n"
58
+ : next === "r"
59
+ ? "\r"
60
+ : next === "t"
61
+ ? "\t"
62
+ : next === "b"
63
+ ? "\b"
64
+ : next === "f"
65
+ ? "\f"
66
+ : next === "v"
67
+ ? "\v"
68
+ : next === "\\" || next === '"'
69
+ ? next
70
+ : undefined
71
+
72
+ bytes.push((escaped ?? next).charCodeAt(0))
73
+ i++
74
+ }
75
+
76
+ return Buffer.from(bytes).toString()
77
+ }
78
+
79
+ export const summarize = fn(
80
+ z.object({
81
+ sessionID: z.string(),
82
+ messageID: z.string(),
83
+ }),
84
+ async (input) => {
85
+ const all = await Session.messages({ sessionID: input.sessionID })
86
+ await Promise.all([
87
+ summarizeSession({ sessionID: input.sessionID, messages: all }),
88
+ summarizeMessage({ messageID: input.messageID, messages: all }),
89
+ ])
90
+ },
91
+ )
92
+
93
+ async function summarizeSession(input: { sessionID: string; messages: MessageV2.WithParts[] }) {
94
+ const files = new Set(
95
+ input.messages
96
+ .flatMap((x) => x.parts)
97
+ .filter((x) => x.type === "patch")
98
+ .flatMap((x) => x.files)
99
+ .map((x) => path.relative(Instance.worktree, x).replaceAll("\\", "/")),
100
+ )
101
+ const diffs = await computeDiff({ messages: input.messages }).then((x) =>
102
+ x.filter((x) => {
103
+ return files.has(x.file)
104
+ }),
105
+ )
106
+ await Session.update(input.sessionID, (draft) => {
107
+ draft.summary = {
108
+ additions: diffs.reduce((sum, x) => sum + x.additions, 0),
109
+ deletions: diffs.reduce((sum, x) => sum + x.deletions, 0),
110
+ files: diffs.length,
111
+ }
112
+ })
113
+ await Storage.write(["session_diff", input.sessionID], diffs)
114
+ Bus.publish(Session.Event.Diff, {
115
+ sessionID: input.sessionID,
116
+ diff: diffs,
117
+ })
118
+ }
119
+
120
+ async function summarizeMessage(input: { messageID: string; messages: MessageV2.WithParts[] }) {
121
+ const messages = input.messages.filter(
122
+ (m) => m.info.id === input.messageID || (m.info.role === "assistant" && m.info.parentID === input.messageID),
123
+ )
124
+ const msgWithParts = messages.find((m) => m.info.id === input.messageID)!
125
+ const userMsg = msgWithParts.info as MessageV2.User
126
+ const diffs = await computeDiff({ messages })
127
+ userMsg.summary = {
128
+ ...userMsg.summary,
129
+ diffs,
130
+ }
131
+ await Session.updateMessage(userMsg)
132
+
133
+ const textPart = msgWithParts.parts.find((p) => p.type === "text" && !p.synthetic) as MessageV2.TextPart
134
+ if (textPart && !userMsg.summary?.title) {
135
+ const agent = await Agent.get("title")
136
+ if (!agent) return
137
+ const stream = await LLM.stream({
138
+ agent,
139
+ user: userMsg,
140
+ tools: {},
141
+ model: agent.model
142
+ ? await Provider.getModel(agent.model.providerID, agent.model.modelID)
143
+ : ((await Provider.getSmallModel(userMsg.model.providerID)) ??
144
+ (await Provider.getModel(userMsg.model.providerID, userMsg.model.modelID))),
145
+ small: true,
146
+ messages: [
147
+ {
148
+ role: "user" as const,
149
+ content: `
150
+ The following is the text to summarize:
151
+ <text>
152
+ ${textPart?.text ?? ""}
153
+ </text>
154
+ `,
155
+ },
156
+ ],
157
+ abort: new AbortController().signal,
158
+ sessionID: userMsg.sessionID,
159
+ system: [],
160
+ retries: 3,
161
+ })
162
+ const result = await stream.text
163
+ log.info("title", { title: result })
164
+ userMsg.summary.title = result
165
+ await Session.updateMessage(userMsg)
166
+ }
167
+ }
168
+
169
+ export const diff = fn(
170
+ z.object({
171
+ sessionID: Identifier.schema("session"),
172
+ messageID: Identifier.schema("message").optional(),
173
+ }),
174
+ async (input) => {
175
+ const diffs = await Storage.read<Snapshot.FileDiff[]>(["session_diff", input.sessionID]).catch(() => [])
176
+ const next = diffs.map((item) => {
177
+ const file = unquoteGitPath(item.file)
178
+ if (file === item.file) return item
179
+ return {
180
+ ...item,
181
+ file,
182
+ }
183
+ })
184
+ const changed = next.some((item, i) => item.file !== diffs[i]?.file)
185
+ if (changed) Storage.write(["session_diff", input.sessionID], next).catch(() => {})
186
+ return next
187
+ },
188
+ )
189
+
190
+ export async function computeDiff(input: { messages: MessageV2.WithParts[] }) {
191
+ let from: string | undefined
192
+ let to: string | undefined
193
+
194
+ // scan assistant messages to find earliest from and latest to
195
+ // snapshot
196
+ for (const item of input.messages) {
197
+ if (!from) {
198
+ for (const part of item.parts) {
199
+ if (part.type === "step-start" && part.snapshot) {
200
+ from = part.snapshot
201
+ break
202
+ }
203
+ }
204
+ }
205
+
206
+ for (const part of item.parts) {
207
+ if (part.type === "step-finish" && part.snapshot) {
208
+ to = part.snapshot
209
+ break
210
+ }
211
+ }
212
+ }
213
+
214
+ if (from && to) return Snapshot.diffFull(from, to)
215
+ return []
216
+ }
217
+ }
@@ -0,0 +1,52 @@
1
+ import { Ripgrep } from "../file/ripgrep"
2
+
3
+ import { Instance } from "../project/instance"
4
+
5
+ import PROMPT_ANTHROPIC from "./prompt/anthropic.txt"
6
+ import PROMPT_ANTHROPIC_WITHOUT_TODO from "./prompt/qwen.txt"
7
+ import PROMPT_BEAST from "./prompt/beast.txt"
8
+ import PROMPT_GEMINI from "./prompt/gemini.txt"
9
+
10
+ import PROMPT_CODEX from "./prompt/codex_header.txt"
11
+ import type { Provider } from "@/provider/provider"
12
+
13
+ export namespace SystemPrompt {
14
+ export function instructions() {
15
+ return PROMPT_CODEX.trim()
16
+ }
17
+
18
+ export function provider(model: Provider.Model) {
19
+ if (model.api.id.includes("gpt-5")) return [PROMPT_CODEX]
20
+ if (model.api.id.includes("gpt-") || model.api.id.includes("o1") || model.api.id.includes("o3"))
21
+ return [PROMPT_BEAST]
22
+ if (model.api.id.includes("gemini-")) return [PROMPT_GEMINI]
23
+ if (model.api.id.includes("claude")) return [PROMPT_ANTHROPIC]
24
+ return [PROMPT_ANTHROPIC_WITHOUT_TODO]
25
+ }
26
+
27
+ export async function environment(model: Provider.Model) {
28
+ const project = Instance.project
29
+ return [
30
+ [
31
+ `You are powered by the model named ${model.api.id}. The exact model ID is ${model.providerID}/${model.api.id}`,
32
+ `Here is some useful information about the environment you are running in:`,
33
+ `<env>`,
34
+ ` Working directory: ${Instance.directory}`,
35
+ ` Is directory a git repo: ${project.vcs === "git" ? "yes" : "no"}`,
36
+ ` Platform: ${process.platform}`,
37
+ ` Today's date: ${new Date().toDateString()}`,
38
+ `</env>`,
39
+ `<files>`,
40
+ ` ${
41
+ project.vcs === "git" && false
42
+ ? await Ripgrep.tree({
43
+ cwd: Instance.directory,
44
+ limit: 200,
45
+ })
46
+ : ""
47
+ }`,
48
+ `</files>`,
49
+ ].join("\n"),
50
+ ]
51
+ }
52
+ }
@@ -0,0 +1,37 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import z from "zod"
4
+ import { Storage } from "../storage/storage"
5
+
6
+ export namespace Todo {
7
+ export const Info = z
8
+ .object({
9
+ content: z.string().describe("Brief description of the task"),
10
+ status: z.string().describe("Current status of the task: pending, in_progress, completed, cancelled"),
11
+ priority: z.string().describe("Priority level of the task: high, medium, low"),
12
+ id: z.string().describe("Unique identifier for the todo item"),
13
+ })
14
+ .meta({ ref: "Todo" })
15
+ export type Info = z.infer<typeof Info>
16
+
17
+ export const Event = {
18
+ Updated: BusEvent.define(
19
+ "todo.updated",
20
+ z.object({
21
+ sessionID: z.string(),
22
+ todos: z.array(Info),
23
+ }),
24
+ ),
25
+ }
26
+
27
+ export async function update(input: { sessionID: string; todos: Info[] }) {
28
+ await Storage.write(["todo", input.sessionID], input.todos)
29
+ Bus.publish(Event.Updated, input)
30
+ }
31
+
32
+ export async function get(sessionID: string) {
33
+ return Storage.read<Info[]>(["todo", sessionID])
34
+ .then((x) => x || [])
35
+ .catch(() => [])
36
+ }
37
+ }