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,215 @@
1
+ import { createConnection } from "net"
2
+ import { Log } from "../util/log"
3
+ import { OAUTH_CALLBACK_PORT, OAUTH_CALLBACK_PATH } from "./oauth-provider"
4
+
5
+ const log = Log.create({ service: "mcp.oauth-callback" })
6
+
7
+ const HTML_SUCCESS = `<!DOCTYPE html>
8
+ <html>
9
+ <head>
10
+ <title>Koro - Authorization Successful</title>
11
+ <style>
12
+ body { font-family: system-ui, -apple-system, sans-serif; display: flex; justify-content: center; align-items: center; height: 100vh; margin: 0; background: #1a1a2e; color: #eee; }
13
+ .container { text-align: center; padding: 2rem; }
14
+ h1 { color: #4ade80; margin-bottom: 1rem; }
15
+ p { color: #aaa; }
16
+ </style>
17
+ </head>
18
+ <body>
19
+ <div class="container">
20
+ <h1>Authorization Successful</h1>
21
+ <p>You can close this window and return to Koro.</p>
22
+ </div>
23
+ <script>setTimeout(() => window.close(), 2000);</script>
24
+ </body>
25
+ </html>`
26
+
27
+ const HTML_ERROR = (error: string) => `<!DOCTYPE html>
28
+ <html>
29
+ <head>
30
+ <title>Koro - Authorization Failed</title>
31
+ <style>
32
+ body { font-family: system-ui, -apple-system, sans-serif; display: flex; justify-content: center; align-items: center; height: 100vh; margin: 0; background: #1a1a2e; color: #eee; }
33
+ .container { text-align: center; padding: 2rem; }
34
+ h1 { color: #f87171; margin-bottom: 1rem; }
35
+ p { color: #aaa; }
36
+ .error { color: #fca5a5; font-family: monospace; margin-top: 1rem; padding: 1rem; background: rgba(248,113,113,0.1); border-radius: 0.5rem; }
37
+ </style>
38
+ </head>
39
+ <body>
40
+ <div class="container">
41
+ <h1>Authorization Failed</h1>
42
+ <p>An error occurred during authorization.</p>
43
+ <div class="error">${error}</div>
44
+ </div>
45
+ </body>
46
+ </html>`
47
+
48
+ interface PendingAuth {
49
+ resolve: (code: string) => void
50
+ reject: (error: Error) => void
51
+ timeout: ReturnType<typeof setTimeout>
52
+ }
53
+
54
+ export namespace McpOAuthCallback {
55
+ let server: ReturnType<typeof Bun.serve> | undefined
56
+ const pendingAuths = new Map<string, PendingAuth>()
57
+ // Reverse index: mcpName → oauthState, so cancelPending(mcpName) can
58
+ // find the right entry in pendingAuths (which is keyed by oauthState).
59
+ const mcpNameToState = new Map<string, string>()
60
+
61
+ const CALLBACK_TIMEOUT_MS = 5 * 60 * 1000 // 5 minutes
62
+
63
+ export async function ensureRunning(): Promise<void> {
64
+ if (server) return
65
+
66
+ const running = await isPortInUse()
67
+ if (running) {
68
+ log.info("oauth callback server already running on another instance", { port: OAUTH_CALLBACK_PORT })
69
+ return
70
+ }
71
+
72
+ server = Bun.serve({
73
+ port: OAUTH_CALLBACK_PORT,
74
+ fetch(req) {
75
+ const url = new URL(req.url)
76
+
77
+ if (url.pathname !== OAUTH_CALLBACK_PATH) {
78
+ return new Response("Not found", { status: 404 })
79
+ }
80
+
81
+ const code = url.searchParams.get("code")
82
+ const state = url.searchParams.get("state")
83
+ const error = url.searchParams.get("error")
84
+ const errorDescription = url.searchParams.get("error_description")
85
+
86
+ log.info("received oauth callback", { hasCode: !!code, state, error })
87
+
88
+ // Enforce state parameter presence
89
+ if (!state) {
90
+ const errorMsg = "Missing required state parameter - potential CSRF attack"
91
+ log.error("oauth callback missing state parameter", { url: url.toString() })
92
+ return new Response(HTML_ERROR(errorMsg), {
93
+ status: 400,
94
+ headers: { "Content-Type": "text/html" },
95
+ })
96
+ }
97
+
98
+ if (error) {
99
+ const errorMsg = errorDescription || error
100
+ if (pendingAuths.has(state)) {
101
+ const pending = pendingAuths.get(state)!
102
+ clearTimeout(pending.timeout)
103
+ pendingAuths.delete(state)
104
+ for (const [name, s] of mcpNameToState) {
105
+ if (s === state) {
106
+ mcpNameToState.delete(name)
107
+ break
108
+ }
109
+ }
110
+ pending.reject(new Error(errorMsg))
111
+ }
112
+ return new Response(HTML_ERROR(errorMsg), {
113
+ headers: { "Content-Type": "text/html" },
114
+ })
115
+ }
116
+
117
+ if (!code) {
118
+ return new Response(HTML_ERROR("No authorization code provided"), {
119
+ status: 400,
120
+ headers: { "Content-Type": "text/html" },
121
+ })
122
+ }
123
+
124
+ // Validate state parameter
125
+ if (!pendingAuths.has(state)) {
126
+ const errorMsg = "Invalid or expired state parameter - potential CSRF attack"
127
+ log.error("oauth callback with invalid state", { state, pendingStates: Array.from(pendingAuths.keys()) })
128
+ return new Response(HTML_ERROR(errorMsg), {
129
+ status: 400,
130
+ headers: { "Content-Type": "text/html" },
131
+ })
132
+ }
133
+
134
+ const pending = pendingAuths.get(state)!
135
+
136
+ clearTimeout(pending.timeout)
137
+ pendingAuths.delete(state)
138
+ // Clean up reverse index
139
+ for (const [name, s] of mcpNameToState) {
140
+ if (s === state) {
141
+ mcpNameToState.delete(name)
142
+ break
143
+ }
144
+ }
145
+ pending.resolve(code)
146
+
147
+ return new Response(HTML_SUCCESS, {
148
+ headers: { "Content-Type": "text/html" },
149
+ })
150
+ },
151
+ })
152
+
153
+ log.info("oauth callback server started", { port: OAUTH_CALLBACK_PORT })
154
+ }
155
+
156
+ export function waitForCallback(oauthState: string, mcpName?: string): Promise<string> {
157
+ if (mcpName) mcpNameToState.set(mcpName, oauthState)
158
+ return new Promise((resolve, reject) => {
159
+ const timeout = setTimeout(() => {
160
+ if (pendingAuths.has(oauthState)) {
161
+ pendingAuths.delete(oauthState)
162
+ if (mcpName) mcpNameToState.delete(mcpName)
163
+ reject(new Error("OAuth callback timeout - authorization took too long"))
164
+ }
165
+ }, CALLBACK_TIMEOUT_MS)
166
+
167
+ pendingAuths.set(oauthState, { resolve, reject, timeout })
168
+ })
169
+ }
170
+
171
+ export function cancelPending(mcpName: string): void {
172
+ // Look up the oauthState for this mcpName via the reverse index
173
+ const oauthState = mcpNameToState.get(mcpName)
174
+ const key = oauthState ?? mcpName
175
+ const pending = pendingAuths.get(key)
176
+ if (pending) {
177
+ clearTimeout(pending.timeout)
178
+ pendingAuths.delete(key)
179
+ mcpNameToState.delete(mcpName)
180
+ pending.reject(new Error("Authorization cancelled"))
181
+ }
182
+ }
183
+
184
+ export async function isPortInUse(): Promise<boolean> {
185
+ return new Promise((resolve) => {
186
+ const socket = createConnection(OAUTH_CALLBACK_PORT, "127.0.0.1")
187
+ socket.on("connect", () => {
188
+ socket.destroy()
189
+ resolve(true)
190
+ })
191
+ socket.on("error", () => {
192
+ resolve(false)
193
+ })
194
+ })
195
+ }
196
+
197
+ export async function stop(): Promise<void> {
198
+ if (server) {
199
+ server.stop()
200
+ server = undefined
201
+ log.info("oauth callback server stopped")
202
+ }
203
+
204
+ for (const [name, pending] of pendingAuths) {
205
+ clearTimeout(pending.timeout)
206
+ pending.reject(new Error("OAuth callback server stopped"))
207
+ }
208
+ pendingAuths.clear()
209
+ mcpNameToState.clear()
210
+ }
211
+
212
+ export function isRunning(): boolean {
213
+ return server !== undefined
214
+ }
215
+ }
@@ -0,0 +1,185 @@
1
+ import type { OAuthClientProvider } from "@modelcontextprotocol/sdk/client/auth.js"
2
+ import type {
3
+ OAuthClientMetadata,
4
+ OAuthTokens,
5
+ OAuthClientInformation,
6
+ OAuthClientInformationFull,
7
+ } from "@modelcontextprotocol/sdk/shared/auth.js"
8
+ import { McpAuth } from "./auth"
9
+ import { Log } from "../util/log"
10
+
11
+ const log = Log.create({ service: "mcp.oauth" })
12
+
13
+ const OAUTH_CALLBACK_PORT = 19876
14
+ const OAUTH_CALLBACK_PATH = "/mcp/oauth/callback"
15
+
16
+ export interface McpOAuthConfig {
17
+ clientId?: string
18
+ clientSecret?: string
19
+ scope?: string
20
+ }
21
+
22
+ export interface McpOAuthCallbacks {
23
+ onRedirect: (url: URL) => void | Promise<void>
24
+ }
25
+
26
+ export class McpOAuthProvider implements OAuthClientProvider {
27
+ constructor(
28
+ private mcpName: string,
29
+ private serverUrl: string,
30
+ private config: McpOAuthConfig,
31
+ private callbacks: McpOAuthCallbacks,
32
+ ) {}
33
+
34
+ get redirectUrl(): string {
35
+ return `http://127.0.0.1:${OAUTH_CALLBACK_PORT}${OAUTH_CALLBACK_PATH}`
36
+ }
37
+
38
+ get clientMetadata(): OAuthClientMetadata {
39
+ return {
40
+ redirect_uris: [this.redirectUrl],
41
+ client_name: "OpenCode",
42
+ client_uri: "https://opencode.ai",
43
+ grant_types: ["authorization_code", "refresh_token"],
44
+ response_types: ["code"],
45
+ token_endpoint_auth_method: this.config.clientSecret ? "client_secret_post" : "none",
46
+ }
47
+ }
48
+
49
+ async clientInformation(): Promise<OAuthClientInformation | undefined> {
50
+ // Check config first (pre-registered client)
51
+ if (this.config.clientId) {
52
+ return {
53
+ client_id: this.config.clientId,
54
+ client_secret: this.config.clientSecret,
55
+ }
56
+ }
57
+
58
+ // Check stored client info (from dynamic registration)
59
+ // Use getForUrl to validate credentials are for the current server URL
60
+ const entry = await McpAuth.getForUrl(this.mcpName, this.serverUrl)
61
+ if (entry?.clientInfo) {
62
+ // Check if client secret has expired
63
+ if (entry.clientInfo.clientSecretExpiresAt && entry.clientInfo.clientSecretExpiresAt < Date.now() / 1000) {
64
+ log.info("client secret expired, need to re-register", { mcpName: this.mcpName })
65
+ return undefined
66
+ }
67
+ return {
68
+ client_id: entry.clientInfo.clientId,
69
+ client_secret: entry.clientInfo.clientSecret,
70
+ }
71
+ }
72
+
73
+ // No client info or URL changed - will trigger dynamic registration
74
+ return undefined
75
+ }
76
+
77
+ async saveClientInformation(info: OAuthClientInformationFull): Promise<void> {
78
+ await McpAuth.updateClientInfo(
79
+ this.mcpName,
80
+ {
81
+ clientId: info.client_id,
82
+ clientSecret: info.client_secret,
83
+ clientIdIssuedAt: info.client_id_issued_at,
84
+ clientSecretExpiresAt: info.client_secret_expires_at,
85
+ },
86
+ this.serverUrl,
87
+ )
88
+ log.info("saved dynamically registered client", {
89
+ mcpName: this.mcpName,
90
+ clientId: info.client_id,
91
+ })
92
+ }
93
+
94
+ async tokens(): Promise<OAuthTokens | undefined> {
95
+ // Use getForUrl to validate tokens are for the current server URL
96
+ const entry = await McpAuth.getForUrl(this.mcpName, this.serverUrl)
97
+ if (!entry?.tokens) return undefined
98
+
99
+ return {
100
+ access_token: entry.tokens.accessToken,
101
+ token_type: "Bearer",
102
+ refresh_token: entry.tokens.refreshToken,
103
+ expires_in: entry.tokens.expiresAt
104
+ ? Math.max(0, Math.floor(entry.tokens.expiresAt - Date.now() / 1000))
105
+ : undefined,
106
+ scope: entry.tokens.scope,
107
+ }
108
+ }
109
+
110
+ async saveTokens(tokens: OAuthTokens): Promise<void> {
111
+ await McpAuth.updateTokens(
112
+ this.mcpName,
113
+ {
114
+ accessToken: tokens.access_token,
115
+ refreshToken: tokens.refresh_token,
116
+ expiresAt: tokens.expires_in ? Date.now() / 1000 + tokens.expires_in : undefined,
117
+ scope: tokens.scope,
118
+ },
119
+ this.serverUrl,
120
+ )
121
+ log.info("saved oauth tokens", { mcpName: this.mcpName })
122
+ }
123
+
124
+ async redirectToAuthorization(authorizationUrl: URL): Promise<void> {
125
+ log.info("redirecting to authorization", { mcpName: this.mcpName, url: authorizationUrl.toString() })
126
+ await this.callbacks.onRedirect(authorizationUrl)
127
+ }
128
+
129
+ async saveCodeVerifier(codeVerifier: string): Promise<void> {
130
+ await McpAuth.updateCodeVerifier(this.mcpName, codeVerifier)
131
+ }
132
+
133
+ async codeVerifier(): Promise<string> {
134
+ const entry = await McpAuth.get(this.mcpName)
135
+ if (!entry?.codeVerifier) {
136
+ throw new Error(`No code verifier saved for MCP server: ${this.mcpName}`)
137
+ }
138
+ return entry.codeVerifier
139
+ }
140
+
141
+ async saveState(state: string): Promise<void> {
142
+ await McpAuth.updateOAuthState(this.mcpName, state)
143
+ }
144
+
145
+ async state(): Promise<string> {
146
+ const entry = await McpAuth.get(this.mcpName)
147
+ if (entry?.oauthState) {
148
+ return entry.oauthState
149
+ }
150
+
151
+ // Generate a new state if none exists — the SDK calls state() as a
152
+ // generator, not just a reader, so we need to produce a value even when
153
+ // startAuth() hasn't pre-saved one (e.g. during automatic auth on first
154
+ // connect).
155
+ const newState = Array.from(crypto.getRandomValues(new Uint8Array(32)))
156
+ .map((b) => b.toString(16).padStart(2, "0"))
157
+ .join("")
158
+ await McpAuth.updateOAuthState(this.mcpName, newState)
159
+ return newState
160
+ }
161
+
162
+ async invalidateCredentials(type: "all" | "client" | "tokens"): Promise<void> {
163
+ log.info("invalidating credentials", { mcpName: this.mcpName, type })
164
+ const entry = await McpAuth.get(this.mcpName)
165
+ if (!entry) {
166
+ return
167
+ }
168
+
169
+ switch (type) {
170
+ case "all":
171
+ await McpAuth.remove(this.mcpName)
172
+ break
173
+ case "client":
174
+ delete entry.clientInfo
175
+ await McpAuth.set(this.mcpName, entry)
176
+ break
177
+ case "tokens":
178
+ delete entry.tokens
179
+ await McpAuth.set(this.mcpName, entry)
180
+ break
181
+ }
182
+ }
183
+ }
184
+
185
+ export { OAUTH_CALLBACK_PORT, OAUTH_CALLBACK_PATH }
@@ -0,0 +1,117 @@
1
+ import type { CoreMessage } from "ai"
2
+ import type { Task } from "@/orchestrator/task-manager"
3
+ import { Global } from "@/global"
4
+ import path from "path"
5
+ import fs from "fs/promises"
6
+
7
+ export interface SessionMemory {
8
+ messages: CoreMessage[]
9
+ filesChanged: string[]
10
+ tasksCompleted: Task[]
11
+ }
12
+
13
+ export interface ProjectMemory {
14
+ codebaseMap: Record<string, string>
15
+ knownPatterns: string[]
16
+ lastUpdated: Date
17
+ }
18
+
19
+ export interface DecisionMemory {
20
+ decisions: Array<{
21
+ timestamp: Date
22
+ context: string
23
+ decision: string
24
+ rationale: string
25
+ }>
26
+ }
27
+
28
+ export class MemoryManager {
29
+ session: SessionMemory = {
30
+ messages: [],
31
+ filesChanged: [],
32
+ tasksCompleted: [],
33
+ }
34
+
35
+ project: ProjectMemory = {
36
+ codebaseMap: {},
37
+ knownPatterns: [],
38
+ lastUpdated: new Date(),
39
+ }
40
+
41
+ decision: DecisionMemory = {
42
+ decisions: [],
43
+ }
44
+
45
+ private projectRoot: string | null = null
46
+
47
+ async loadProject(root: string): Promise<void> {
48
+ this.projectRoot = root
49
+ const memPath = path.join(root, ".koro", "project-memory.json")
50
+ const decPath = path.join(root, ".koro", "decisions.json")
51
+
52
+ try {
53
+ const memRaw = await fs.readFile(memPath, "utf-8")
54
+ this.project = JSON.parse(memRaw)
55
+ this.project.lastUpdated = new Date(this.project.lastUpdated)
56
+ } catch {
57
+ // no project memory yet
58
+ }
59
+
60
+ try {
61
+ const decRaw = await fs.readFile(decPath, "utf-8")
62
+ this.decision = JSON.parse(decRaw)
63
+ this.decision.decisions = this.decision.decisions.map((d) => ({
64
+ ...d,
65
+ timestamp: new Date(d.timestamp),
66
+ }))
67
+ } catch {
68
+ // no decisions yet
69
+ }
70
+ }
71
+
72
+ async saveProject(): Promise<void> {
73
+ if (!this.projectRoot) return
74
+ const dir = path.join(this.projectRoot, ".koro")
75
+ await fs.mkdir(dir, { recursive: true })
76
+
77
+ this.project.lastUpdated = new Date()
78
+ await fs.writeFile(path.join(dir, "project-memory.json"), JSON.stringify(this.project, null, 2))
79
+ await fs.writeFile(path.join(dir, "decisions.json"), JSON.stringify(this.decision, null, 2))
80
+ }
81
+
82
+ recordDecision(context: string, decision: string, rationale: string): void {
83
+ this.decision.decisions.push({
84
+ timestamp: new Date(),
85
+ context,
86
+ decision,
87
+ rationale,
88
+ })
89
+ }
90
+
91
+ buildContextSummary(): string {
92
+ const parts: string[] = []
93
+
94
+ if (this.session.filesChanged.length > 0) {
95
+ parts.push(`Files changed: ${this.session.filesChanged.join(", ")}`)
96
+ }
97
+
98
+ if (this.session.tasksCompleted.length > 0) {
99
+ parts.push(`Tasks completed: ${this.session.tasksCompleted.map((t) => t.description).join("; ")}`)
100
+ }
101
+
102
+ if (Object.keys(this.project.codebaseMap).length > 0) {
103
+ parts.push(
104
+ `Codebase context:\n${Object.entries(this.project.codebaseMap)
105
+ .map(([f, s]) => ` ${f}: ${s}`)
106
+ .join("\n")}`,
107
+ )
108
+ }
109
+
110
+ if (this.decision.decisions.length > 0) {
111
+ const recent = this.decision.decisions.slice(-5)
112
+ parts.push(`Recent decisions:\n${recent.map((d) => ` - ${d.decision} (${d.rationale})`).join("\n")}`)
113
+ }
114
+
115
+ return parts.join("\n\n")
116
+ }
117
+ }
package/src/node.ts ADDED
@@ -0,0 +1 @@
1
+ export { Server } from "./server/server"
@@ -0,0 +1,180 @@
1
+ import semver from "semver"
2
+ import z from "zod"
3
+ import { NamedError } from "@opencode-ai/util/error"
4
+ import { Global } from "../global"
5
+ import { Log } from "../util/log"
6
+ import path from "path"
7
+ import { readdir, rm } from "fs/promises"
8
+ import { Filesystem } from "@/util/filesystem"
9
+ import { Flock } from "@/util/flock"
10
+ import { Arborist } from "@npmcli/arborist"
11
+
12
+ export namespace Npm {
13
+ const log = Log.create({ service: "npm" })
14
+
15
+ export const InstallFailedError = NamedError.create(
16
+ "NpmInstallFailedError",
17
+ z.object({
18
+ pkg: z.string(),
19
+ }),
20
+ )
21
+
22
+ function directory(pkg: string) {
23
+ return path.join(Global.Path.cache, "packages", pkg)
24
+ }
25
+
26
+ function resolveEntryPoint(name: string, dir: string) {
27
+ let entrypoint: string | undefined
28
+ try {
29
+ entrypoint = typeof Bun !== "undefined" ? import.meta.resolve(name, dir) : import.meta.resolve(dir)
30
+ } catch {}
31
+ const result = {
32
+ directory: dir,
33
+ entrypoint,
34
+ }
35
+ return result
36
+ }
37
+
38
+ export async function outdated(pkg: string, cachedVersion: string): Promise<boolean> {
39
+ const response = await fetch(`https://registry.npmjs.org/${pkg}`)
40
+ if (!response.ok) {
41
+ log.warn("Failed to resolve latest version, using cached", { pkg, cachedVersion })
42
+ return false
43
+ }
44
+
45
+ const data = (await response.json()) as { "dist-tags"?: { latest?: string } }
46
+ const latestVersion = data?.["dist-tags"]?.latest
47
+ if (!latestVersion) {
48
+ log.warn("No latest version found, using cached", { pkg, cachedVersion })
49
+ return false
50
+ }
51
+
52
+ const range = /[\s^~*xX<>|=]/.test(cachedVersion)
53
+ if (range) return !semver.satisfies(latestVersion, cachedVersion)
54
+
55
+ return semver.lt(cachedVersion, latestVersion)
56
+ }
57
+
58
+ export async function add(pkg: string) {
59
+ const dir = directory(pkg)
60
+ await using _ = await Flock.acquire(`npm-install:${Filesystem.resolve(dir)}`)
61
+ log.info("installing package", {
62
+ pkg,
63
+ })
64
+
65
+ const arborist = new Arborist({
66
+ path: dir,
67
+ binLinks: true,
68
+ progress: false,
69
+ savePrefix: "",
70
+ })
71
+ const tree = await arborist.loadVirtual().catch(() => {})
72
+ if (tree) {
73
+ const first = tree.edgesOut.values().next().value?.to
74
+ if (first) {
75
+ return resolveEntryPoint(first.name, first.path)
76
+ }
77
+ }
78
+
79
+ const result = await arborist
80
+ .reify({
81
+ add: [pkg],
82
+ save: true,
83
+ saveType: "prod",
84
+ })
85
+ .catch((cause) => {
86
+ throw new InstallFailedError(
87
+ { pkg },
88
+ {
89
+ cause,
90
+ },
91
+ )
92
+ })
93
+
94
+ const first = result.edgesOut.values().next().value?.to
95
+ if (!first) throw new InstallFailedError({ pkg })
96
+ return resolveEntryPoint(first.name, first.path)
97
+ }
98
+
99
+ export async function install(dir: string) {
100
+ await using _ = await Flock.acquire(`npm-install:${dir}`)
101
+ log.info("checking dependencies", { dir })
102
+
103
+ const reify = async () => {
104
+ const arb = new Arborist({
105
+ path: dir,
106
+ binLinks: true,
107
+ progress: false,
108
+ savePrefix: "",
109
+ })
110
+ await arb.reify().catch(() => {})
111
+ }
112
+
113
+ if (!(await Filesystem.exists(path.join(dir, "node_modules")))) {
114
+ log.info("node_modules missing, reifying")
115
+ await reify()
116
+ return
117
+ }
118
+
119
+ const pkg = await Filesystem.readJson(path.join(dir, "package.json")).catch(() => ({}))
120
+ const lock = await Filesystem.readJson(path.join(dir, "package-lock.json")).catch(() => ({}))
121
+
122
+ const declared = new Set([
123
+ ...Object.keys(pkg.dependencies || {}),
124
+ ...Object.keys(pkg.devDependencies || {}),
125
+ ...Object.keys(pkg.peerDependencies || {}),
126
+ ...Object.keys(pkg.optionalDependencies || {}),
127
+ ])
128
+
129
+ const root = lock.packages?.[""] || {}
130
+ const locked = new Set([
131
+ ...Object.keys(root.dependencies || {}),
132
+ ...Object.keys(root.devDependencies || {}),
133
+ ...Object.keys(root.peerDependencies || {}),
134
+ ...Object.keys(root.optionalDependencies || {}),
135
+ ])
136
+
137
+ for (const name of declared) {
138
+ if (!locked.has(name)) {
139
+ log.info("dependency not in lock file, reifying", { name })
140
+ await reify()
141
+ return
142
+ }
143
+ }
144
+
145
+ log.info("dependencies in sync")
146
+ }
147
+
148
+ export async function which(pkg: string) {
149
+ const dir = directory(pkg)
150
+ const binDir = path.join(dir, "node_modules", ".bin")
151
+
152
+ const pick = async () => {
153
+ const files = await readdir(binDir).catch(() => [])
154
+ if (files.length === 0) return undefined
155
+ if (files.length === 1) return files[0]
156
+ // Multiple binaries — resolve from package.json bin field like npx does
157
+ const pkgJson = await Filesystem.readJson<{ bin?: string | Record<string, string> }>(
158
+ path.join(dir, "node_modules", pkg, "package.json"),
159
+ ).catch(() => undefined)
160
+ if (pkgJson?.bin) {
161
+ const unscoped = pkg.startsWith("@") ? pkg.split("/")[1] : pkg
162
+ const bin = pkgJson.bin
163
+ if (typeof bin === "string") return unscoped
164
+ const keys = Object.keys(bin)
165
+ if (keys.length === 1) return keys[0]
166
+ return bin[unscoped] ? unscoped : keys[0]
167
+ }
168
+ return files[0]
169
+ }
170
+
171
+ const bin = await pick()
172
+ if (bin) return path.join(binDir, bin)
173
+
174
+ await rm(path.join(dir, "package-lock.json"), { force: true })
175
+ await add(pkg)
176
+ const resolved = await pick()
177
+ if (!resolved) return
178
+ return path.join(binDir, resolved)
179
+ }
180
+ }