koro-ai 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 (670) hide show
  1. package/AGENTS.md +69 -0
  2. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +15 -0
  5. package/bin/koro +16 -0
  6. package/bin/opencode +179 -0
  7. package/bunfig.toml +7 -0
  8. package/drizzle.config.ts +10 -0
  9. package/git +0 -0
  10. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  11. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  12. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  13. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  14. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  15. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  16. package/migration/20260225215848_workspace/migration.sql +7 -0
  17. package/migration/20260225215848_workspace/snapshot.json +959 -0
  18. package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
  19. package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
  20. package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
  21. package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
  22. package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
  23. package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
  24. package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
  25. package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
  26. package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
  27. package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
  28. package/migration/20260323234822_events/migration.sql +13 -0
  29. package/migration/20260323234822_events/snapshot.json +1271 -0
  30. package/package.json +163 -0
  31. package/parsers-config.ts +290 -0
  32. package/script/build-node.ts +56 -0
  33. package/script/build.ts +276 -0
  34. package/script/check-migrations.ts +16 -0
  35. package/script/postinstall.mjs +131 -0
  36. package/script/publish.ts +181 -0
  37. package/script/schema.ts +63 -0
  38. package/script/seed-e2e.ts +60 -0
  39. package/script/upgrade-opentui.ts +64 -0
  40. package/specs/effect-migration.md +294 -0
  41. package/specs/tui-plugins.md +436 -0
  42. package/src/account/account.sql.ts +39 -0
  43. package/src/account/index.ts +424 -0
  44. package/src/account/repo.ts +163 -0
  45. package/src/account/schema.ts +91 -0
  46. package/src/acp/README.md +174 -0
  47. package/src/acp/agent.ts +1763 -0
  48. package/src/acp/session.ts +116 -0
  49. package/src/acp/types.ts +24 -0
  50. package/src/agent/agent.ts +476 -0
  51. package/src/agent/generate.txt +75 -0
  52. package/src/agent/prompt/compaction.txt +15 -0
  53. package/src/agent/prompt/explore.txt +18 -0
  54. package/src/agent/prompt/summary.txt +11 -0
  55. package/src/agent/prompt/title.txt +44 -0
  56. package/src/auth/index.ts +109 -0
  57. package/src/bus/bus-event.ts +40 -0
  58. package/src/bus/global.ts +10 -0
  59. package/src/bus/index.ts +185 -0
  60. package/src/cli/bootstrap.ts +17 -0
  61. package/src/cli/cmd/account.ts +257 -0
  62. package/src/cli/cmd/acp.ts +70 -0
  63. package/src/cli/cmd/agent.ts +245 -0
  64. package/src/cli/cmd/cmd.ts +7 -0
  65. package/src/cli/cmd/db.ts +119 -0
  66. package/src/cli/cmd/debug/agent.ts +167 -0
  67. package/src/cli/cmd/debug/config.ts +16 -0
  68. package/src/cli/cmd/debug/file.ts +97 -0
  69. package/src/cli/cmd/debug/index.ts +48 -0
  70. package/src/cli/cmd/debug/lsp.ts +53 -0
  71. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  72. package/src/cli/cmd/debug/scrap.ts +16 -0
  73. package/src/cli/cmd/debug/skill.ts +16 -0
  74. package/src/cli/cmd/debug/snapshot.ts +52 -0
  75. package/src/cli/cmd/export.ts +89 -0
  76. package/src/cli/cmd/generate.ts +38 -0
  77. package/src/cli/cmd/github.ts +1646 -0
  78. package/src/cli/cmd/import.ts +207 -0
  79. package/src/cli/cmd/mcp.ts +754 -0
  80. package/src/cli/cmd/models.ts +78 -0
  81. package/src/cli/cmd/plug.ts +233 -0
  82. package/src/cli/cmd/pr.ts +127 -0
  83. package/src/cli/cmd/providers.ts +478 -0
  84. package/src/cli/cmd/run.ts +676 -0
  85. package/src/cli/cmd/serve.ts +24 -0
  86. package/src/cli/cmd/session.ts +159 -0
  87. package/src/cli/cmd/stats.ts +410 -0
  88. package/src/cli/cmd/tui/app.tsx +919 -0
  89. package/src/cli/cmd/tui/attach.ts +88 -0
  90. package/src/cli/cmd/tui/component/border.tsx +21 -0
  91. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  92. package/src/cli/cmd/tui/component/dialog-command.tsx +171 -0
  93. package/src/cli/cmd/tui/component/dialog-import-share.tsx +118 -0
  94. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  95. package/src/cli/cmd/tui/component/dialog-model.tsx +179 -0
  96. package/src/cli/cmd/tui/component/dialog-provider.tsx +329 -0
  97. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  98. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  99. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  100. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  101. package/src/cli/cmd/tui/component/dialog-status.tsx +168 -0
  102. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  103. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  104. package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
  105. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +320 -0
  106. package/src/cli/cmd/tui/component/error-component.tsx +92 -0
  107. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  108. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  109. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +672 -0
  110. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  111. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  112. package/src/cli/cmd/tui/component/prompt/index.tsx +1310 -0
  113. package/src/cli/cmd/tui/component/prompt/part.ts +16 -0
  114. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  115. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  116. package/src/cli/cmd/tui/component/startup-loading.tsx +63 -0
  117. package/src/cli/cmd/tui/component/task-panel.tsx +44 -0
  118. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  119. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  120. package/src/cli/cmd/tui/component/token-bar.tsx +60 -0
  121. package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +151 -0
  122. package/src/cli/cmd/tui/context/args.tsx +15 -0
  123. package/src/cli/cmd/tui/context/directory.ts +13 -0
  124. package/src/cli/cmd/tui/context/exit.tsx +60 -0
  125. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  126. package/src/cli/cmd/tui/context/keybind.tsx +105 -0
  127. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  128. package/src/cli/cmd/tui/context/local.tsx +412 -0
  129. package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
  130. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  131. package/src/cli/cmd/tui/context/route.tsx +52 -0
  132. package/src/cli/cmd/tui/context/sdk.tsx +128 -0
  133. package/src/cli/cmd/tui/context/sync.tsx +504 -0
  134. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  135. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  136. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  137. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  138. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  139. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  140. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  141. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  142. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  143. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  144. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  145. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  146. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  147. package/src/cli/cmd/tui/context/theme/gtr.json +245 -0
  148. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  149. package/src/cli/cmd/tui/context/theme/koro.json +241 -0
  150. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  151. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  152. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  153. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  154. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  155. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  156. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  157. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  158. package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
  159. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  160. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  161. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  162. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  163. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  164. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  165. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  166. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  167. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  168. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  169. package/src/cli/cmd/tui/context/theme.tsx +1240 -0
  170. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  171. package/src/cli/cmd/tui/event.ts +49 -0
  172. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
  173. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +152 -0
  174. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
  175. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +63 -0
  176. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
  177. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
  178. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
  179. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +96 -0
  180. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +48 -0
  181. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +270 -0
  182. package/src/cli/cmd/tui/plugin/api.tsx +430 -0
  183. package/src/cli/cmd/tui/plugin/index.ts +3 -0
  184. package/src/cli/cmd/tui/plugin/internal.ts +27 -0
  185. package/src/cli/cmd/tui/plugin/runtime.ts +1033 -0
  186. package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
  187. package/src/cli/cmd/tui/routes/home.tsx +84 -0
  188. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +65 -0
  189. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +110 -0
  190. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  191. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  192. package/src/cli/cmd/tui/routes/session/footer.tsx +93 -0
  193. package/src/cli/cmd/tui/routes/session/index.tsx +2270 -0
  194. package/src/cli/cmd/tui/routes/session/permission.tsx +691 -0
  195. package/src/cli/cmd/tui/routes/session/question.tsx +468 -0
  196. package/src/cli/cmd/tui/routes/session/sidebar.tsx +74 -0
  197. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +131 -0
  198. package/src/cli/cmd/tui/thread.ts +232 -0
  199. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  200. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +89 -0
  201. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +211 -0
  202. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  203. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +130 -0
  204. package/src/cli/cmd/tui/ui/dialog-select.tsx +409 -0
  205. package/src/cli/cmd/tui/ui/dialog.tsx +192 -0
  206. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  207. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  208. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  209. package/src/cli/cmd/tui/util/clipboard.ts +192 -0
  210. package/src/cli/cmd/tui/util/editor.ts +37 -0
  211. package/src/cli/cmd/tui/util/model.ts +23 -0
  212. package/src/cli/cmd/tui/util/scroll.ts +23 -0
  213. package/src/cli/cmd/tui/util/selection.ts +25 -0
  214. package/src/cli/cmd/tui/util/signal.ts +7 -0
  215. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  216. package/src/cli/cmd/tui/util/transcript.ts +112 -0
  217. package/src/cli/cmd/tui/win32.ts +129 -0
  218. package/src/cli/cmd/tui/worker.ts +175 -0
  219. package/src/cli/cmd/uninstall.ts +353 -0
  220. package/src/cli/cmd/upgrade.ts +73 -0
  221. package/src/cli/cmd/web.ts +81 -0
  222. package/src/cli/effect/prompt.ts +25 -0
  223. package/src/cli/error.ts +46 -0
  224. package/src/cli/heap.ts +59 -0
  225. package/src/cli/logo.ts +14 -0
  226. package/src/cli/network.ts +60 -0
  227. package/src/cli/ui.ts +133 -0
  228. package/src/cli/upgrade.ts +31 -0
  229. package/src/command/index.ts +195 -0
  230. package/src/command/template/initialize.txt +66 -0
  231. package/src/command/template/review.txt +101 -0
  232. package/src/config/config.ts +1591 -0
  233. package/src/config/markdown.ts +99 -0
  234. package/src/config/paths.ts +181 -0
  235. package/src/config/tui-migrate.ts +155 -0
  236. package/src/config/tui-schema.ts +36 -0
  237. package/src/config/tui.ts +171 -0
  238. package/src/control-plane/adaptors/index.ts +20 -0
  239. package/src/control-plane/adaptors/worktree.ts +38 -0
  240. package/src/control-plane/schema.ts +17 -0
  241. package/src/control-plane/sse.ts +66 -0
  242. package/src/control-plane/types.ts +21 -0
  243. package/src/control-plane/workspace.sql.ts +17 -0
  244. package/src/control-plane/workspace.ts +154 -0
  245. package/src/effect/cross-spawn-spawner.ts +502 -0
  246. package/src/effect/instance-ref.ts +6 -0
  247. package/src/effect/instance-registry.ts +12 -0
  248. package/src/effect/instance-state.ts +82 -0
  249. package/src/effect/run-service.ts +33 -0
  250. package/src/effect/runner.ts +216 -0
  251. package/src/env/index.ts +28 -0
  252. package/src/file/ignore.ts +82 -0
  253. package/src/file/index.ts +686 -0
  254. package/src/file/protected.ts +59 -0
  255. package/src/file/ripgrep.ts +376 -0
  256. package/src/file/time.ts +133 -0
  257. package/src/file/watcher.ts +171 -0
  258. package/src/filesystem/index.ts +226 -0
  259. package/src/flag/flag.ts +155 -0
  260. package/src/format/formatter.ts +413 -0
  261. package/src/format/index.ts +203 -0
  262. package/src/git/index.ts +303 -0
  263. package/src/global/index.ts +161 -0
  264. package/src/id/id.ts +85 -0
  265. package/src/ide/index.ts +74 -0
  266. package/src/index.ts +240 -0
  267. package/src/installation/index.ts +355 -0
  268. package/src/installation/meta.ts +7 -0
  269. package/src/lsp/client.ts +252 -0
  270. package/src/lsp/index.ts +558 -0
  271. package/src/lsp/language.ts +120 -0
  272. package/src/lsp/launch.ts +21 -0
  273. package/src/lsp/server.ts +1958 -0
  274. package/src/mcp/auth.ts +173 -0
  275. package/src/mcp/index.ts +921 -0
  276. package/src/mcp/oauth-callback.ts +215 -0
  277. package/src/mcp/oauth-provider.ts +185 -0
  278. package/src/memory/index.ts +117 -0
  279. package/src/node.ts +1 -0
  280. package/src/npm/index.ts +180 -0
  281. package/src/orchestrator/agent-registry.ts +38 -0
  282. package/src/orchestrator/conflict.ts +25 -0
  283. package/src/orchestrator/context-manager.ts +22 -0
  284. package/src/orchestrator/index.ts +9 -0
  285. package/src/orchestrator/scheduler.ts +30 -0
  286. package/src/orchestrator/state-tracker.ts +71 -0
  287. package/src/orchestrator/task-manager.ts +69 -0
  288. package/src/patch/index.ts +680 -0
  289. package/src/permission/arity.ts +163 -0
  290. package/src/permission/evaluate.ts +15 -0
  291. package/src/permission/index.ts +325 -0
  292. package/src/permission/schema.ts +17 -0
  293. package/src/plugin/codex.ts +596 -0
  294. package/src/plugin/github-copilot/copilot.ts +353 -0
  295. package/src/plugin/github-copilot/models.ts +144 -0
  296. package/src/plugin/index.ts +281 -0
  297. package/src/plugin/install.ts +439 -0
  298. package/src/plugin/loader.ts +174 -0
  299. package/src/plugin/meta.ts +188 -0
  300. package/src/plugin/shared.ts +307 -0
  301. package/src/project/bootstrap.ts +31 -0
  302. package/src/project/instance.ts +175 -0
  303. package/src/project/project.sql.ts +16 -0
  304. package/src/project/project.ts +519 -0
  305. package/src/project/schema.ts +16 -0
  306. package/src/project/state.ts +70 -0
  307. package/src/project/vcs.ts +240 -0
  308. package/src/provider/auth.ts +253 -0
  309. package/src/provider/error.ts +197 -0
  310. package/src/provider/models-snapshot.ts +60410 -0
  311. package/src/provider/models.ts +162 -0
  312. package/src/provider/provider.ts +1677 -0
  313. package/src/provider/schema.ts +38 -0
  314. package/src/provider/sdk/copilot/README.md +5 -0
  315. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -0
  316. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  317. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +19 -0
  318. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  319. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +815 -0
  320. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  321. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  322. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +83 -0
  323. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  324. package/src/provider/sdk/copilot/index.ts +2 -0
  325. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  326. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +335 -0
  327. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  328. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  329. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  330. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +214 -0
  331. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1769 -0
  332. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
  333. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  334. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
  335. package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
  336. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
  337. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
  338. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
  339. package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
  340. package/src/provider/transform.ts +1046 -0
  341. package/src/pty/index.ts +401 -0
  342. package/src/pty/schema.ts +17 -0
  343. package/src/question/index.ts +224 -0
  344. package/src/question/schema.ts +17 -0
  345. package/src/server/error.ts +36 -0
  346. package/src/server/event.ts +7 -0
  347. package/src/server/instance.ts +314 -0
  348. package/src/server/mdns.ts +60 -0
  349. package/src/server/middleware.ts +33 -0
  350. package/src/server/projectors.ts +28 -0
  351. package/src/server/router.ts +99 -0
  352. package/src/server/routes/config.ts +92 -0
  353. package/src/server/routes/event.ts +83 -0
  354. package/src/server/routes/experimental.ts +271 -0
  355. package/src/server/routes/file.ts +197 -0
  356. package/src/server/routes/global.ts +312 -0
  357. package/src/server/routes/mcp.ts +225 -0
  358. package/src/server/routes/permission.ts +69 -0
  359. package/src/server/routes/project.ts +118 -0
  360. package/src/server/routes/provider.ts +171 -0
  361. package/src/server/routes/pty.ts +211 -0
  362. package/src/server/routes/question.ts +99 -0
  363. package/src/server/routes/session.ts +1031 -0
  364. package/src/server/routes/tui.ts +379 -0
  365. package/src/server/routes/workspace.ts +94 -0
  366. package/src/server/server.ts +312 -0
  367. package/src/session/compaction.ts +428 -0
  368. package/src/session/index.ts +887 -0
  369. package/src/session/instruction.ts +258 -0
  370. package/src/session/llm.ts +370 -0
  371. package/src/session/message-v2.ts +1031 -0
  372. package/src/session/message.ts +191 -0
  373. package/src/session/overflow.ts +22 -0
  374. package/src/session/processor.ts +523 -0
  375. package/src/session/projectors.ts +135 -0
  376. package/src/session/prompt/anthropic.txt +105 -0
  377. package/src/session/prompt/beast.txt +147 -0
  378. package/src/session/prompt/build-switch.txt +5 -0
  379. package/src/session/prompt/codex.txt +79 -0
  380. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  381. package/src/session/prompt/default.txt +105 -0
  382. package/src/session/prompt/gemini.txt +155 -0
  383. package/src/session/prompt/gpt.txt +107 -0
  384. package/src/session/prompt/kimi.txt +114 -0
  385. package/src/session/prompt/max-steps.txt +16 -0
  386. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  387. package/src/session/prompt/plan.txt +26 -0
  388. package/src/session/prompt/trinity.txt +97 -0
  389. package/src/session/prompt.ts +1908 -0
  390. package/src/session/retry.ts +106 -0
  391. package/src/session/revert.ts +176 -0
  392. package/src/session/schema.ts +38 -0
  393. package/src/session/session.sql.ts +103 -0
  394. package/src/session/status.ts +102 -0
  395. package/src/session/summary.ts +177 -0
  396. package/src/session/system.ts +76 -0
  397. package/src/session/todo.ts +95 -0
  398. package/src/share/share-next.ts +369 -0
  399. package/src/share/share.sql.ts +13 -0
  400. package/src/shell/shell.ts +110 -0
  401. package/src/skill/discovery.ts +116 -0
  402. package/src/skill/index.ts +277 -0
  403. package/src/snapshot/index.ts +571 -0
  404. package/src/sql.d.ts +4 -0
  405. package/src/storage/db.bun.ts +8 -0
  406. package/src/storage/db.node.ts +8 -0
  407. package/src/storage/db.ts +174 -0
  408. package/src/storage/json-migration.ts +425 -0
  409. package/src/storage/schema.sql.ts +10 -0
  410. package/src/storage/schema.ts +5 -0
  411. package/src/storage/storage.ts +353 -0
  412. package/src/sync/README.md +179 -0
  413. package/src/sync/event.sql.ts +16 -0
  414. package/src/sync/index.ts +263 -0
  415. package/src/sync/schema.ts +14 -0
  416. package/src/token/index.ts +77 -0
  417. package/src/tool/apply_patch.ts +281 -0
  418. package/src/tool/apply_patch.txt +33 -0
  419. package/src/tool/bash.ts +496 -0
  420. package/src/tool/bash.txt +117 -0
  421. package/src/tool/batch.ts +183 -0
  422. package/src/tool/batch.txt +24 -0
  423. package/src/tool/codesearch.ts +132 -0
  424. package/src/tool/codesearch.txt +12 -0
  425. package/src/tool/edit.ts +667 -0
  426. package/src/tool/edit.txt +10 -0
  427. package/src/tool/external-directory.ts +37 -0
  428. package/src/tool/glob.ts +78 -0
  429. package/src/tool/glob.txt +6 -0
  430. package/src/tool/grep.ts +156 -0
  431. package/src/tool/grep.txt +8 -0
  432. package/src/tool/invalid.ts +17 -0
  433. package/src/tool/ls.ts +121 -0
  434. package/src/tool/ls.txt +1 -0
  435. package/src/tool/lsp.ts +97 -0
  436. package/src/tool/lsp.txt +19 -0
  437. package/src/tool/multiedit.ts +46 -0
  438. package/src/tool/multiedit.txt +41 -0
  439. package/src/tool/plan-enter.txt +14 -0
  440. package/src/tool/plan-exit.txt +13 -0
  441. package/src/tool/plan.ts +131 -0
  442. package/src/tool/question.ts +46 -0
  443. package/src/tool/question.txt +10 -0
  444. package/src/tool/read.ts +296 -0
  445. package/src/tool/read.txt +14 -0
  446. package/src/tool/registry.ts +248 -0
  447. package/src/tool/schema.ts +17 -0
  448. package/src/tool/skill.ts +105 -0
  449. package/src/tool/task.ts +166 -0
  450. package/src/tool/task.txt +60 -0
  451. package/src/tool/todo.ts +48 -0
  452. package/src/tool/todowrite.txt +167 -0
  453. package/src/tool/tool.ts +112 -0
  454. package/src/tool/truncate.ts +144 -0
  455. package/src/tool/truncation-dir.ts +4 -0
  456. package/src/tool/webfetch.ts +206 -0
  457. package/src/tool/webfetch.txt +13 -0
  458. package/src/tool/websearch.ts +150 -0
  459. package/src/tool/websearch.txt +14 -0
  460. package/src/tool/write.ts +84 -0
  461. package/src/tool/write.txt +8 -0
  462. package/src/util/abort.ts +35 -0
  463. package/src/util/archive.ts +17 -0
  464. package/src/util/color.ts +19 -0
  465. package/src/util/context.ts +25 -0
  466. package/src/util/data-url.ts +9 -0
  467. package/src/util/defer.ts +12 -0
  468. package/src/util/effect-http-client.ts +11 -0
  469. package/src/util/effect-zod.ts +98 -0
  470. package/src/util/error.ts +77 -0
  471. package/src/util/filesystem.ts +245 -0
  472. package/src/util/flock.ts +333 -0
  473. package/src/util/fn.ts +21 -0
  474. package/src/util/format.ts +20 -0
  475. package/src/util/glob.ts +34 -0
  476. package/src/util/hash.ts +7 -0
  477. package/src/util/iife.ts +3 -0
  478. package/src/util/keybind.ts +103 -0
  479. package/src/util/lazy.ts +23 -0
  480. package/src/util/locale.ts +81 -0
  481. package/src/util/lock.ts +98 -0
  482. package/src/util/log.ts +182 -0
  483. package/src/util/network.ts +9 -0
  484. package/src/util/process.ts +176 -0
  485. package/src/util/queue.ts +32 -0
  486. package/src/util/record.ts +3 -0
  487. package/src/util/rpc.ts +66 -0
  488. package/src/util/schema.ts +53 -0
  489. package/src/util/scrap.ts +10 -0
  490. package/src/util/signal.ts +12 -0
  491. package/src/util/timeout.ts +14 -0
  492. package/src/util/token.ts +7 -0
  493. package/src/util/update-schema.ts +13 -0
  494. package/src/util/which.ts +14 -0
  495. package/src/util/wildcard.ts +59 -0
  496. package/src/worktree/index.ts +612 -0
  497. package/sst-env.d.ts +10 -0
  498. package/test/AGENTS.md +81 -0
  499. package/test/account/repo.test.ts +326 -0
  500. package/test/account/service.test.ts +393 -0
  501. package/test/acp/agent-interface.test.ts +51 -0
  502. package/test/acp/event-subscription.test.ts +685 -0
  503. package/test/agent/agent.test.ts +717 -0
  504. package/test/auth/auth.test.ts +58 -0
  505. package/test/bus/bus-effect.test.ts +164 -0
  506. package/test/bus/bus-integration.test.ts +87 -0
  507. package/test/bus/bus.test.ts +219 -0
  508. package/test/cli/account.test.ts +26 -0
  509. package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
  510. package/test/cli/github-action.test.ts +198 -0
  511. package/test/cli/github-remote.test.ts +80 -0
  512. package/test/cli/import.test.ts +54 -0
  513. package/test/cli/plugin-auth-picker.test.ts +120 -0
  514. package/test/cli/tui/keybind-plugin.test.ts +90 -0
  515. package/test/cli/tui/plugin-add.test.ts +107 -0
  516. package/test/cli/tui/plugin-install.test.ts +89 -0
  517. package/test/cli/tui/plugin-lifecycle.test.ts +225 -0
  518. package/test/cli/tui/plugin-loader-entrypoint.test.ts +492 -0
  519. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  520. package/test/cli/tui/plugin-loader.test.ts +752 -0
  521. package/test/cli/tui/plugin-toggle.test.ts +159 -0
  522. package/test/cli/tui/slot-replace.test.tsx +47 -0
  523. package/test/cli/tui/theme-store.test.ts +51 -0
  524. package/test/cli/tui/thread.test.ts +128 -0
  525. package/test/cli/tui/transcript.test.ts +426 -0
  526. package/test/config/agent-color.test.ts +71 -0
  527. package/test/config/config.test.ts +2348 -0
  528. package/test/config/fixtures/empty-frontmatter.md +4 -0
  529. package/test/config/fixtures/frontmatter.md +28 -0
  530. package/test/config/fixtures/markdown-header.md +11 -0
  531. package/test/config/fixtures/no-frontmatter.md +1 -0
  532. package/test/config/fixtures/weird-model-id.md +13 -0
  533. package/test/config/markdown.test.ts +228 -0
  534. package/test/config/tui.test.ts +752 -0
  535. package/test/control-plane/sse.test.ts +56 -0
  536. package/test/effect/cross-spawn-spawner.test.ts +412 -0
  537. package/test/effect/instance-state.test.ts +482 -0
  538. package/test/effect/run-service.test.ts +46 -0
  539. package/test/effect/runner.test.ts +523 -0
  540. package/test/fake/provider.ts +81 -0
  541. package/test/file/fsmonitor.test.ts +62 -0
  542. package/test/file/ignore.test.ts +10 -0
  543. package/test/file/index.test.ts +946 -0
  544. package/test/file/path-traversal.test.ts +198 -0
  545. package/test/file/ripgrep.test.ts +54 -0
  546. package/test/file/time.test.ts +445 -0
  547. package/test/file/watcher.test.ts +247 -0
  548. package/test/filesystem/filesystem.test.ts +319 -0
  549. package/test/fixture/db.ts +11 -0
  550. package/test/fixture/fixture.test.ts +26 -0
  551. package/test/fixture/fixture.ts +172 -0
  552. package/test/fixture/flock-worker.ts +72 -0
  553. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  554. package/test/fixture/plug-worker.ts +93 -0
  555. package/test/fixture/plugin-meta-worker.ts +26 -0
  556. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  557. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  558. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  559. package/test/fixture/skills/index.json +6 -0
  560. package/test/fixture/tui-plugin.ts +337 -0
  561. package/test/fixture/tui-runtime.ts +27 -0
  562. package/test/format/format.test.ts +171 -0
  563. package/test/git/git.test.ts +128 -0
  564. package/test/ide/ide.test.ts +82 -0
  565. package/test/installation/installation.test.ts +151 -0
  566. package/test/keybind.test.ts +421 -0
  567. package/test/lib/effect.ts +53 -0
  568. package/test/lib/filesystem.ts +10 -0
  569. package/test/lib/llm-server.ts +795 -0
  570. package/test/lsp/client.test.ts +95 -0
  571. package/test/lsp/index.test.ts +55 -0
  572. package/test/lsp/launch.test.ts +22 -0
  573. package/test/lsp/lifecycle.test.ts +147 -0
  574. package/test/mcp/headers.test.ts +153 -0
  575. package/test/mcp/lifecycle.test.ts +750 -0
  576. package/test/mcp/oauth-auto-connect.test.ts +199 -0
  577. package/test/mcp/oauth-browser.test.ts +249 -0
  578. package/test/memory/abort-leak.test.ts +137 -0
  579. package/test/patch/patch.test.ts +348 -0
  580. package/test/permission/arity.test.ts +33 -0
  581. package/test/permission/next.test.ts +1148 -0
  582. package/test/permission-task.test.ts +323 -0
  583. package/test/plugin/auth-override.test.ts +74 -0
  584. package/test/plugin/codex.test.ts +123 -0
  585. package/test/plugin/github-copilot-models.test.ts +117 -0
  586. package/test/plugin/install-concurrency.test.ts +140 -0
  587. package/test/plugin/install.test.ts +570 -0
  588. package/test/plugin/loader-shared.test.ts +1136 -0
  589. package/test/plugin/meta.test.ts +137 -0
  590. package/test/plugin/trigger.test.ts +111 -0
  591. package/test/preload.ts +90 -0
  592. package/test/project/migrate-global.test.ts +140 -0
  593. package/test/project/project.test.ts +459 -0
  594. package/test/project/state.test.ts +115 -0
  595. package/test/project/vcs.test.ts +228 -0
  596. package/test/project/worktree-remove.test.ts +96 -0
  597. package/test/project/worktree.test.ts +173 -0
  598. package/test/provider/amazon-bedrock.test.ts +447 -0
  599. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  600. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  601. package/test/provider/gitlab-duo.test.ts +412 -0
  602. package/test/provider/provider.test.ts +2284 -0
  603. package/test/provider/transform.test.ts +2839 -0
  604. package/test/pty/pty-output-isolation.test.ts +141 -0
  605. package/test/pty/pty-session.test.ts +92 -0
  606. package/test/pty/pty-shell.test.ts +59 -0
  607. package/test/question/question.test.ts +453 -0
  608. package/test/server/global-session-list.test.ts +89 -0
  609. package/test/server/project-init-git.test.ts +121 -0
  610. package/test/server/session-actions.test.ts +83 -0
  611. package/test/server/session-list.test.ts +98 -0
  612. package/test/server/session-messages.test.ts +159 -0
  613. package/test/server/session-select.test.ts +84 -0
  614. package/test/session/compaction.test.ts +1212 -0
  615. package/test/session/instruction.test.ts +286 -0
  616. package/test/session/llm.test.ts +1098 -0
  617. package/test/session/message-v2.test.ts +957 -0
  618. package/test/session/messages-pagination.test.ts +885 -0
  619. package/test/session/processor-effect.test.ts +747 -0
  620. package/test/session/prompt-effect.test.ts +1241 -0
  621. package/test/session/prompt.test.ts +518 -0
  622. package/test/session/retry.test.ts +232 -0
  623. package/test/session/revert-compact.test.ts +621 -0
  624. package/test/session/session.test.ts +142 -0
  625. package/test/session/snapshot-tool-race.test.ts +242 -0
  626. package/test/session/structured-output-integration.test.ts +233 -0
  627. package/test/session/structured-output.test.ts +391 -0
  628. package/test/session/system.test.ts +59 -0
  629. package/test/share/share-next.test.ts +333 -0
  630. package/test/shell/shell.test.ts +73 -0
  631. package/test/skill/discovery.test.ts +116 -0
  632. package/test/skill/skill.test.ts +392 -0
  633. package/test/snapshot/snapshot.test.ts +1312 -0
  634. package/test/storage/db.test.ts +14 -0
  635. package/test/storage/json-migration.test.ts +849 -0
  636. package/test/storage/storage.test.ts +295 -0
  637. package/test/sync/index.test.ts +191 -0
  638. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  639. package/test/tool/apply_patch.test.ts +567 -0
  640. package/test/tool/bash.test.ts +1099 -0
  641. package/test/tool/edit.test.ts +681 -0
  642. package/test/tool/external-directory.test.ts +198 -0
  643. package/test/tool/fixtures/large-image.png +0 -0
  644. package/test/tool/fixtures/models-api.json +65179 -0
  645. package/test/tool/grep.test.ts +111 -0
  646. package/test/tool/question.test.ts +126 -0
  647. package/test/tool/read.test.ts +546 -0
  648. package/test/tool/registry.test.ts +126 -0
  649. package/test/tool/skill.test.ts +167 -0
  650. package/test/tool/task.test.ts +49 -0
  651. package/test/tool/tool-define.test.ts +101 -0
  652. package/test/tool/truncation.test.ts +161 -0
  653. package/test/tool/webfetch.test.ts +101 -0
  654. package/test/tool/write.test.ts +353 -0
  655. package/test/util/data-url.test.ts +14 -0
  656. package/test/util/effect-zod.test.ts +61 -0
  657. package/test/util/error.test.ts +38 -0
  658. package/test/util/filesystem.test.ts +656 -0
  659. package/test/util/flock.test.ts +383 -0
  660. package/test/util/format.test.ts +59 -0
  661. package/test/util/glob.test.ts +164 -0
  662. package/test/util/iife.test.ts +36 -0
  663. package/test/util/lazy.test.ts +50 -0
  664. package/test/util/lock.test.ts +72 -0
  665. package/test/util/module.test.ts +59 -0
  666. package/test/util/process.test.ts +128 -0
  667. package/test/util/timeout.test.ts +21 -0
  668. package/test/util/which.test.ts +100 -0
  669. package/test/util/wildcard.test.ts +90 -0
  670. package/tsconfig.json +23 -0
