@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
package/src/index.ts ADDED
@@ -0,0 +1,159 @@
1
+ import yargs from "yargs"
2
+ import { hideBin } from "yargs/helpers"
3
+ import { RunCommand } from "./cli/cmd/run"
4
+ import { GenerateCommand } from "./cli/cmd/generate"
5
+ import { Log } from "./util/log"
6
+ import { AuthCommand } from "./cli/cmd/auth"
7
+ import { AgentCommand } from "./cli/cmd/agent"
8
+ import { UpgradeCommand } from "./cli/cmd/upgrade"
9
+ import { UninstallCommand } from "./cli/cmd/uninstall"
10
+ import { ModelsCommand } from "./cli/cmd/models"
11
+ import { UI } from "./cli/ui"
12
+ import { Installation } from "./installation"
13
+ import { NamedError } from "@/util/error"
14
+ import { FormatError } from "./cli/error"
15
+ import { ServeCommand } from "./cli/cmd/serve"
16
+ import { DebugCommand } from "./cli/cmd/debug"
17
+ import { StatsCommand } from "./cli/cmd/stats"
18
+ import { McpCommand } from "./cli/cmd/mcp"
19
+ import { GithubCommand } from "./cli/cmd/github"
20
+ import { ExportCommand } from "./cli/cmd/export"
21
+ import { ImportCommand } from "./cli/cmd/import"
22
+ import { AttachCommand } from "./cli/cmd/tui/attach"
23
+ import { TuiThreadCommand } from "./cli/cmd/tui/thread"
24
+ import { AcpCommand } from "./cli/cmd/acp"
25
+ import { EOL } from "os"
26
+ import { WebCommand } from "./cli/cmd/web"
27
+ import { PrCommand } from "./cli/cmd/pr"
28
+ import { SessionCommand } from "./cli/cmd/session"
29
+
30
+ process.on("unhandledRejection", (e) => {
31
+ Log.Default.error("rejection", {
32
+ e: e instanceof Error ? e.message : e,
33
+ })
34
+ })
35
+
36
+ process.on("uncaughtException", (e) => {
37
+ Log.Default.error("exception", {
38
+ e: e instanceof Error ? e.message : e,
39
+ })
40
+ })
41
+
42
+ const cli = yargs(hideBin(process.argv))
43
+ .parserConfiguration({ "populate--": true })
44
+ .scriptName("bloxycode")
45
+ .wrap(100)
46
+ .help("help", "show help")
47
+ .alias("help", "h")
48
+ .version("version", "show version number", Installation.VERSION)
49
+ .alias("version", "v")
50
+ .option("print-logs", {
51
+ describe: "print logs to stderr",
52
+ type: "boolean",
53
+ })
54
+ .option("log-level", {
55
+ describe: "log level",
56
+ type: "string",
57
+ choices: ["DEBUG", "INFO", "WARN", "ERROR"],
58
+ })
59
+ .middleware(async (opts) => {
60
+ await Log.init({
61
+ print: process.argv.includes("--print-logs"),
62
+ dev: Installation.isLocal(),
63
+ level: (() => {
64
+ if (opts.logLevel) return opts.logLevel as Log.Level
65
+ if (Installation.isLocal()) return "DEBUG"
66
+ return "INFO"
67
+ })(),
68
+ })
69
+
70
+ process.env.AGENT = "1"
71
+ process.env.BLOXYCODE = "1"
72
+
73
+ Log.Default.info("bloxycode", {
74
+ version: Installation.VERSION,
75
+ args: process.argv.slice(2),
76
+ })
77
+ })
78
+ .usage("\n" + UI.logo())
79
+ .completion("completion", "generate shell completion script")
80
+ .command(AcpCommand)
81
+ .command(McpCommand)
82
+ .command(TuiThreadCommand)
83
+ .command(AttachCommand)
84
+ .command(RunCommand)
85
+ .command(GenerateCommand)
86
+ .command(DebugCommand)
87
+ .command(AuthCommand)
88
+ .command(AgentCommand)
89
+ .command(UpgradeCommand)
90
+ .command(UninstallCommand)
91
+ .command(ServeCommand)
92
+ .command(WebCommand)
93
+ .command(ModelsCommand)
94
+ .command(StatsCommand)
95
+ .command(ExportCommand)
96
+ .command(ImportCommand)
97
+ .command(GithubCommand)
98
+ .command(PrCommand)
99
+ .command(SessionCommand)
100
+ .fail((msg, err) => {
101
+ if (
102
+ msg?.startsWith("Unknown argument") ||
103
+ msg?.startsWith("Not enough non-option arguments") ||
104
+ msg?.startsWith("Invalid values:")
105
+ ) {
106
+ if (err) throw err
107
+ cli.showHelp("log")
108
+ }
109
+ if (err) throw err
110
+ process.exit(1)
111
+ })
112
+ .strict()
113
+
114
+ try {
115
+ await cli.parse()
116
+ } catch (e) {
117
+ let data: Record<string, any> = {}
118
+ if (e instanceof NamedError) {
119
+ const obj = e.toObject()
120
+ Object.assign(data, {
121
+ ...obj.data,
122
+ })
123
+ }
124
+
125
+ if (e instanceof Error) {
126
+ Object.assign(data, {
127
+ name: e.name,
128
+ message: e.message,
129
+ cause: e.cause?.toString(),
130
+ stack: e.stack,
131
+ })
132
+ }
133
+
134
+ if (e instanceof ResolveMessage) {
135
+ Object.assign(data, {
136
+ name: e.name,
137
+ message: e.message,
138
+ code: e.code,
139
+ specifier: e.specifier,
140
+ referrer: e.referrer,
141
+ position: e.position,
142
+ importKind: e.importKind,
143
+ })
144
+ }
145
+ Log.Default.error("fatal", data)
146
+ const formatted = FormatError(e)
147
+ if (formatted) UI.error(formatted)
148
+ if (formatted === undefined) {
149
+ UI.error("Unexpected error, check log file at " + Log.file() + " for more details" + EOL)
150
+ console.error(e instanceof Error ? e.message : String(e))
151
+ }
152
+ process.exitCode = 1
153
+ } finally {
154
+ // Some subprocesses don't react properly to SIGTERM and similar signals.
155
+ // Most notably, some docker-container-based MCP servers don't handle such signals unless
156
+ // run using `docker run --init`.
157
+ // Explicitly exit to avoid any hanging subprocesses.
158
+ process.exit()
159
+ }
@@ -0,0 +1,246 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import path from "path"
3
+ import { $ } from "bun"
4
+ import z from "zod"
5
+ import { NamedError } from "@/util/error"
6
+ import { Log } from "../util/log"
7
+ import { iife } from "@/util/iife"
8
+ import { Flag } from "../flag/flag"
9
+
10
+ declare global {
11
+ const BLOXYCODE_VERSION: string
12
+ const BLOXYCODE_CHANNEL: string
13
+ }
14
+
15
+ export namespace Installation {
16
+ const log = Log.create({ service: "installation" })
17
+
18
+ export type Method = Awaited<ReturnType<typeof method>>
19
+
20
+ export const Event = {
21
+ Updated: BusEvent.define(
22
+ "installation.updated",
23
+ z.object({
24
+ version: z.string(),
25
+ }),
26
+ ),
27
+ UpdateAvailable: BusEvent.define(
28
+ "installation.update-available",
29
+ z.object({
30
+ version: z.string(),
31
+ }),
32
+ ),
33
+ }
34
+
35
+ export const Info = z
36
+ .object({
37
+ version: z.string(),
38
+ latest: z.string(),
39
+ })
40
+ .meta({
41
+ ref: "InstallationInfo",
42
+ })
43
+ export type Info = z.infer<typeof Info>
44
+
45
+ export async function info() {
46
+ return {
47
+ version: VERSION,
48
+ latest: await latest(),
49
+ }
50
+ }
51
+
52
+ export function isPreview() {
53
+ return CHANNEL !== "latest"
54
+ }
55
+
56
+ export function isLocal() {
57
+ return CHANNEL === "local"
58
+ }
59
+
60
+ export async function method() {
61
+ if (process.execPath.includes(path.join(".bloxycode", "bin"))) return "curl"
62
+ if (process.execPath.includes(path.join(".local", "bin"))) return "curl"
63
+ const exec = process.execPath.toLowerCase()
64
+
65
+ const checks = [
66
+ {
67
+ name: "npm" as const,
68
+ command: () => $`npm list -g --depth=0`.throws(false).quiet().text(),
69
+ },
70
+ {
71
+ name: "yarn" as const,
72
+ command: () => $`yarn global list`.throws(false).quiet().text(),
73
+ },
74
+ {
75
+ name: "pnpm" as const,
76
+ command: () => $`pnpm list -g --depth=0`.throws(false).quiet().text(),
77
+ },
78
+ {
79
+ name: "bun" as const,
80
+ command: () => $`bun pm ls -g`.throws(false).quiet().text(),
81
+ },
82
+ {
83
+ name: "brew" as const,
84
+ command: () => $`brew list --formula opencode`.throws(false).quiet().text(),
85
+ },
86
+ {
87
+ name: "scoop" as const,
88
+ command: () => $`scoop list opencode`.throws(false).quiet().text(),
89
+ },
90
+ {
91
+ name: "choco" as const,
92
+ command: () => $`choco list --limit-output opencode`.throws(false).quiet().text(),
93
+ },
94
+ ]
95
+
96
+ checks.sort((a, b) => {
97
+ const aMatches = exec.includes(a.name)
98
+ const bMatches = exec.includes(b.name)
99
+ if (aMatches && !bMatches) return -1
100
+ if (!aMatches && bMatches) return 1
101
+ return 0
102
+ })
103
+
104
+ for (const check of checks) {
105
+ const output = await check.command()
106
+ const installedName =
107
+ check.name === "brew" || check.name === "choco" || check.name === "scoop" ? "opencode" : "opencode-ai"
108
+ if (output.includes(installedName)) {
109
+ return check.name
110
+ }
111
+ }
112
+
113
+ return "unknown"
114
+ }
115
+
116
+ export const UpgradeFailedError = NamedError.create(
117
+ "UpgradeFailedError",
118
+ z.object({
119
+ stderr: z.string(),
120
+ }),
121
+ )
122
+
123
+ async function getBrewFormula() {
124
+ const tapFormula = await $`brew list --formula bloxystudios/tap/opencode`.throws(false).quiet().text()
125
+ if (tapFormula.includes("opencode")) return "bloxystudios/tap/opencode"
126
+ const coreFormula = await $`brew list --formula opencode`.throws(false).quiet().text()
127
+ if (coreFormula.includes("opencode")) return "opencode"
128
+ return "opencode"
129
+ }
130
+
131
+ export async function upgrade(method: Method, target: string) {
132
+ let cmd
133
+ switch (method) {
134
+ case "curl":
135
+ cmd = $`curl -fsSL https://opencode.ai/install | bash`.env({
136
+ ...process.env,
137
+ VERSION: target,
138
+ })
139
+ break
140
+ case "npm":
141
+ cmd = $`npm install -g opencode-ai@${target}`
142
+ break
143
+ case "pnpm":
144
+ cmd = $`pnpm install -g opencode-ai@${target}`
145
+ break
146
+ case "bun":
147
+ cmd = $`bun install -g opencode-ai@${target}`
148
+ break
149
+ case "brew": {
150
+ const formula = await getBrewFormula()
151
+ cmd = $`brew upgrade ${formula}`.env({
152
+ HOMEBREW_NO_AUTO_UPDATE: "1",
153
+ ...process.env,
154
+ })
155
+ break
156
+ }
157
+ case "choco":
158
+ cmd = $`echo Y | choco upgrade opencode --version=${target}`
159
+ break
160
+ case "scoop":
161
+ cmd = $`scoop install opencode@${target}`
162
+ break
163
+ default:
164
+ throw new Error(`Unknown method: ${method}`)
165
+ }
166
+ const result = await cmd.quiet().throws(false)
167
+ if (result.exitCode !== 0) {
168
+ const stderr = method === "choco" ? "not running from an elevated command shell" : result.stderr.toString("utf8")
169
+ throw new UpgradeFailedError({
170
+ stderr: stderr,
171
+ })
172
+ }
173
+ log.info("upgraded", {
174
+ method,
175
+ target,
176
+ stdout: result.stdout.toString(),
177
+ stderr: result.stderr.toString(),
178
+ })
179
+ await $`${process.execPath} --version`.nothrow().quiet().text()
180
+ }
181
+
182
+ export const VERSION = typeof BLOXYCODE_VERSION === "string" ? BLOXYCODE_VERSION : "local"
183
+ export const CHANNEL = typeof BLOXYCODE_CHANNEL === "string" ? BLOXYCODE_CHANNEL : "local"
184
+ export const USER_AGENT = `opencode/${CHANNEL}/${VERSION}/${Flag.BLOXYCODE_CLIENT}`
185
+
186
+ export async function latest(installMethod?: Method) {
187
+ const detectedMethod = installMethod || (await method())
188
+
189
+ if (detectedMethod === "brew") {
190
+ const formula = await getBrewFormula()
191
+ if (formula === "opencode") {
192
+ return fetch("https://formulae.brew.sh/api/formula/bloxycode.json")
193
+ .then((res) => {
194
+ if (!res.ok) throw new Error(res.statusText)
195
+ return res.json()
196
+ })
197
+ .then((data: any) => data.versions.stable)
198
+ }
199
+ }
200
+
201
+ if (detectedMethod === "npm" || detectedMethod === "bun" || detectedMethod === "pnpm") {
202
+ const registry = await iife(async () => {
203
+ const r = (await $`npm config get registry`.quiet().nothrow().text()).trim()
204
+ const reg = r || "https://registry.npmjs.org"
205
+ return reg.endsWith("/") ? reg.slice(0, -1) : reg
206
+ })
207
+ const channel = CHANNEL
208
+ return fetch(`${registry}/opencode-ai/${channel}`)
209
+ .then((res) => {
210
+ if (!res.ok) throw new Error(res.statusText)
211
+ return res.json()
212
+ })
213
+ .then((data: any) => data.version)
214
+ }
215
+
216
+ if (detectedMethod === "choco") {
217
+ return fetch(
218
+ "https://community.chocolatey.org/api/v2/Packages?$filter=Id%20eq%20%27opencode%27%20and%20IsLatestVersion&$select=Version",
219
+ { headers: { Accept: "application/json;odata=verbose" } },
220
+ )
221
+ .then((res) => {
222
+ if (!res.ok) throw new Error(res.statusText)
223
+ return res.json()
224
+ })
225
+ .then((data: any) => data.d.results[0].Version)
226
+ }
227
+
228
+ if (detectedMethod === "scoop") {
229
+ return fetch("https://raw.githubusercontent.com/ScoopInstaller/Main/master/bucket/bloxycode.json", {
230
+ headers: { Accept: "application/json" },
231
+ })
232
+ .then((res) => {
233
+ if (!res.ok) throw new Error(res.statusText)
234
+ return res.json()
235
+ })
236
+ .then((data: any) => data.version)
237
+ }
238
+
239
+ return fetch("https://api.github.com/repos/bloxystudios/bloxycode/releases/latest")
240
+ .then((res) => {
241
+ if (!res.ok) throw new Error(res.statusText)
242
+ return res.json()
243
+ })
244
+ .then((data: any) => data.tag_name.replace(/^v/, ""))
245
+ }
246
+ }
@@ -0,0 +1,252 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import path from "path"
4
+ import { pathToFileURL, fileURLToPath } from "url"
5
+ import { createMessageConnection, StreamMessageReader, StreamMessageWriter } from "vscode-jsonrpc/node"
6
+ import type { Diagnostic as VSCodeDiagnostic } from "vscode-languageserver-types"
7
+ import { Log } from "../util/log"
8
+ import { LANGUAGE_EXTENSIONS } from "./language"
9
+ import z from "zod"
10
+ import type { LSPServer } from "./server"
11
+ import { NamedError } from "@/util/error"
12
+ import { withTimeout } from "../util/timeout"
13
+ import { Instance } from "../project/instance"
14
+ import { Filesystem } from "../util/filesystem"
15
+
16
+ const DIAGNOSTICS_DEBOUNCE_MS = 150
17
+
18
+ export namespace LSPClient {
19
+ const log = Log.create({ service: "lsp.client" })
20
+
21
+ export type Info = NonNullable<Awaited<ReturnType<typeof create>>>
22
+
23
+ export type Diagnostic = VSCodeDiagnostic
24
+
25
+ export const InitializeError = NamedError.create(
26
+ "LSPInitializeError",
27
+ z.object({
28
+ serverID: z.string(),
29
+ }),
30
+ )
31
+
32
+ export const Event = {
33
+ Diagnostics: BusEvent.define(
34
+ "lsp.client.diagnostics",
35
+ z.object({
36
+ serverID: z.string(),
37
+ path: z.string(),
38
+ }),
39
+ ),
40
+ }
41
+
42
+ export async function create(input: { serverID: string; server: LSPServer.Handle; root: string }) {
43
+ const l = log.clone().tag("serverID", input.serverID)
44
+ l.info("starting client")
45
+
46
+ const connection = createMessageConnection(
47
+ new StreamMessageReader(input.server.process.stdout as any),
48
+ new StreamMessageWriter(input.server.process.stdin as any),
49
+ )
50
+
51
+ const diagnostics = new Map<string, Diagnostic[]>()
52
+ connection.onNotification("textDocument/publishDiagnostics", (params) => {
53
+ const filePath = Filesystem.normalizePath(fileURLToPath(params.uri))
54
+ l.info("textDocument/publishDiagnostics", {
55
+ path: filePath,
56
+ count: params.diagnostics.length,
57
+ })
58
+ const exists = diagnostics.has(filePath)
59
+ diagnostics.set(filePath, params.diagnostics)
60
+ if (!exists && input.serverID === "typescript") return
61
+ Bus.publish(Event.Diagnostics, { path: filePath, serverID: input.serverID })
62
+ })
63
+ connection.onRequest("window/workDoneProgress/create", (params) => {
64
+ l.info("window/workDoneProgress/create", params)
65
+ return null
66
+ })
67
+ connection.onRequest("workspace/configuration", async () => {
68
+ // Return server initialization options
69
+ return [input.server.initialization ?? {}]
70
+ })
71
+ connection.onRequest("client/registerCapability", async () => {})
72
+ connection.onRequest("client/unregisterCapability", async () => {})
73
+ connection.onRequest("workspace/workspaceFolders", async () => [
74
+ {
75
+ name: "workspace",
76
+ uri: pathToFileURL(input.root).href,
77
+ },
78
+ ])
79
+ connection.listen()
80
+
81
+ l.info("sending initialize")
82
+ await withTimeout(
83
+ connection.sendRequest("initialize", {
84
+ rootUri: pathToFileURL(input.root).href,
85
+ processId: input.server.process.pid,
86
+ workspaceFolders: [
87
+ {
88
+ name: "workspace",
89
+ uri: pathToFileURL(input.root).href,
90
+ },
91
+ ],
92
+ initializationOptions: {
93
+ ...input.server.initialization,
94
+ },
95
+ capabilities: {
96
+ window: {
97
+ workDoneProgress: true,
98
+ },
99
+ workspace: {
100
+ configuration: true,
101
+ didChangeWatchedFiles: {
102
+ dynamicRegistration: true,
103
+ },
104
+ },
105
+ textDocument: {
106
+ synchronization: {
107
+ didOpen: true,
108
+ didChange: true,
109
+ },
110
+ publishDiagnostics: {
111
+ versionSupport: true,
112
+ },
113
+ },
114
+ },
115
+ }),
116
+ 45_000,
117
+ ).catch((err) => {
118
+ l.error("initialize error", { error: err })
119
+ throw new InitializeError(
120
+ { serverID: input.serverID },
121
+ {
122
+ cause: err,
123
+ },
124
+ )
125
+ })
126
+
127
+ await connection.sendNotification("initialized", {})
128
+
129
+ if (input.server.initialization) {
130
+ await connection.sendNotification("workspace/didChangeConfiguration", {
131
+ settings: input.server.initialization,
132
+ })
133
+ }
134
+
135
+ const files: {
136
+ [path: string]: number
137
+ } = {}
138
+
139
+ const result = {
140
+ root: input.root,
141
+ get serverID() {
142
+ return input.serverID
143
+ },
144
+ get connection() {
145
+ return connection
146
+ },
147
+ notify: {
148
+ async open(input: { path: string }) {
149
+ input.path = path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path)
150
+ const file = Bun.file(input.path)
151
+ const text = await file.text()
152
+ const extension = path.extname(input.path)
153
+ const languageId = LANGUAGE_EXTENSIONS[extension] ?? "plaintext"
154
+
155
+ const version = files[input.path]
156
+ if (version !== undefined) {
157
+ log.info("workspace/didChangeWatchedFiles", input)
158
+ await connection.sendNotification("workspace/didChangeWatchedFiles", {
159
+ changes: [
160
+ {
161
+ uri: pathToFileURL(input.path).href,
162
+ type: 2, // Changed
163
+ },
164
+ ],
165
+ })
166
+
167
+ const next = version + 1
168
+ files[input.path] = next
169
+ log.info("textDocument/didChange", {
170
+ path: input.path,
171
+ version: next,
172
+ })
173
+ await connection.sendNotification("textDocument/didChange", {
174
+ textDocument: {
175
+ uri: pathToFileURL(input.path).href,
176
+ version: next,
177
+ },
178
+ contentChanges: [{ text }],
179
+ })
180
+ return
181
+ }
182
+
183
+ log.info("workspace/didChangeWatchedFiles", input)
184
+ await connection.sendNotification("workspace/didChangeWatchedFiles", {
185
+ changes: [
186
+ {
187
+ uri: pathToFileURL(input.path).href,
188
+ type: 1, // Created
189
+ },
190
+ ],
191
+ })
192
+
193
+ log.info("textDocument/didOpen", input)
194
+ diagnostics.delete(input.path)
195
+ await connection.sendNotification("textDocument/didOpen", {
196
+ textDocument: {
197
+ uri: pathToFileURL(input.path).href,
198
+ languageId,
199
+ version: 0,
200
+ text,
201
+ },
202
+ })
203
+ files[input.path] = 0
204
+ return
205
+ },
206
+ },
207
+ get diagnostics() {
208
+ return diagnostics
209
+ },
210
+ async waitForDiagnostics(input: { path: string }) {
211
+ const normalizedPath = Filesystem.normalizePath(
212
+ path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path),
213
+ )
214
+ log.info("waiting for diagnostics", { path: normalizedPath })
215
+ let unsub: () => void
216
+ let debounceTimer: ReturnType<typeof setTimeout> | undefined
217
+ return await withTimeout(
218
+ new Promise<void>((resolve) => {
219
+ unsub = Bus.subscribe(Event.Diagnostics, (event) => {
220
+ if (event.properties.path === normalizedPath && event.properties.serverID === result.serverID) {
221
+ // Debounce to allow LSP to send follow-up diagnostics (e.g., semantic after syntax)
222
+ if (debounceTimer) clearTimeout(debounceTimer)
223
+ debounceTimer = setTimeout(() => {
224
+ log.info("got diagnostics", { path: normalizedPath })
225
+ unsub?.()
226
+ resolve()
227
+ }, DIAGNOSTICS_DEBOUNCE_MS)
228
+ }
229
+ })
230
+ }),
231
+ 3000,
232
+ )
233
+ .catch(() => {})
234
+ .finally(() => {
235
+ if (debounceTimer) clearTimeout(debounceTimer)
236
+ unsub?.()
237
+ })
238
+ },
239
+ async shutdown() {
240
+ l.info("shutting down")
241
+ connection.end()
242
+ connection.dispose()
243
+ input.server.process.kill()
244
+ l.info("shutdown")
245
+ },
246
+ }
247
+
248
+ l.info("initialized")
249
+
250
+ return result
251
+ }
252
+ }