@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,225 @@
1
+ import { Hono } from "hono"
2
+ import { describeRoute, validator, resolver } from "hono-openapi"
3
+ import z from "zod"
4
+ import { MCP } from "../../mcp"
5
+ import { Config } from "../../config/config"
6
+ import { errors } from "../error"
7
+ import { lazy } from "../../util/lazy"
8
+
9
+ export const McpRoutes = lazy(() =>
10
+ new Hono()
11
+ .get(
12
+ "/",
13
+ describeRoute({
14
+ summary: "Get MCP status",
15
+ description: "Get the status of all Model Context Protocol (MCP) servers.",
16
+ operationId: "mcp.status",
17
+ responses: {
18
+ 200: {
19
+ description: "MCP server status",
20
+ content: {
21
+ "application/json": {
22
+ schema: resolver(z.record(z.string(), MCP.Status)),
23
+ },
24
+ },
25
+ },
26
+ },
27
+ }),
28
+ async (c) => {
29
+ return c.json(await MCP.status())
30
+ },
31
+ )
32
+ .post(
33
+ "/",
34
+ describeRoute({
35
+ summary: "Add MCP server",
36
+ description: "Dynamically add a new Model Context Protocol (MCP) server to the system.",
37
+ operationId: "mcp.add",
38
+ responses: {
39
+ 200: {
40
+ description: "MCP server added successfully",
41
+ content: {
42
+ "application/json": {
43
+ schema: resolver(z.record(z.string(), MCP.Status)),
44
+ },
45
+ },
46
+ },
47
+ ...errors(400),
48
+ },
49
+ }),
50
+ validator(
51
+ "json",
52
+ z.object({
53
+ name: z.string(),
54
+ config: Config.Mcp,
55
+ }),
56
+ ),
57
+ async (c) => {
58
+ const { name, config } = c.req.valid("json")
59
+ const result = await MCP.add(name, config)
60
+ return c.json(result.status)
61
+ },
62
+ )
63
+ .post(
64
+ "/:name/auth",
65
+ describeRoute({
66
+ summary: "Start MCP OAuth",
67
+ description: "Start OAuth authentication flow for a Model Context Protocol (MCP) server.",
68
+ operationId: "mcp.auth.start",
69
+ responses: {
70
+ 200: {
71
+ description: "OAuth flow started",
72
+ content: {
73
+ "application/json": {
74
+ schema: resolver(
75
+ z.object({
76
+ authorizationUrl: z.string().describe("URL to open in browser for authorization"),
77
+ }),
78
+ ),
79
+ },
80
+ },
81
+ },
82
+ ...errors(400, 404),
83
+ },
84
+ }),
85
+ async (c) => {
86
+ const name = c.req.param("name")
87
+ const supportsOAuth = await MCP.supportsOAuth(name)
88
+ if (!supportsOAuth) {
89
+ return c.json({ error: `MCP server ${name} does not support OAuth` }, 400)
90
+ }
91
+ const result = await MCP.startAuth(name)
92
+ return c.json(result)
93
+ },
94
+ )
95
+ .post(
96
+ "/:name/auth/callback",
97
+ describeRoute({
98
+ summary: "Complete MCP OAuth",
99
+ description:
100
+ "Complete OAuth authentication for a Model Context Protocol (MCP) server using the authorization code.",
101
+ operationId: "mcp.auth.callback",
102
+ responses: {
103
+ 200: {
104
+ description: "OAuth authentication completed",
105
+ content: {
106
+ "application/json": {
107
+ schema: resolver(MCP.Status),
108
+ },
109
+ },
110
+ },
111
+ ...errors(400, 404),
112
+ },
113
+ }),
114
+ validator(
115
+ "json",
116
+ z.object({
117
+ code: z.string().describe("Authorization code from OAuth callback"),
118
+ }),
119
+ ),
120
+ async (c) => {
121
+ const name = c.req.param("name")
122
+ const { code } = c.req.valid("json")
123
+ const status = await MCP.finishAuth(name, code)
124
+ return c.json(status)
125
+ },
126
+ )
127
+ .post(
128
+ "/:name/auth/authenticate",
129
+ describeRoute({
130
+ summary: "Authenticate MCP OAuth",
131
+ description: "Start OAuth flow and wait for callback (opens browser)",
132
+ operationId: "mcp.auth.authenticate",
133
+ responses: {
134
+ 200: {
135
+ description: "OAuth authentication completed",
136
+ content: {
137
+ "application/json": {
138
+ schema: resolver(MCP.Status),
139
+ },
140
+ },
141
+ },
142
+ ...errors(400, 404),
143
+ },
144
+ }),
145
+ async (c) => {
146
+ const name = c.req.param("name")
147
+ const supportsOAuth = await MCP.supportsOAuth(name)
148
+ if (!supportsOAuth) {
149
+ return c.json({ error: `MCP server ${name} does not support OAuth` }, 400)
150
+ }
151
+ const status = await MCP.authenticate(name)
152
+ return c.json(status)
153
+ },
154
+ )
155
+ .delete(
156
+ "/:name/auth",
157
+ describeRoute({
158
+ summary: "Remove MCP OAuth",
159
+ description: "Remove OAuth credentials for an MCP server",
160
+ operationId: "mcp.auth.remove",
161
+ responses: {
162
+ 200: {
163
+ description: "OAuth credentials removed",
164
+ content: {
165
+ "application/json": {
166
+ schema: resolver(z.object({ success: z.literal(true) })),
167
+ },
168
+ },
169
+ },
170
+ ...errors(404),
171
+ },
172
+ }),
173
+ async (c) => {
174
+ const name = c.req.param("name")
175
+ await MCP.removeAuth(name)
176
+ return c.json({ success: true as const })
177
+ },
178
+ )
179
+ .post(
180
+ "/:name/connect",
181
+ describeRoute({
182
+ description: "Connect an MCP server",
183
+ operationId: "mcp.connect",
184
+ responses: {
185
+ 200: {
186
+ description: "MCP server connected successfully",
187
+ content: {
188
+ "application/json": {
189
+ schema: resolver(z.boolean()),
190
+ },
191
+ },
192
+ },
193
+ },
194
+ }),
195
+ validator("param", z.object({ name: z.string() })),
196
+ async (c) => {
197
+ const { name } = c.req.valid("param")
198
+ await MCP.connect(name)
199
+ return c.json(true)
200
+ },
201
+ )
202
+ .post(
203
+ "/:name/disconnect",
204
+ describeRoute({
205
+ description: "Disconnect an MCP server",
206
+ operationId: "mcp.disconnect",
207
+ responses: {
208
+ 200: {
209
+ description: "MCP server disconnected successfully",
210
+ content: {
211
+ "application/json": {
212
+ schema: resolver(z.boolean()),
213
+ },
214
+ },
215
+ },
216
+ },
217
+ }),
218
+ validator("param", z.object({ name: z.string() })),
219
+ async (c) => {
220
+ const { name } = c.req.valid("param")
221
+ await MCP.disconnect(name)
222
+ return c.json(true)
223
+ },
224
+ ),
225
+ )
@@ -0,0 +1,68 @@
1
+ import { Hono } from "hono"
2
+ import { describeRoute, validator, resolver } from "hono-openapi"
3
+ import z from "zod"
4
+ import { PermissionNext } from "@/permission/next"
5
+ import { errors } from "../error"
6
+ import { lazy } from "../../util/lazy"
7
+
8
+ export const PermissionRoutes = lazy(() =>
9
+ new Hono()
10
+ .post(
11
+ "/:requestID/reply",
12
+ describeRoute({
13
+ summary: "Respond to permission request",
14
+ description: "Approve or deny a permission request from the AI assistant.",
15
+ operationId: "permission.reply",
16
+ responses: {
17
+ 200: {
18
+ description: "Permission processed successfully",
19
+ content: {
20
+ "application/json": {
21
+ schema: resolver(z.boolean()),
22
+ },
23
+ },
24
+ },
25
+ ...errors(400, 404),
26
+ },
27
+ }),
28
+ validator(
29
+ "param",
30
+ z.object({
31
+ requestID: z.string(),
32
+ }),
33
+ ),
34
+ validator("json", z.object({ reply: PermissionNext.Reply, message: z.string().optional() })),
35
+ async (c) => {
36
+ const params = c.req.valid("param")
37
+ const json = c.req.valid("json")
38
+ await PermissionNext.reply({
39
+ requestID: params.requestID,
40
+ reply: json.reply,
41
+ message: json.message,
42
+ })
43
+ return c.json(true)
44
+ },
45
+ )
46
+ .get(
47
+ "/",
48
+ describeRoute({
49
+ summary: "List pending permissions",
50
+ description: "Get all pending permission requests across all sessions.",
51
+ operationId: "permission.list",
52
+ responses: {
53
+ 200: {
54
+ description: "List of pending permissions",
55
+ content: {
56
+ "application/json": {
57
+ schema: resolver(PermissionNext.Request.array()),
58
+ },
59
+ },
60
+ },
61
+ },
62
+ }),
63
+ async (c) => {
64
+ const permissions = await PermissionNext.list()
65
+ return c.json(permissions)
66
+ },
67
+ ),
68
+ )
@@ -0,0 +1,82 @@
1
+ import { Hono } from "hono"
2
+ import { describeRoute, validator } from "hono-openapi"
3
+ import { resolver } from "hono-openapi"
4
+ import { Instance } from "../../project/instance"
5
+ import { Project } from "../../project/project"
6
+ import z from "zod"
7
+ import { errors } from "../error"
8
+ import { lazy } from "../../util/lazy"
9
+
10
+ export const ProjectRoutes = lazy(() =>
11
+ new Hono()
12
+ .get(
13
+ "/",
14
+ describeRoute({
15
+ summary: "List all projects",
16
+ description: "Get a list of projects that have been opened with BloxyCode.",
17
+ operationId: "project.list",
18
+ responses: {
19
+ 200: {
20
+ description: "List of projects",
21
+ content: {
22
+ "application/json": {
23
+ schema: resolver(Project.Info.array()),
24
+ },
25
+ },
26
+ },
27
+ },
28
+ }),
29
+ async (c) => {
30
+ const projects = await Project.list()
31
+ return c.json(projects)
32
+ },
33
+ )
34
+ .get(
35
+ "/current",
36
+ describeRoute({
37
+ summary: "Get current project",
38
+ description: "Retrieve the currently active project that BloxyCode is working with.",
39
+ operationId: "project.current",
40
+ responses: {
41
+ 200: {
42
+ description: "Current project information",
43
+ content: {
44
+ "application/json": {
45
+ schema: resolver(Project.Info),
46
+ },
47
+ },
48
+ },
49
+ },
50
+ }),
51
+ async (c) => {
52
+ return c.json(Instance.project)
53
+ },
54
+ )
55
+ .patch(
56
+ "/:projectID",
57
+ describeRoute({
58
+ summary: "Update project",
59
+ description: "Update project properties such as name, icon, and commands.",
60
+ operationId: "project.update",
61
+ responses: {
62
+ 200: {
63
+ description: "Updated project information",
64
+ content: {
65
+ "application/json": {
66
+ schema: resolver(Project.Info),
67
+ },
68
+ },
69
+ },
70
+ ...errors(400, 404),
71
+ },
72
+ }),
73
+ validator("param", z.object({ projectID: z.string() })),
74
+ validator("json", Project.update.schema.omit({ projectID: true })),
75
+ async (c) => {
76
+ const projectID = c.req.valid("param").projectID
77
+ const body = c.req.valid("json")
78
+ const project = await Project.update({ ...body, projectID })
79
+ return c.json(project)
80
+ },
81
+ ),
82
+ )
@@ -0,0 +1,165 @@
1
+ import { Hono } from "hono"
2
+ import { describeRoute, validator, resolver } from "hono-openapi"
3
+ import z from "zod"
4
+ import { Config } from "../../config/config"
5
+ import { Provider } from "../../provider/provider"
6
+ import { ModelsDev } from "../../provider/models"
7
+ import { ProviderAuth } from "../../provider/auth"
8
+ import { mapValues } from "remeda"
9
+ import { errors } from "../error"
10
+ import { lazy } from "../../util/lazy"
11
+
12
+ export const ProviderRoutes = lazy(() =>
13
+ new Hono()
14
+ .get(
15
+ "/",
16
+ describeRoute({
17
+ summary: "List providers",
18
+ description: "Get a list of all available AI providers, including both available and connected ones.",
19
+ operationId: "provider.list",
20
+ responses: {
21
+ 200: {
22
+ description: "List of providers",
23
+ content: {
24
+ "application/json": {
25
+ schema: resolver(
26
+ z.object({
27
+ all: ModelsDev.Provider.array(),
28
+ default: z.record(z.string(), z.string()),
29
+ connected: z.array(z.string()),
30
+ }),
31
+ ),
32
+ },
33
+ },
34
+ },
35
+ },
36
+ }),
37
+ async (c) => {
38
+ const config = await Config.get()
39
+ const disabled = new Set(config.disabled_providers ?? [])
40
+ const enabled = config.enabled_providers ? new Set(config.enabled_providers) : undefined
41
+
42
+ const allProviders = await ModelsDev.get()
43
+ const filteredProviders: Record<string, (typeof allProviders)[string]> = {}
44
+ for (const [key, value] of Object.entries(allProviders)) {
45
+ if ((enabled ? enabled.has(key) : true) && !disabled.has(key)) {
46
+ filteredProviders[key] = value
47
+ }
48
+ }
49
+
50
+ const connected = await Provider.list()
51
+ const providers = Object.assign(
52
+ mapValues(filteredProviders, (x) => Provider.fromModelsDevProvider(x)),
53
+ connected,
54
+ )
55
+ return c.json({
56
+ all: Object.values(providers),
57
+ default: mapValues(providers, (item) => Provider.sort(Object.values(item.models))[0].id),
58
+ connected: Object.keys(connected),
59
+ })
60
+ },
61
+ )
62
+ .get(
63
+ "/auth",
64
+ describeRoute({
65
+ summary: "Get provider auth methods",
66
+ description: "Retrieve available authentication methods for all AI providers.",
67
+ operationId: "provider.auth",
68
+ responses: {
69
+ 200: {
70
+ description: "Provider auth methods",
71
+ content: {
72
+ "application/json": {
73
+ schema: resolver(z.record(z.string(), z.array(ProviderAuth.Method))),
74
+ },
75
+ },
76
+ },
77
+ },
78
+ }),
79
+ async (c) => {
80
+ return c.json(await ProviderAuth.methods())
81
+ },
82
+ )
83
+ .post(
84
+ "/:providerID/oauth/authorize",
85
+ describeRoute({
86
+ summary: "OAuth authorize",
87
+ description: "Initiate OAuth authorization for a specific AI provider to get an authorization URL.",
88
+ operationId: "provider.oauth.authorize",
89
+ responses: {
90
+ 200: {
91
+ description: "Authorization URL and method",
92
+ content: {
93
+ "application/json": {
94
+ schema: resolver(ProviderAuth.Authorization.optional()),
95
+ },
96
+ },
97
+ },
98
+ ...errors(400),
99
+ },
100
+ }),
101
+ validator(
102
+ "param",
103
+ z.object({
104
+ providerID: z.string().meta({ description: "Provider ID" }),
105
+ }),
106
+ ),
107
+ validator(
108
+ "json",
109
+ z.object({
110
+ method: z.number().meta({ description: "Auth method index" }),
111
+ }),
112
+ ),
113
+ async (c) => {
114
+ const providerID = c.req.valid("param").providerID
115
+ const { method } = c.req.valid("json")
116
+ const result = await ProviderAuth.authorize({
117
+ providerID,
118
+ method,
119
+ })
120
+ return c.json(result)
121
+ },
122
+ )
123
+ .post(
124
+ "/:providerID/oauth/callback",
125
+ describeRoute({
126
+ summary: "OAuth callback",
127
+ description: "Handle the OAuth callback from a provider after user authorization.",
128
+ operationId: "provider.oauth.callback",
129
+ responses: {
130
+ 200: {
131
+ description: "OAuth callback processed successfully",
132
+ content: {
133
+ "application/json": {
134
+ schema: resolver(z.boolean()),
135
+ },
136
+ },
137
+ },
138
+ ...errors(400),
139
+ },
140
+ }),
141
+ validator(
142
+ "param",
143
+ z.object({
144
+ providerID: z.string().meta({ description: "Provider ID" }),
145
+ }),
146
+ ),
147
+ validator(
148
+ "json",
149
+ z.object({
150
+ method: z.number().meta({ description: "Auth method index" }),
151
+ code: z.string().optional().meta({ description: "OAuth authorization code" }),
152
+ }),
153
+ ),
154
+ async (c) => {
155
+ const providerID = c.req.valid("param").providerID
156
+ const { method, code } = c.req.valid("json")
157
+ await ProviderAuth.callback({
158
+ providerID,
159
+ method,
160
+ code,
161
+ })
162
+ return c.json(true)
163
+ },
164
+ ),
165
+ )
@@ -0,0 +1,169 @@
1
+ import { Hono } from "hono"
2
+ import { describeRoute, validator, resolver } from "hono-openapi"
3
+ import { upgradeWebSocket } from "hono/bun"
4
+ import z from "zod"
5
+ import { Pty } from "@/pty"
6
+ import { Storage } from "../../storage/storage"
7
+ import { errors } from "../error"
8
+ import { lazy } from "../../util/lazy"
9
+
10
+ export const PtyRoutes = lazy(() =>
11
+ new Hono()
12
+ .get(
13
+ "/",
14
+ describeRoute({
15
+ summary: "List PTY sessions",
16
+ description: "Get a list of all active pseudo-terminal (PTY) sessions managed by BloxyCode.",
17
+ operationId: "pty.list",
18
+ responses: {
19
+ 200: {
20
+ description: "List of sessions",
21
+ content: {
22
+ "application/json": {
23
+ schema: resolver(Pty.Info.array()),
24
+ },
25
+ },
26
+ },
27
+ },
28
+ }),
29
+ async (c) => {
30
+ return c.json(Pty.list())
31
+ },
32
+ )
33
+ .post(
34
+ "/",
35
+ describeRoute({
36
+ summary: "Create PTY session",
37
+ description: "Create a new pseudo-terminal (PTY) session for running shell commands and processes.",
38
+ operationId: "pty.create",
39
+ responses: {
40
+ 200: {
41
+ description: "Created session",
42
+ content: {
43
+ "application/json": {
44
+ schema: resolver(Pty.Info),
45
+ },
46
+ },
47
+ },
48
+ ...errors(400),
49
+ },
50
+ }),
51
+ validator("json", Pty.CreateInput),
52
+ async (c) => {
53
+ const info = await Pty.create(c.req.valid("json"))
54
+ return c.json(info)
55
+ },
56
+ )
57
+ .get(
58
+ "/:ptyID",
59
+ describeRoute({
60
+ summary: "Get PTY session",
61
+ description: "Retrieve detailed information about a specific pseudo-terminal (PTY) session.",
62
+ operationId: "pty.get",
63
+ responses: {
64
+ 200: {
65
+ description: "Session info",
66
+ content: {
67
+ "application/json": {
68
+ schema: resolver(Pty.Info),
69
+ },
70
+ },
71
+ },
72
+ ...errors(404),
73
+ },
74
+ }),
75
+ validator("param", z.object({ ptyID: z.string() })),
76
+ async (c) => {
77
+ const info = Pty.get(c.req.valid("param").ptyID)
78
+ if (!info) {
79
+ throw new Storage.NotFoundError({ message: "Session not found" })
80
+ }
81
+ return c.json(info)
82
+ },
83
+ )
84
+ .put(
85
+ "/:ptyID",
86
+ describeRoute({
87
+ summary: "Update PTY session",
88
+ description: "Update properties of an existing pseudo-terminal (PTY) session.",
89
+ operationId: "pty.update",
90
+ responses: {
91
+ 200: {
92
+ description: "Updated session",
93
+ content: {
94
+ "application/json": {
95
+ schema: resolver(Pty.Info),
96
+ },
97
+ },
98
+ },
99
+ ...errors(400),
100
+ },
101
+ }),
102
+ validator("param", z.object({ ptyID: z.string() })),
103
+ validator("json", Pty.UpdateInput),
104
+ async (c) => {
105
+ const info = await Pty.update(c.req.valid("param").ptyID, c.req.valid("json"))
106
+ return c.json(info)
107
+ },
108
+ )
109
+ .delete(
110
+ "/:ptyID",
111
+ describeRoute({
112
+ summary: "Remove PTY session",
113
+ description: "Remove and terminate a specific pseudo-terminal (PTY) session.",
114
+ operationId: "pty.remove",
115
+ responses: {
116
+ 200: {
117
+ description: "Session removed",
118
+ content: {
119
+ "application/json": {
120
+ schema: resolver(z.boolean()),
121
+ },
122
+ },
123
+ },
124
+ ...errors(404),
125
+ },
126
+ }),
127
+ validator("param", z.object({ ptyID: z.string() })),
128
+ async (c) => {
129
+ await Pty.remove(c.req.valid("param").ptyID)
130
+ return c.json(true)
131
+ },
132
+ )
133
+ .get(
134
+ "/:ptyID/connect",
135
+ describeRoute({
136
+ summary: "Connect to PTY session",
137
+ description: "Establish a WebSocket connection to interact with a pseudo-terminal (PTY) session in real-time.",
138
+ operationId: "pty.connect",
139
+ responses: {
140
+ 200: {
141
+ description: "Connected session",
142
+ content: {
143
+ "application/json": {
144
+ schema: resolver(z.boolean()),
145
+ },
146
+ },
147
+ },
148
+ ...errors(404),
149
+ },
150
+ }),
151
+ validator("param", z.object({ ptyID: z.string() })),
152
+ upgradeWebSocket((c) => {
153
+ const id = c.req.param("ptyID")
154
+ let handler: ReturnType<typeof Pty.connect>
155
+ if (!Pty.get(id)) throw new Error("Session not found")
156
+ return {
157
+ onOpen(_event, ws) {
158
+ handler = Pty.connect(id, ws)
159
+ },
160
+ onMessage(event) {
161
+ handler?.onMessage(String(event.data))
162
+ },
163
+ onClose() {
164
+ handler?.onClose()
165
+ },
166
+ }
167
+ }),
168
+ ),
169
+ )