@@ -0,0 +1,46 @@
1
+ import { ConfigMarkdown } from "@/config/markdown"
2
+ import { errorFormat } from "@/util/error"
3
+ import { Config } from "../config/config"
4
+ import { MCP } from "../mcp"
5
+ import { Provider } from "../provider/provider"
6
+ import { UI } from "./ui"
7
+
8
+ export function FormatError(input: unknown) {
9
+ if (MCP.Failed.isInstance(input))
10
+ return `MCP server "${input.data.name}" failed. Note, Koro does not support MCP authentication yet.`
11
+ if (Provider.ModelNotFoundError.isInstance(input)) {
12
+ const { providerID, modelID, suggestions } = input.data
13
+ return [
14
+ `Model not found: ${providerID}/${modelID}`,
15
+ ...(Array.isArray(suggestions) && suggestions.length ? ["Did you mean: " + suggestions.join(", ")] : []),
16
+ `Try: \`koro models\` to list available models`,
17
+ `Or check your config (opencode.json) provider/model names`,
18
+ ].join("\n")
19
+ }
20
+ if (Provider.InitError.isInstance(input)) {
21
+ return `Failed to initialize provider "${input.data.providerID}". Check credentials and configuration.`
22
+ }
23
+ if (Config.JsonError.isInstance(input)) {
24
+ return (
25
+ `Config file at ${input.data.path} is not valid JSON(C)` + (input.data.message ? `: ${input.data.message}` : "")
26
+ )
27
+ }
28
+ if (Config.ConfigDirectoryTypoError.isInstance(input)) {
29
+ return `Directory "${input.data.dir}" in ${input.data.path} is not valid. Rename the directory to "${input.data.suggestion}" or remove it. This is a common typo.`
30
+ }
31
+ if (ConfigMarkdown.FrontmatterError.isInstance(input)) {
32
+ return input.data.message
33
+ }
34
+ if (Config.InvalidError.isInstance(input))
35
+ return [
36
+ `Configuration is invalid${input.data.path && input.data.path !== "config" ? ` at ${input.data.path}` : ""}` +
37
+ (input.data.message ? `: ${input.data.message}` : ""),
38
+ ...(input.data.issues?.map((issue) => "↳ " + issue.message + " " + issue.path.join(".")) ?? []),
39
+ ].join("\n")
40
+
41
+ if (UI.CancelledError.isInstance(input)) return ""
42
+ }
43
+
44
+ export function FormatUnknownError(input: unknown): string {
45
+ return errorFormat(input)
46
+ }
@@ -0,0 +1,59 @@
1
+ import path from "path"
2
+ import { writeHeapSnapshot } from "node:v8"
3
+ import { Flag } from "@/flag/flag"
4
+ import { Global } from "@/global"
5
+ import { Log } from "@/util/log"
6
+
7
+ const log = Log.create({ service: "heap" })
8
+ const MINUTE = 60_000
9
+ const LIMIT = 2 * 1024 * 1024 * 1024
10
+
11
+ export namespace Heap {
12
+ let timer: Timer | undefined
13
+ let lock = false
14
+ let armed = true
15
+
16
+ export function start() {
17
+ if (!Flag.OPENCODE_AUTO_HEAP_SNAPSHOT) return
18
+ if (timer) return
19
+
20
+ const run = async () => {
21
+ if (lock) return
22
+
23
+ const stat = process.memoryUsage()
24
+ if (stat.rss <= LIMIT) {
25
+ armed = true
26
+ return
27
+ }
28
+ if (!armed) return
29
+
30
+ lock = true
31
+ armed = false
32
+ const file = path.join(
33
+ Global.Path.log,
34
+ `heap-${process.pid}-${new Date().toISOString().replace(/[:.]/g, "")}.heapsnapshot`,
35
+ )
36
+ log.warn("heap usage exceeded limit", {
37
+ rss: stat.rss,
38
+ heap: stat.heapUsed,
39
+ file,
40
+ })
41
+
42
+ await Promise.resolve()
43
+ .then(() => writeHeapSnapshot(file))
44
+ .catch((err) => {
45
+ log.error("failed to write heap snapshot", {
46
+ error: err instanceof Error ? err.message : String(err),
47
+ file,
48
+ })
49
+ })
50
+
51
+ lock = false
52
+ }
53
+
54
+ timer = setInterval(() => {
55
+ void run()
56
+ }, MINUTE)
57
+ timer.unref?.()
58
+ }
59
+ }
@@ -0,0 +1,14 @@
1
+ export const logo = {
2
+ left: [
3
+ "██ ▄█▀ ▄████▄",
4
+ "████ ██ ██",
5
+ "██ ▀█▄ ▀████▀",
6
+ ],
7
+ right: [
8
+ "█████▄ ▄████▄",
9
+ "██▄▄██▄ ██ ██",
10
+ "██ ██ ▀████▀",
11
+ ],
12
+ }
13
+
14
+ export const marks = "_^~"
@@ -0,0 +1,60 @@
1
+ import type { Argv, InferredOptionTypes } from "yargs"
2
+ import { Config } from "../config/config"
3
+
4
+ const options = {
5
+ port: {
6
+ type: "number" as const,
7
+ describe: "port to listen on",
8
+ default: 0,
9
+ },
10
+ hostname: {
11
+ type: "string" as const,
12
+ describe: "hostname to listen on",
13
+ default: "127.0.0.1",
14
+ },
15
+ mdns: {
16
+ type: "boolean" as const,
17
+ describe: "enable mDNS service discovery (defaults hostname to 0.0.0.0)",
18
+ default: false,
19
+ },
20
+ "mdns-domain": {
21
+ type: "string" as const,
22
+ describe: "custom domain name for mDNS service (default: koro.local)",
23
+ default: "koro.local",
24
+ },
25
+ cors: {
26
+ type: "string" as const,
27
+ array: true,
28
+ describe: "additional domains to allow for CORS",
29
+ default: [] as string[],
30
+ },
31
+ }
32
+
33
+ export type NetworkOptions = InferredOptionTypes<typeof options>
34
+
35
+ export function withNetworkOptions<T>(yargs: Argv<T>) {
36
+ return yargs.options(options)
37
+ }
38
+
39
+ export async function resolveNetworkOptions(args: NetworkOptions) {
40
+ const config = await Config.getGlobal()
41
+ const portExplicitlySet = process.argv.includes("--port")
42
+ const hostnameExplicitlySet = process.argv.includes("--hostname")
43
+ const mdnsExplicitlySet = process.argv.includes("--mdns")
44
+ const mdnsDomainExplicitlySet = process.argv.includes("--mdns-domain")
45
+ const corsExplicitlySet = process.argv.includes("--cors")
46
+
47
+ const mdns = mdnsExplicitlySet ? args.mdns : (config?.server?.mdns ?? args.mdns)
48
+ const mdnsDomain = mdnsDomainExplicitlySet ? args["mdns-domain"] : (config?.server?.mdnsDomain ?? args["mdns-domain"])
49
+ const port = portExplicitlySet ? args.port : (config?.server?.port ?? args.port)
50
+ const hostname = hostnameExplicitlySet
51
+ ? args.hostname
52
+ : mdns && !config?.server?.hostname
53
+ ? "0.0.0.0"
54
+ : (config?.server?.hostname ?? args.hostname)
55
+ const configCors = config?.server?.cors ?? []
56
+ const argsCors = Array.isArray(args.cors) ? args.cors : args.cors ? [args.cors] : []
57
+ const cors = [...configCors, ...argsCors]
58
+
59
+ return { hostname, port, mdns, mdnsDomain, cors }
60
+ }
package/src/cli/ui.ts ADDED
@@ -0,0 +1,133 @@
1
+ import z from "zod"
2
+ import { EOL } from "os"
3
+ import { NamedError } from "@opencode-ai/util/error"
4
+ import { logo as glyphs } from "./logo"
5
+
6
+ export namespace UI {
7
+ const wordmark = [
8
+ `⠀ ▄ `,
9
+ `█▀▀█ █▀▀█ █▀▀█ █▀▀▄ █▀▀▀ █▀▀█ █▀▀█ █▀▀█`,
10
+ `█ █ █ █ █▀▀▀ █ █ █ █ █ █ █ █▀▀▀`,
11
+ `▀▀▀▀ █▀▀▀ ▀▀▀▀ ▀ ▀ ▀▀▀▀ ▀▀▀▀ ▀▀▀▀ ▀▀▀▀`,
12
+ ]
13
+
14
+ export const CancelledError = NamedError.create("UICancelledError", z.void())
15
+
16
+ export const Style = {
17
+ TEXT_HIGHLIGHT: "\x1b[96m",
18
+ TEXT_HIGHLIGHT_BOLD: "\x1b[96m\x1b[1m",
19
+ TEXT_DIM: "\x1b[90m",
20
+ TEXT_DIM_BOLD: "\x1b[90m\x1b[1m",
21
+ TEXT_NORMAL: "\x1b[0m",
22
+ TEXT_NORMAL_BOLD: "\x1b[1m",
23
+ TEXT_WARNING: "\x1b[93m",
24
+ TEXT_WARNING_BOLD: "\x1b[93m\x1b[1m",
25
+ TEXT_DANGER: "\x1b[91m",
26
+ TEXT_DANGER_BOLD: "\x1b[91m\x1b[1m",
27
+ TEXT_SUCCESS: "\x1b[92m",
28
+ TEXT_SUCCESS_BOLD: "\x1b[92m\x1b[1m",
29
+ TEXT_INFO: "\x1b[94m",
30
+ TEXT_INFO_BOLD: "\x1b[94m\x1b[1m",
31
+ }
32
+
33
+ export function println(...message: string[]) {
34
+ print(...message)
35
+ process.stderr.write(EOL)
36
+ }
37
+
38
+ export function print(...message: string[]) {
39
+ blank = false
40
+ process.stderr.write(message.join(" "))
41
+ }
42
+
43
+ let blank = false
44
+ export function empty() {
45
+ if (blank) return
46
+ println("" + Style.TEXT_NORMAL)
47
+ blank = true
48
+ }
49
+
50
+ export function logo(pad?: string) {
51
+ if (!process.stdout.isTTY && !process.stderr.isTTY) {
52
+ const result = []
53
+ for (const row of wordmark) {
54
+ if (pad) result.push(pad)
55
+ result.push(row)
56
+ result.push(EOL)
57
+ }
58
+ return result.join("").trimEnd()
59
+ }
60
+
61
+ const result: string[] = []
62
+ const reset = "\x1b[0m"
63
+ const left = {
64
+ fg: "\x1b[90m",
65
+ shadow: "\x1b[38;5;235m",
66
+ bg: "\x1b[48;5;235m",
67
+ }
68
+ const right = {
69
+ fg: reset,
70
+ shadow: "\x1b[38;5;238m",
71
+ bg: "\x1b[48;5;238m",
72
+ }
73
+ const gap = " "
74
+ const draw = (line: string, fg: string, shadow: string, bg: string) => {
75
+ const parts: string[] = []
76
+ for (const char of line) {
77
+ if (char === "_") {
78
+ parts.push(bg, " ", reset)
79
+ continue
80
+ }
81
+ if (char === "^") {
82
+ parts.push(fg, bg, "▀", reset)
83
+ continue
84
+ }
85
+ if (char === "~") {
86
+ parts.push(shadow, "▀", reset)
87
+ continue
88
+ }
89
+ if (char === " ") {
90
+ parts.push(" ")
91
+ continue
92
+ }
93
+ parts.push(fg, char, reset)
94
+ }
95
+ return parts.join("")
96
+ }
97
+ glyphs.left.forEach((row, index) => {
98
+ if (pad) result.push(pad)
99
+ result.push(draw(row, left.fg, left.shadow, left.bg))
100
+ result.push(gap)
101
+ const other = glyphs.right[index] ?? ""
102
+ result.push(draw(other, right.fg, right.shadow, right.bg))
103
+ result.push(EOL)
104
+ })
105
+ return result.join("").trimEnd()
106
+ }
107
+
108
+ export async function input(prompt: string): Promise<string> {
109
+ const readline = require("readline")
110
+ const rl = readline.createInterface({
111
+ input: process.stdin,
112
+ output: process.stdout,
113
+ })
114
+
115
+ return new Promise((resolve) => {
116
+ rl.question(prompt, (answer: string) => {
117
+ rl.close()
118
+ resolve(answer.trim())
119
+ })
120
+ })
121
+ }
122
+
123
+ export function error(message: string) {
124
+ if (message.startsWith("Error: ")) {
125
+ message = message.slice("Error: ".length)
126
+ }
127
+ println(Style.TEXT_DANGER_BOLD + "Error: " + Style.TEXT_NORMAL + message)
128
+ }
129
+
130
+ export function markdown(text: string): string {
131
+ return text
132
+ }
133
+ }
@@ -0,0 +1,31 @@
1
+ import { Bus } from "@/bus"
2
+ import { Config } from "@/config/config"
3
+ import { Flag } from "@/flag/flag"
4
+ import { Installation } from "@/installation"
5
+
6
+ export async function upgrade() {
7
+ const config = await Config.getGlobal()
8
+ const method = await Installation.method()
9
+ const latest = await Installation.latest(method).catch(() => {})
10
+ if (!latest) return
11
+
12
+ if (Flag.OPENCODE_ALWAYS_NOTIFY_UPDATE) {
13
+ await Bus.publish(Installation.Event.UpdateAvailable, { version: latest })
14
+ return
15
+ }
16
+
17
+ if (Installation.VERSION === latest) return
18
+ if (config.autoupdate === false || Flag.OPENCODE_DISABLE_AUTOUPDATE) return
19
+
20
+ const kind = Installation.getReleaseType(Installation.VERSION, latest)
21
+
22
+ if (config.autoupdate === "notify" || kind !== "patch") {
23
+ await Bus.publish(Installation.Event.UpdateAvailable, { version: latest })
24
+ return
25
+ }
26
+
27
+ if (method === "unknown") return
28
+ await Installation.upgrade(method, latest)
29
+ .then(() => Bus.publish(Installation.Event.Updated, { version: latest }))
30
+ .catch(() => {})
31
+ }
@@ -0,0 +1,195 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { InstanceState } from "@/effect/instance-state"
3
+ import { makeRuntime } from "@/effect/run-service"
4
+ import { SessionID, MessageID } from "@/session/schema"
5
+ import { Effect, Layer, ServiceMap } from "effect"
6
+ import z from "zod"
7
+ import { Config } from "../config/config"
8
+ import { MCP } from "../mcp"
9
+ import { Skill } from "../skill"
10
+ import { Log } from "../util/log"
11
+ import PROMPT_INITIALIZE from "./template/initialize.txt"
12
+ import PROMPT_REVIEW from "./template/review.txt"
13
+
14
+ export namespace Command {
15
+ const log = Log.create({ service: "command" })
16
+
17
+ type State = {
18
+ commands: Record<string, Info>
19
+ }
20
+
21
+ export const Event = {
22
+ Executed: BusEvent.define(
23
+ "command.executed",
24
+ z.object({
25
+ name: z.string(),
26
+ sessionID: SessionID.zod,
27
+ arguments: z.string(),
28
+ messageID: MessageID.zod,
29
+ }),
30
+ ),
31
+ }
32
+
33
+ export const Info = z
34
+ .object({
35
+ name: z.string(),
36
+ description: z.string().optional(),
37
+ agent: z.string().optional(),
38
+ model: z.string().optional(),
39
+ source: z.enum(["command", "mcp", "skill"]).optional(),
40
+ // workaround for zod not supporting async functions natively so we use getters
41
+ // https://zod.dev/v4/changelog?id=zfunction
42
+ template: z.promise(z.string()).or(z.string()),
43
+ subtask: z.boolean().optional(),
44
+ hints: z.array(z.string()),
45
+ })
46
+ .meta({
47
+ ref: "Command",
48
+ })
49
+
50
+ // for some reason zod is inferring `string` for z.promise(z.string()).or(z.string()) so we have to manually override it
51
+ export type Info = Omit<z.infer<typeof Info>, "template"> & { template: Promise<string> | string }
52
+
53
+ export function hints(template: string) {
54
+ const result: string[] = []
55
+ const numbered = template.match(/\$\d+/g)
56
+ if (numbered) {
57
+ for (const match of [...new Set(numbered)].sort()) result.push(match)
58
+ }
59
+ if (template.includes("$ARGUMENTS")) result.push("$ARGUMENTS")
60
+ return result
61
+ }
62
+
63
+ export const Default = {
64
+ INIT: "init",
65
+ REVIEW: "review",
66
+ } as const
67
+
68
+ export interface Interface {
69
+ readonly get: (name: string) => Effect.Effect<Info | undefined>
70
+ readonly list: () => Effect.Effect<Info[]>
71
+ }
72
+
73
+ export class Service extends ServiceMap.Service<Service, Interface>()("@opencode/Command") {}
74
+
75
+ export const layer = Layer.effect(
76
+ Service,
77
+ Effect.gen(function* () {
78
+ const config = yield* Config.Service
79
+ const mcp = yield* MCP.Service
80
+ const skill = yield* Skill.Service
81
+
82
+ const init = Effect.fn("Command.state")(function* (ctx) {
83
+ const cfg = yield* config.get()
84
+ const commands: Record<string, Info> = {}
85
+
86
+ commands[Default.INIT] = {
87
+ name: Default.INIT,
88
+ description: "guided AGENTS.md setup",
89
+ source: "command",
90
+ get template() {
91
+ return PROMPT_INITIALIZE.replace("${path}", ctx.worktree)
92
+ },
93
+ hints: hints(PROMPT_INITIALIZE),
94
+ }
95
+ commands[Default.REVIEW] = {
96
+ name: Default.REVIEW,
97
+ description: "review changes [commit|branch|pr], defaults to uncommitted",
98
+ source: "command",
99
+ get template() {
100
+ return PROMPT_REVIEW.replace("${path}", ctx.worktree)
101
+ },
102
+ subtask: true,
103
+ hints: hints(PROMPT_REVIEW),
104
+ }
105
+
106
+ for (const [name, command] of Object.entries(cfg.command ?? {})) {
107
+ commands[name] = {
108
+ name,
109
+ agent: command.agent,
110
+ model: command.model,
111
+ description: command.description,
112
+ source: "command",
113
+ get template() {
114
+ return command.template
115
+ },
116
+ subtask: command.subtask,
117
+ hints: hints(command.template),
118
+ }
119
+ }
120
+
121
+ for (const [name, prompt] of Object.entries(yield* mcp.prompts())) {
122
+ commands[name] = {
123
+ name,
124
+ source: "mcp",
125
+ description: prompt.description,
126
+ get template() {
127
+ return Effect.runPromise(
128
+ mcp
129
+ .getPrompt(
130
+ prompt.client,
131
+ prompt.name,
132
+ prompt.arguments
133
+ ? Object.fromEntries(prompt.arguments.map((argument, i) => [argument.name, `$${i + 1}`]))
134
+ : {},
135
+ )
136
+ .pipe(
137
+ Effect.map(
138
+ (template) =>
139
+ template?.messages
140
+ .map((message) => (message.content.type === "text" ? message.content.text : ""))
141
+ .join("\n") || "",
142
+ ),
143
+ ),
144
+ )
145
+ },
146
+ hints: prompt.arguments?.map((_, i) => `$${i + 1}`) ?? [],
147
+ }
148
+ }
149
+
150
+ for (const item of yield* skill.all()) {
151
+ if (commands[item.name]) continue
152
+ commands[item.name] = {
153
+ name: item.name,
154
+ description: item.description,
155
+ source: "skill",
156
+ get template() {
157
+ return item.content
158
+ },
159
+ hints: [],
160
+ }
161
+ }
162
+
163
+ return {
164
+ commands,
165
+ }
166
+ })
167
+
168
+ const state = yield* InstanceState.make<State>((ctx) => init(ctx))
169
+
170
+ const get = Effect.fn("Command.get")(function* (name: string) {
171
+ const s = yield* InstanceState.get(state)
172
+ return s.commands[name]
173
+ })
174
+
175
+ const list = Effect.fn("Command.list")(function* () {
176
+ const s = yield* InstanceState.get(state)
177
+ return Object.values(s.commands)
178
+ })
179
+
180
+ return Service.of({ get, list })
181
+ }),
182
+ )
183
+
184
+ export const defaultLayer = layer.pipe(
185
+ Layer.provide(Config.defaultLayer),
186
+ Layer.provide(MCP.defaultLayer),
187
+ Layer.provide(Skill.defaultLayer),
188
+ )
189
+
190
+ const { runPromise } = makeRuntime(Service, defaultLayer)
191
+
192
+ export async function list() {
193
+ return runPromise((svc) => svc.list())
194
+ }
195
+ }
@@ -0,0 +1,66 @@
1
+ Create or update `AGENTS.md` for this repository.
2
+
3
+ The goal is a compact instruction file that helps future OpenCode sessions avoid mistakes and ramp up quickly. Every line should answer: "Would an agent likely miss this without help?" If not, leave it out.
4
+
5
+ User-provided focus or constraints (honor these):
6
+ $ARGUMENTS
7
+
8
+ ## How to investigate
9
+
10
+ Read the highest-value sources first:
11
+ - `README*`, root manifests, workspace config, lockfiles
12
+ - build, test, lint, formatter, typecheck, and codegen config
13
+ - CI workflows and pre-commit / task runner config
14
+ - existing instruction files (`AGENTS.md`, `CLAUDE.md`, `.cursor/rules/`, `.cursorrules`, `.github/copilot-instructions.md`)
15
+ - repo-local OpenCode config such as `opencode.json`
16
+
17
+ If architecture is still unclear after reading config and docs, inspect a small number of representative code files to find the real entrypoints, package boundaries, and execution flow. Prefer reading the files that explain how the system is wired together over random leaf files.
18
+
19
+ Prefer executable sources of truth over prose. If docs conflict with config or scripts, trust the executable source and only keep what you can verify.
20
+
21
+ ## What to extract
22
+
23
+ Look for the highest-signal facts for an agent working in this repo:
24
+ - exact developer commands, especially non-obvious ones
25
+ - how to run a single test, a single package, or a focused verification step
26
+ - required command order when it matters, such as `lint -> typecheck -> test`
27
+ - monorepo or multi-package boundaries, ownership of major directories, and the real app/library entrypoints
28
+ - framework or toolchain quirks: generated code, migrations, codegen, build artifacts, special env loading, dev servers, infra deploy flow
29
+ - repo-specific style or workflow conventions that differ from defaults
30
+ - testing quirks: fixtures, integration test prerequisites, snapshot workflows, required services, flaky or expensive suites
31
+ - important constraints from existing instruction files worth preserving
32
+
33
+ Good `AGENTS.md` content is usually hard-earned context that took reading multiple files to infer.
34
+
35
+ ## Questions
36
+
37
+ Only ask the user questions if the repo cannot answer something important. Use the `question` tool for one short batch at most.
38
+
39
+ Good questions:
40
+ - undocumented team conventions
41
+ - branch / PR / release expectations
42
+ - missing setup or test prerequisites that are known but not written down
43
+
44
+ Do not ask about anything the repo already makes clear.
45
+
46
+ ## Writing rules
47
+
48
+ Include only high-signal, repo-specific guidance such as:
49
+ - exact commands and shortcuts the agent would otherwise guess wrong
50
+ - architecture notes that are not obvious from filenames
51
+ - conventions that differ from language or framework defaults
52
+ - setup requirements, environment quirks, and operational gotchas
53
+ - references to existing instruction sources that matter
54
+
55
+ Exclude:
56
+ - generic software advice
57
+ - long tutorials or exhaustive file trees
58
+ - obvious language conventions
59
+ - speculative claims or anything you could not verify
60
+ - content better stored in another file referenced via `opencode.json` `instructions`
61
+
62
+ When in doubt, omit.
63
+
64
+ Prefer short sections and bullets. If the repo is simple, keep the file simple. If the repo is large, summarize the few structural facts that actually change how an agent should work.
65
+
66
+ If `AGENTS.md` already exists at `${path}`, improve it in place rather than rewriting blindly. Preserve verified useful guidance, delete fluff or stale claims, and reconcile it with the current codebase.