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,546 @@
1
+ import { afterEach, describe, expect, test } from "bun:test"
2
+ import path from "path"
3
+ import { ReadTool } from "../../src/tool/read"
4
+ import { Instance } from "../../src/project/instance"
5
+ import { Filesystem } from "../../src/util/filesystem"
6
+ import { tmpdir } from "../fixture/fixture"
7
+ import { Permission } from "../../src/permission"
8
+ import { Agent } from "../../src/agent/agent"
9
+ import { SessionID, MessageID } from "../../src/session/schema"
10
+
11
+ const FIXTURES_DIR = path.join(import.meta.dir, "fixtures")
12
+
13
+ afterEach(async () => {
14
+ await Instance.disposeAll()
15
+ })
16
+
17
+ const ctx = {
18
+ sessionID: SessionID.make("ses_test"),
19
+ messageID: MessageID.make(""),
20
+ callID: "",
21
+ agent: "build",
22
+ abort: AbortSignal.any([]),
23
+ messages: [],
24
+ metadata: () => {},
25
+ ask: async () => {},
26
+ }
27
+
28
+ const full = (p: string) => (process.platform === "win32" ? Filesystem.normalizePath(p) : p)
29
+ const glob = (p: string) =>
30
+ process.platform === "win32" ? Filesystem.normalizePathPattern(p) : p.replaceAll("\\", "/")
31
+
32
+ describe("tool.read external_directory permission", () => {
33
+ test("allows reading absolute path inside project directory", async () => {
34
+ await using tmp = await tmpdir({
35
+ init: async (dir) => {
36
+ await Bun.write(path.join(dir, "test.txt"), "hello world")
37
+ },
38
+ })
39
+ await Instance.provide({
40
+ directory: tmp.path,
41
+ fn: async () => {
42
+ const read = await ReadTool.init()
43
+ const result = await read.execute({ filePath: path.join(tmp.path, "test.txt") }, ctx)
44
+ expect(result.output).toContain("hello world")
45
+ },
46
+ })
47
+ })
48
+
49
+ test("allows reading file in subdirectory inside project directory", async () => {
50
+ await using tmp = await tmpdir({
51
+ init: async (dir) => {
52
+ await Bun.write(path.join(dir, "subdir", "test.txt"), "nested content")
53
+ },
54
+ })
55
+ await Instance.provide({
56
+ directory: tmp.path,
57
+ fn: async () => {
58
+ const read = await ReadTool.init()
59
+ const result = await read.execute({ filePath: path.join(tmp.path, "subdir", "test.txt") }, ctx)
60
+ expect(result.output).toContain("nested content")
61
+ },
62
+ })
63
+ })
64
+
65
+ test("asks for external_directory permission when reading absolute path outside project", async () => {
66
+ await using outerTmp = await tmpdir({
67
+ init: async (dir) => {
68
+ await Bun.write(path.join(dir, "secret.txt"), "secret data")
69
+ },
70
+ })
71
+ await using tmp = await tmpdir({ git: true })
72
+ await Instance.provide({
73
+ directory: tmp.path,
74
+ fn: async () => {
75
+ const read = await ReadTool.init()
76
+ const requests: Array<Omit<Permission.Request, "id" | "sessionID" | "tool">> = []
77
+ const testCtx = {
78
+ ...ctx,
79
+ ask: async (req: Omit<Permission.Request, "id" | "sessionID" | "tool">) => {
80
+ requests.push(req)
81
+ },
82
+ }
83
+ await read.execute({ filePath: path.join(outerTmp.path, "secret.txt") }, testCtx)
84
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
85
+ expect(extDirReq).toBeDefined()
86
+ expect(extDirReq!.patterns).toContain(glob(path.join(outerTmp.path, "*")))
87
+ },
88
+ })
89
+ })
90
+
91
+ if (process.platform === "win32") {
92
+ test("normalizes read permission paths on Windows", async () => {
93
+ await using tmp = await tmpdir({
94
+ git: true,
95
+ init: async (dir) => {
96
+ await Bun.write(path.join(dir, "test.txt"), "hello world")
97
+ },
98
+ })
99
+ await Instance.provide({
100
+ directory: tmp.path,
101
+ fn: async () => {
102
+ const read = await ReadTool.init()
103
+ const requests: Array<Omit<Permission.Request, "id" | "sessionID" | "tool">> = []
104
+ const testCtx = {
105
+ ...ctx,
106
+ ask: async (req: Omit<Permission.Request, "id" | "sessionID" | "tool">) => {
107
+ requests.push(req)
108
+ },
109
+ }
110
+ const target = path.join(tmp.path, "test.txt")
111
+ const alt = target
112
+ .replace(/^[A-Za-z]:/, "")
113
+ .replaceAll("\\", "/")
114
+ .toLowerCase()
115
+ await read.execute({ filePath: alt }, testCtx)
116
+ const readReq = requests.find((r) => r.permission === "read")
117
+ expect(readReq).toBeDefined()
118
+ expect(readReq!.patterns).toEqual([full(target)])
119
+ },
120
+ })
121
+ })
122
+ }
123
+
124
+ test("asks for directory-scoped external_directory permission when reading external directory", async () => {
125
+ await using outerTmp = await tmpdir({
126
+ init: async (dir) => {
127
+ await Bun.write(path.join(dir, "external", "a.txt"), "a")
128
+ },
129
+ })
130
+ await using tmp = await tmpdir({ git: true })
131
+ await Instance.provide({
132
+ directory: tmp.path,
133
+ fn: async () => {
134
+ const read = await ReadTool.init()
135
+ const requests: Array<Omit<Permission.Request, "id" | "sessionID" | "tool">> = []
136
+ const testCtx = {
137
+ ...ctx,
138
+ ask: async (req: Omit<Permission.Request, "id" | "sessionID" | "tool">) => {
139
+ requests.push(req)
140
+ },
141
+ }
142
+ await read.execute({ filePath: path.join(outerTmp.path, "external") }, testCtx)
143
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
144
+ expect(extDirReq).toBeDefined()
145
+ expect(extDirReq!.patterns).toContain(glob(path.join(outerTmp.path, "external", "*")))
146
+ },
147
+ })
148
+ })
149
+
150
+ test("asks for external_directory permission when reading relative path outside project", async () => {
151
+ await using tmp = await tmpdir({ git: true })
152
+ await Instance.provide({
153
+ directory: tmp.path,
154
+ fn: async () => {
155
+ const read = await ReadTool.init()
156
+ const requests: Array<Omit<Permission.Request, "id" | "sessionID" | "tool">> = []
157
+ const testCtx = {
158
+ ...ctx,
159
+ ask: async (req: Omit<Permission.Request, "id" | "sessionID" | "tool">) => {
160
+ requests.push(req)
161
+ },
162
+ }
163
+ // This will fail because file doesn't exist, but we can check if permission was asked
164
+ await read.execute({ filePath: "../outside.txt" }, testCtx).catch(() => {})
165
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
166
+ expect(extDirReq).toBeDefined()
167
+ },
168
+ })
169
+ })
170
+
171
+ test("does not ask for external_directory permission when reading inside project", async () => {
172
+ await using tmp = await tmpdir({
173
+ git: true,
174
+ init: async (dir) => {
175
+ await Bun.write(path.join(dir, "internal.txt"), "internal content")
176
+ },
177
+ })
178
+ await Instance.provide({
179
+ directory: tmp.path,
180
+ fn: async () => {
181
+ const read = await ReadTool.init()
182
+ const requests: Array<Omit<Permission.Request, "id" | "sessionID" | "tool">> = []
183
+ const testCtx = {
184
+ ...ctx,
185
+ ask: async (req: Omit<Permission.Request, "id" | "sessionID" | "tool">) => {
186
+ requests.push(req)
187
+ },
188
+ }
189
+ await read.execute({ filePath: path.join(tmp.path, "internal.txt") }, testCtx)
190
+ const extDirReq = requests.find((r) => r.permission === "external_directory")
191
+ expect(extDirReq).toBeUndefined()
192
+ },
193
+ })
194
+ })
195
+ })
196
+
197
+ describe("tool.read env file permissions", () => {
198
+ const cases: [string, boolean][] = [
199
+ [".env", true],
200
+ [".env.local", true],
201
+ [".env.production", true],
202
+ [".env.development.local", true],
203
+ [".env.example", false],
204
+ [".envrc", false],
205
+ ["environment.ts", false],
206
+ ]
207
+
208
+ describe.each(["build", "plan"])("agent=%s", (agentName) => {
209
+ test.each(cases)("%s asks=%s", async (filename, shouldAsk) => {
210
+ await using tmp = await tmpdir({
211
+ init: (dir) => Bun.write(path.join(dir, filename), "content"),
212
+ })
213
+ await Instance.provide({
214
+ directory: tmp.path,
215
+ fn: async () => {
216
+ const agent = await Agent.get(agentName)
217
+ let askedForEnv = false
218
+ const ctxWithPermissions = {
219
+ ...ctx,
220
+ ask: async (req: Omit<Permission.Request, "id" | "sessionID" | "tool">) => {
221
+ for (const pattern of req.patterns) {
222
+ const rule = Permission.evaluate(req.permission, pattern, agent.permission)
223
+ if (rule.action === "ask" && req.permission === "read") {
224
+ askedForEnv = true
225
+ }
226
+ if (rule.action === "deny") {
227
+ throw new Permission.DeniedError({ ruleset: agent.permission })
228
+ }
229
+ }
230
+ },
231
+ }
232
+ const read = await ReadTool.init()
233
+ await read.execute({ filePath: path.join(tmp.path, filename) }, ctxWithPermissions)
234
+ expect(askedForEnv).toBe(shouldAsk)
235
+ },
236
+ })
237
+ })
238
+ })
239
+ })
240
+
241
+ describe("tool.read truncation", () => {
242
+ test("truncates large file by bytes and sets truncated metadata", async () => {
243
+ await using tmp = await tmpdir({
244
+ init: async (dir) => {
245
+ const base = await Filesystem.readText(path.join(FIXTURES_DIR, "models-api.json"))
246
+ const target = 60 * 1024
247
+ const content = base.length >= target ? base : base.repeat(Math.ceil(target / base.length))
248
+ await Filesystem.write(path.join(dir, "large.json"), content)
249
+ },
250
+ })
251
+ await Instance.provide({
252
+ directory: tmp.path,
253
+ fn: async () => {
254
+ const read = await ReadTool.init()
255
+ const result = await read.execute({ filePath: path.join(tmp.path, "large.json") }, ctx)
256
+ expect(result.metadata.truncated).toBe(true)
257
+ expect(result.output).toContain("Output capped at")
258
+ expect(result.output).toContain("Use offset=")
259
+ },
260
+ })
261
+ })
262
+
263
+ test("truncates by line count when limit is specified", async () => {
264
+ await using tmp = await tmpdir({
265
+ init: async (dir) => {
266
+ const lines = Array.from({ length: 100 }, (_, i) => `line${i}`).join("\n")
267
+ await Bun.write(path.join(dir, "many-lines.txt"), lines)
268
+ },
269
+ })
270
+ await Instance.provide({
271
+ directory: tmp.path,
272
+ fn: async () => {
273
+ const read = await ReadTool.init()
274
+ const result = await read.execute({ filePath: path.join(tmp.path, "many-lines.txt"), limit: 10 }, ctx)
275
+ expect(result.metadata.truncated).toBe(true)
276
+ expect(result.output).toContain("Showing lines 1-10 of 100")
277
+ expect(result.output).toContain("Use offset=11")
278
+ expect(result.output).toContain("line0")
279
+ expect(result.output).toContain("line9")
280
+ expect(result.output).not.toContain("line10")
281
+ },
282
+ })
283
+ })
284
+
285
+ test("does not truncate small file", async () => {
286
+ await using tmp = await tmpdir({
287
+ init: async (dir) => {
288
+ await Bun.write(path.join(dir, "small.txt"), "hello world")
289
+ },
290
+ })
291
+ await Instance.provide({
292
+ directory: tmp.path,
293
+ fn: async () => {
294
+ const read = await ReadTool.init()
295
+ const result = await read.execute({ filePath: path.join(tmp.path, "small.txt") }, ctx)
296
+ expect(result.metadata.truncated).toBe(false)
297
+ expect(result.output).toContain("End of file")
298
+ },
299
+ })
300
+ })
301
+
302
+ test("respects offset parameter", async () => {
303
+ await using tmp = await tmpdir({
304
+ init: async (dir) => {
305
+ const lines = Array.from({ length: 20 }, (_, i) => `line${i + 1}`).join("\n")
306
+ await Bun.write(path.join(dir, "offset.txt"), lines)
307
+ },
308
+ })
309
+ await Instance.provide({
310
+ directory: tmp.path,
311
+ fn: async () => {
312
+ const read = await ReadTool.init()
313
+ const result = await read.execute({ filePath: path.join(tmp.path, "offset.txt"), offset: 10, limit: 5 }, ctx)
314
+ expect(result.output).toContain("10: line10")
315
+ expect(result.output).toContain("14: line14")
316
+ expect(result.output).not.toContain("9: line10")
317
+ expect(result.output).not.toContain("15: line15")
318
+ expect(result.output).toContain("line10")
319
+ expect(result.output).toContain("line14")
320
+ expect(result.output).not.toContain("line0")
321
+ expect(result.output).not.toContain("line15")
322
+ },
323
+ })
324
+ })
325
+
326
+ test("throws when offset is beyond end of file", async () => {
327
+ await using tmp = await tmpdir({
328
+ init: async (dir) => {
329
+ const lines = Array.from({ length: 3 }, (_, i) => `line${i + 1}`).join("\n")
330
+ await Bun.write(path.join(dir, "short.txt"), lines)
331
+ },
332
+ })
333
+ await Instance.provide({
334
+ directory: tmp.path,
335
+ fn: async () => {
336
+ const read = await ReadTool.init()
337
+ await expect(
338
+ read.execute({ filePath: path.join(tmp.path, "short.txt"), offset: 4, limit: 5 }, ctx),
339
+ ).rejects.toThrow("Offset 4 is out of range for this file (3 lines)")
340
+ },
341
+ })
342
+ })
343
+
344
+ test("allows reading empty file at default offset", async () => {
345
+ await using tmp = await tmpdir({
346
+ init: async (dir) => {
347
+ await Bun.write(path.join(dir, "empty.txt"), "")
348
+ },
349
+ })
350
+ await Instance.provide({
351
+ directory: tmp.path,
352
+ fn: async () => {
353
+ const read = await ReadTool.init()
354
+ const result = await read.execute({ filePath: path.join(tmp.path, "empty.txt") }, ctx)
355
+ expect(result.metadata.truncated).toBe(false)
356
+ expect(result.output).toContain("End of file - total 0 lines")
357
+ },
358
+ })
359
+ })
360
+
361
+ test("throws when offset > 1 for empty file", async () => {
362
+ await using tmp = await tmpdir({
363
+ init: async (dir) => {
364
+ await Bun.write(path.join(dir, "empty.txt"), "")
365
+ },
366
+ })
367
+ await Instance.provide({
368
+ directory: tmp.path,
369
+ fn: async () => {
370
+ const read = await ReadTool.init()
371
+ await expect(read.execute({ filePath: path.join(tmp.path, "empty.txt"), offset: 2 }, ctx)).rejects.toThrow(
372
+ "Offset 2 is out of range for this file (0 lines)",
373
+ )
374
+ },
375
+ })
376
+ })
377
+
378
+ test("does not mark final directory page as truncated", async () => {
379
+ await using tmp = await tmpdir({
380
+ init: async (dir) => {
381
+ await Promise.all(
382
+ Array.from({ length: 10 }, (_, i) => Bun.write(path.join(dir, "dir", `file-${i + 1}.txt`), `line${i}`)),
383
+ )
384
+ },
385
+ })
386
+ await Instance.provide({
387
+ directory: tmp.path,
388
+ fn: async () => {
389
+ const read = await ReadTool.init()
390
+ const result = await read.execute({ filePath: path.join(tmp.path, "dir"), offset: 6, limit: 5 }, ctx)
391
+ expect(result.metadata.truncated).toBe(false)
392
+ expect(result.output).not.toContain("Showing 5 of 10 entries")
393
+ },
394
+ })
395
+ })
396
+
397
+ test("truncates long lines", async () => {
398
+ await using tmp = await tmpdir({
399
+ init: async (dir) => {
400
+ const longLine = "x".repeat(3000)
401
+ await Bun.write(path.join(dir, "long-line.txt"), longLine)
402
+ },
403
+ })
404
+ await Instance.provide({
405
+ directory: tmp.path,
406
+ fn: async () => {
407
+ const read = await ReadTool.init()
408
+ const result = await read.execute({ filePath: path.join(tmp.path, "long-line.txt") }, ctx)
409
+ expect(result.output).toContain("(line truncated to 2000 chars)")
410
+ expect(result.output.length).toBeLessThan(3000)
411
+ },
412
+ })
413
+ })
414
+
415
+ test("image files set truncated to false", async () => {
416
+ await using tmp = await tmpdir({
417
+ init: async (dir) => {
418
+ // 1x1 red PNG
419
+ const png = Buffer.from(
420
+ "iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mP8z8DwHwAFBQIAX8jx0gAAAABJRU5ErkJggg==",
421
+ "base64",
422
+ )
423
+ await Bun.write(path.join(dir, "image.png"), png)
424
+ },
425
+ })
426
+ await Instance.provide({
427
+ directory: tmp.path,
428
+ fn: async () => {
429
+ const read = await ReadTool.init()
430
+ const result = await read.execute({ filePath: path.join(tmp.path, "image.png") }, ctx)
431
+ expect(result.metadata.truncated).toBe(false)
432
+ expect(result.attachments).toBeDefined()
433
+ expect(result.attachments?.length).toBe(1)
434
+ expect(result.attachments?.[0]).not.toHaveProperty("id")
435
+ expect(result.attachments?.[0]).not.toHaveProperty("sessionID")
436
+ expect(result.attachments?.[0]).not.toHaveProperty("messageID")
437
+ },
438
+ })
439
+ })
440
+
441
+ test("large image files are properly attached without error", async () => {
442
+ await Instance.provide({
443
+ directory: FIXTURES_DIR,
444
+ fn: async () => {
445
+ const read = await ReadTool.init()
446
+ const result = await read.execute({ filePath: path.join(FIXTURES_DIR, "large-image.png") }, ctx)
447
+ expect(result.metadata.truncated).toBe(false)
448
+ expect(result.attachments).toBeDefined()
449
+ expect(result.attachments?.length).toBe(1)
450
+ expect(result.attachments?.[0].type).toBe("file")
451
+ expect(result.attachments?.[0]).not.toHaveProperty("id")
452
+ expect(result.attachments?.[0]).not.toHaveProperty("sessionID")
453
+ expect(result.attachments?.[0]).not.toHaveProperty("messageID")
454
+ },
455
+ })
456
+ })
457
+
458
+ test(".fbs files (FlatBuffers schema) are read as text, not images", async () => {
459
+ await using tmp = await tmpdir({
460
+ init: async (dir) => {
461
+ // FlatBuffers schema content
462
+ const fbsContent = `namespace MyGame;
463
+
464
+ table Monster {
465
+ pos:Vec3;
466
+ name:string;
467
+ inventory:[ubyte];
468
+ }
469
+
470
+ root_type Monster;`
471
+ await Bun.write(path.join(dir, "schema.fbs"), fbsContent)
472
+ },
473
+ })
474
+ await Instance.provide({
475
+ directory: tmp.path,
476
+ fn: async () => {
477
+ const read = await ReadTool.init()
478
+ const result = await read.execute({ filePath: path.join(tmp.path, "schema.fbs") }, ctx)
479
+ // Should be read as text, not as image
480
+ expect(result.attachments).toBeUndefined()
481
+ expect(result.output).toContain("namespace MyGame")
482
+ expect(result.output).toContain("table Monster")
483
+ },
484
+ })
485
+ })
486
+ })
487
+
488
+ describe("tool.read loaded instructions", () => {
489
+ test("loads AGENTS.md from parent directory and includes in metadata", async () => {
490
+ await using tmp = await tmpdir({
491
+ init: async (dir) => {
492
+ await Bun.write(path.join(dir, "subdir", "AGENTS.md"), "# Test Instructions\nDo something special.")
493
+ await Bun.write(path.join(dir, "subdir", "nested", "test.txt"), "test content")
494
+ },
495
+ })
496
+ await Instance.provide({
497
+ directory: tmp.path,
498
+ fn: async () => {
499
+ const read = await ReadTool.init()
500
+ const result = await read.execute({ filePath: path.join(tmp.path, "subdir", "nested", "test.txt") }, ctx)
501
+ expect(result.output).toContain("test content")
502
+ expect(result.output).toContain("system-reminder")
503
+ expect(result.output).toContain("Test Instructions")
504
+ expect(result.metadata.loaded).toBeDefined()
505
+ expect(result.metadata.loaded).toContain(path.join(tmp.path, "subdir", "AGENTS.md"))
506
+ },
507
+ })
508
+ })
509
+ })
510
+
511
+ describe("tool.read binary detection", () => {
512
+ test("rejects text extension files with null bytes", async () => {
513
+ await using tmp = await tmpdir({
514
+ init: async (dir) => {
515
+ const bytes = Buffer.from([0x68, 0x65, 0x6c, 0x6c, 0x6f, 0x00, 0x77, 0x6f, 0x72, 0x6c, 0x64])
516
+ await Bun.write(path.join(dir, "null-byte.txt"), bytes)
517
+ },
518
+ })
519
+ await Instance.provide({
520
+ directory: tmp.path,
521
+ fn: async () => {
522
+ const read = await ReadTool.init()
523
+ await expect(read.execute({ filePath: path.join(tmp.path, "null-byte.txt") }, ctx)).rejects.toThrow(
524
+ "Cannot read binary file",
525
+ )
526
+ },
527
+ })
528
+ })
529
+
530
+ test("rejects known binary extensions", async () => {
531
+ await using tmp = await tmpdir({
532
+ init: async (dir) => {
533
+ await Bun.write(path.join(dir, "module.wasm"), "not really wasm")
534
+ },
535
+ })
536
+ await Instance.provide({
537
+ directory: tmp.path,
538
+ fn: async () => {
539
+ const read = await ReadTool.init()
540
+ await expect(read.execute({ filePath: path.join(tmp.path, "module.wasm") }, ctx)).rejects.toThrow(
541
+ "Cannot read binary file",
542
+ )
543
+ },
544
+ })
545
+ })
546
+ })
@@ -0,0 +1,126 @@
1
+ import { afterEach, describe, expect, test } from "bun:test"
2
+ import path from "path"
3
+ import fs from "fs/promises"
4
+ import { tmpdir } from "../fixture/fixture"
5
+ import { Instance } from "../../src/project/instance"
6
+ import { ToolRegistry } from "../../src/tool/registry"
7
+
8
+ afterEach(async () => {
9
+ await Instance.disposeAll()
10
+ })
11
+
12
+ describe("tool.registry", () => {
13
+ test("loads tools from .opencode/tool (singular)", async () => {
14
+ await using tmp = await tmpdir({
15
+ init: async (dir) => {
16
+ const opencodeDir = path.join(dir, ".opencode")
17
+ await fs.mkdir(opencodeDir, { recursive: true })
18
+
19
+ const toolDir = path.join(opencodeDir, "tool")
20
+ await fs.mkdir(toolDir, { recursive: true })
21
+
22
+ await Bun.write(
23
+ path.join(toolDir, "hello.ts"),
24
+ [
25
+ "export default {",
26
+ " description: 'hello tool',",
27
+ " args: {},",
28
+ " execute: async () => {",
29
+ " return 'hello world'",
30
+ " },",
31
+ "}",
32
+ "",
33
+ ].join("\n"),
34
+ )
35
+ },
36
+ })
37
+
38
+ await Instance.provide({
39
+ directory: tmp.path,
40
+ fn: async () => {
41
+ const ids = await ToolRegistry.ids()
42
+ expect(ids).toContain("hello")
43
+ },
44
+ })
45
+ })
46
+
47
+ test("loads tools from .opencode/tools (plural)", async () => {
48
+ await using tmp = await tmpdir({
49
+ init: async (dir) => {
50
+ const opencodeDir = path.join(dir, ".opencode")
51
+ await fs.mkdir(opencodeDir, { recursive: true })
52
+
53
+ const toolsDir = path.join(opencodeDir, "tools")
54
+ await fs.mkdir(toolsDir, { recursive: true })
55
+
56
+ await Bun.write(
57
+ path.join(toolsDir, "hello.ts"),
58
+ [
59
+ "export default {",
60
+ " description: 'hello tool',",
61
+ " args: {},",
62
+ " execute: async () => {",
63
+ " return 'hello world'",
64
+ " },",
65
+ "}",
66
+ "",
67
+ ].join("\n"),
68
+ )
69
+ },
70
+ })
71
+
72
+ await Instance.provide({
73
+ directory: tmp.path,
74
+ fn: async () => {
75
+ const ids = await ToolRegistry.ids()
76
+ expect(ids).toContain("hello")
77
+ },
78
+ })
79
+ })
80
+
81
+ test("loads tools with external dependencies without crashing", async () => {
82
+ await using tmp = await tmpdir({
83
+ init: async (dir) => {
84
+ const opencodeDir = path.join(dir, ".opencode")
85
+ await fs.mkdir(opencodeDir, { recursive: true })
86
+
87
+ const toolsDir = path.join(opencodeDir, "tools")
88
+ await fs.mkdir(toolsDir, { recursive: true })
89
+
90
+ await Bun.write(
91
+ path.join(opencodeDir, "package.json"),
92
+ JSON.stringify({
93
+ name: "custom-tools",
94
+ dependencies: {
95
+ "@opencode-ai/plugin": "^0.0.0",
96
+ cowsay: "^1.6.0",
97
+ },
98
+ }),
99
+ )
100
+
101
+ await Bun.write(
102
+ path.join(toolsDir, "cowsay.ts"),
103
+ [
104
+ "import { say } from 'cowsay'",
105
+ "export default {",
106
+ " description: 'tool that imports cowsay at top level',",
107
+ " args: { text: { type: 'string' } },",
108
+ " execute: async ({ text }: { text: string }) => {",
109
+ " return say({ text })",
110
+ " },",
111
+ "}",
112
+ "",
113
+ ].join("\n"),
114
+ )
115
+ },
116
+ })
117
+
118
+ await Instance.provide({
119
+ directory: tmp.path,
120
+ fn: async () => {
121
+ const ids = await ToolRegistry.ids()
122
+ expect(ids).toContain("cowsay")
123
+ },
124
+ })
125
+ })
126
+ })