easc-cli 1.1.28

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 (404) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +108 -0
  5. package/bunfig.toml +7 -0
  6. package/package.json +132 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +172 -0
  9. package/script/deploy.ts +64 -0
  10. package/script/postinstall.mjs +125 -0
  11. package/script/publish-registries.ts +187 -0
  12. package/script/publish.ts +70 -0
  13. package/script/schema.ts +47 -0
  14. package/script/seed-e2e.ts +50 -0
  15. package/src/acp/README.md +164 -0
  16. package/src/acp/agent.ts +1285 -0
  17. package/src/acp/session.ts +105 -0
  18. package/src/acp/types.ts +22 -0
  19. package/src/agent/agent.ts +332 -0
  20. package/src/agent/generate.txt +75 -0
  21. package/src/agent/prompt/compaction.txt +12 -0
  22. package/src/agent/prompt/explore.txt +18 -0
  23. package/src/agent/prompt/summary.txt +11 -0
  24. package/src/agent/prompt/title.txt +43 -0
  25. package/src/auth/eliseart.ts +76 -0
  26. package/src/auth/index.ts +73 -0
  27. package/src/bun/index.ts +134 -0
  28. package/src/bus/bus-event.ts +43 -0
  29. package/src/bus/global.ts +10 -0
  30. package/src/bus/index.ts +105 -0
  31. package/src/cli/bootstrap.ts +17 -0
  32. package/src/cli/cmd/account.ts +81 -0
  33. package/src/cli/cmd/acp.ts +69 -0
  34. package/src/cli/cmd/agent.ts +257 -0
  35. package/src/cli/cmd/auth.ts +427 -0
  36. package/src/cli/cmd/cmd.ts +7 -0
  37. package/src/cli/cmd/debug/agent.ts +166 -0
  38. package/src/cli/cmd/debug/config.ts +16 -0
  39. package/src/cli/cmd/debug/file.ts +97 -0
  40. package/src/cli/cmd/debug/index.ts +48 -0
  41. package/src/cli/cmd/debug/lsp.ts +52 -0
  42. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  43. package/src/cli/cmd/debug/scrap.ts +16 -0
  44. package/src/cli/cmd/debug/skill.ts +16 -0
  45. package/src/cli/cmd/debug/snapshot.ts +52 -0
  46. package/src/cli/cmd/export.ts +88 -0
  47. package/src/cli/cmd/generate.ts +38 -0
  48. package/src/cli/cmd/github.ts +1548 -0
  49. package/src/cli/cmd/import.ts +98 -0
  50. package/src/cli/cmd/mcp.ts +827 -0
  51. package/src/cli/cmd/models.ts +77 -0
  52. package/src/cli/cmd/pr.ts +112 -0
  53. package/src/cli/cmd/run.ts +407 -0
  54. package/src/cli/cmd/serve.ts +20 -0
  55. package/src/cli/cmd/session.ts +135 -0
  56. package/src/cli/cmd/stats.ts +402 -0
  57. package/src/cli/cmd/tui/app.tsx +774 -0
  58. package/src/cli/cmd/tui/attach.ts +31 -0
  59. package/src/cli/cmd/tui/component/border.tsx +21 -0
  60. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  61. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  62. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  63. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  64. package/src/cli/cmd/tui/component/dialog-provider.tsx +256 -0
  65. package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
  66. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  67. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  68. package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
  69. package/src/cli/cmd/tui/component/dialog-supabase.tsx +102 -0
  70. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  71. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  72. package/src/cli/cmd/tui/component/logo.tsx +88 -0
  73. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +653 -0
  74. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  75. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  76. package/src/cli/cmd/tui/component/prompt/index.tsx +1182 -0
  77. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  78. package/src/cli/cmd/tui/component/spinner.tsx +16 -0
  79. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  80. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  81. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  82. package/src/cli/cmd/tui/context/args.tsx +14 -0
  83. package/src/cli/cmd/tui/context/directory.ts +13 -0
  84. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  85. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  86. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  87. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  88. package/src/cli/cmd/tui/context/local.tsx +402 -0
  89. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  90. package/src/cli/cmd/tui/context/route.tsx +46 -0
  91. package/src/cli/cmd/tui/context/sdk.tsx +94 -0
  92. package/src/cli/cmd/tui/context/sync.tsx +445 -0
  93. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  94. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  95. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  96. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  97. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  98. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  99. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  100. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  101. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  102. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  103. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  104. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  105. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  106. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  108. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  109. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  110. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  111. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  112. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  113. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  114. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  115. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  116. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  117. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  118. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  119. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  120. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  121. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  122. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  123. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  124. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  125. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  126. package/src/cli/cmd/tui/event.ts +48 -0
  127. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  129. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  130. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  131. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  132. package/src/cli/cmd/tui/routes/session/dialog-tool.tsx +63 -0
  133. package/src/cli/cmd/tui/routes/session/footer.tsx +129 -0
  134. package/src/cli/cmd/tui/routes/session/header.tsx +136 -0
  135. package/src/cli/cmd/tui/routes/session/index.tsx +2132 -0
  136. package/src/cli/cmd/tui/routes/session/permission.tsx +495 -0
  137. package/src/cli/cmd/tui/routes/session/question.tsx +435 -0
  138. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  139. package/src/cli/cmd/tui/thread.ts +165 -0
  140. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  141. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  142. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  143. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  144. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  145. package/src/cli/cmd/tui/ui/dialog-select.tsx +376 -0
  146. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  147. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  148. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  149. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  150. package/src/cli/cmd/tui/util/clipboard.ts +160 -0
  151. package/src/cli/cmd/tui/util/editor.ts +32 -0
  152. package/src/cli/cmd/tui/util/signal.ts +7 -0
  153. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  154. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  155. package/src/cli/cmd/tui/worker.ts +152 -0
  156. package/src/cli/cmd/uninstall.ts +357 -0
  157. package/src/cli/cmd/upgrade.ts +73 -0
  158. package/src/cli/cmd/web.ts +81 -0
  159. package/src/cli/error.ts +57 -0
  160. package/src/cli/network.ts +53 -0
  161. package/src/cli/ui.ts +84 -0
  162. package/src/cli/upgrade.ts +25 -0
  163. package/src/command/index.ts +131 -0
  164. package/src/command/template/initialize.txt +10 -0
  165. package/src/command/template/review.txt +99 -0
  166. package/src/config/config.ts +1361 -0
  167. package/src/config/markdown.ts +93 -0
  168. package/src/env/index.ts +26 -0
  169. package/src/file/ignore.ts +83 -0
  170. package/src/file/index.ts +411 -0
  171. package/src/file/ripgrep.ts +407 -0
  172. package/src/file/time.ts +64 -0
  173. package/src/file/watcher.ts +127 -0
  174. package/src/flag/flag.ts +54 -0
  175. package/src/format/formatter.ts +342 -0
  176. package/src/format/index.ts +137 -0
  177. package/src/global/index.ts +55 -0
  178. package/src/id/id.ts +83 -0
  179. package/src/ide/index.ts +76 -0
  180. package/src/index.ts +162 -0
  181. package/src/installation/index.ts +246 -0
  182. package/src/lsp/client.ts +252 -0
  183. package/src/lsp/index.ts +485 -0
  184. package/src/lsp/language.ts +119 -0
  185. package/src/lsp/server.ts +2046 -0
  186. package/src/mcp/auth.ts +135 -0
  187. package/src/mcp/index.ts +931 -0
  188. package/src/mcp/oauth-callback.ts +200 -0
  189. package/src/mcp/oauth-provider.ts +154 -0
  190. package/src/patch/index.ts +680 -0
  191. package/src/permission/arity.ts +163 -0
  192. package/src/permission/index.ts +210 -0
  193. package/src/permission/next.ts +269 -0
  194. package/src/plugin/codex.ts +493 -0
  195. package/src/plugin/copilot.ts +269 -0
  196. package/src/plugin/index.ts +135 -0
  197. package/src/project/bootstrap.ts +35 -0
  198. package/src/project/instance.ts +91 -0
  199. package/src/project/project.ts +339 -0
  200. package/src/project/state.ts +66 -0
  201. package/src/project/vcs.ts +76 -0
  202. package/src/provider/auth.ts +147 -0
  203. package/src/provider/models-macro.ts +11 -0
  204. package/src/provider/models.ts +112 -0
  205. package/src/provider/provider.ts +1391 -0
  206. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  207. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  208. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  209. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  210. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  211. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  212. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  213. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  214. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
  215. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  216. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  217. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  218. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  219. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  220. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  221. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  222. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  223. package/src/provider/transform.ts +733 -0
  224. package/src/pty/index.ts +232 -0
  225. package/src/question/index.ts +171 -0
  226. package/src/scheduler/index.ts +61 -0
  227. package/src/server/error.ts +36 -0
  228. package/src/server/event.ts +7 -0
  229. package/src/server/mdns.ts +59 -0
  230. package/src/server/routes/config.ts +92 -0
  231. package/src/server/routes/experimental.ts +208 -0
  232. package/src/server/routes/file.ts +197 -0
  233. package/src/server/routes/global.ts +135 -0
  234. package/src/server/routes/mcp.ts +361 -0
  235. package/src/server/routes/permission.ts +68 -0
  236. package/src/server/routes/project.ts +82 -0
  237. package/src/server/routes/provider.ts +165 -0
  238. package/src/server/routes/pty.ts +169 -0
  239. package/src/server/routes/question.ts +98 -0
  240. package/src/server/routes/session.ts +935 -0
  241. package/src/server/routes/tui.ts +379 -0
  242. package/src/server/server.ts +573 -0
  243. package/src/session/compaction.ts +225 -0
  244. package/src/session/index.ts +488 -0
  245. package/src/session/llm.ts +279 -0
  246. package/src/session/message-v2.ts +702 -0
  247. package/src/session/message.ts +189 -0
  248. package/src/session/processor.ts +406 -0
  249. package/src/session/prompt/anthropic-20250930.txt +166 -0
  250. package/src/session/prompt/anthropic.txt +105 -0
  251. package/src/session/prompt/anthropic_spoof.txt +1 -0
  252. package/src/session/prompt/beast.txt +147 -0
  253. package/src/session/prompt/build-switch.txt +5 -0
  254. package/src/session/prompt/codex_header.txt +79 -0
  255. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  256. package/src/session/prompt/gemini.txt +155 -0
  257. package/src/session/prompt/max-steps.txt +16 -0
  258. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  259. package/src/session/prompt/plan.txt +26 -0
  260. package/src/session/prompt/qwen.txt +109 -0
  261. package/src/session/prompt.ts +1820 -0
  262. package/src/session/retry.ts +90 -0
  263. package/src/session/revert.ts +108 -0
  264. package/src/session/status.ts +76 -0
  265. package/src/session/summary.ts +150 -0
  266. package/src/session/system.ts +152 -0
  267. package/src/session/todo.ts +37 -0
  268. package/src/share/share-next.ts +200 -0
  269. package/src/share/share.ts +92 -0
  270. package/src/shell/shell.ts +67 -0
  271. package/src/skill/index.ts +1 -0
  272. package/src/skill/skill.ts +136 -0
  273. package/src/snapshot/index.ts +236 -0
  274. package/src/storage/storage.ts +227 -0
  275. package/src/tool/apply_patch.ts +269 -0
  276. package/src/tool/apply_patch.txt +33 -0
  277. package/src/tool/bash.ts +259 -0
  278. package/src/tool/bash.txt +115 -0
  279. package/src/tool/batch.ts +175 -0
  280. package/src/tool/batch.txt +24 -0
  281. package/src/tool/codesearch.ts +132 -0
  282. package/src/tool/codesearch.txt +12 -0
  283. package/src/tool/edit.ts +645 -0
  284. package/src/tool/edit.txt +10 -0
  285. package/src/tool/external-directory.ts +32 -0
  286. package/src/tool/glob.ts +77 -0
  287. package/src/tool/glob.txt +6 -0
  288. package/src/tool/grep.ts +154 -0
  289. package/src/tool/grep.txt +8 -0
  290. package/src/tool/invalid.ts +17 -0
  291. package/src/tool/ls.ts +121 -0
  292. package/src/tool/ls.txt +1 -0
  293. package/src/tool/lsp.ts +96 -0
  294. package/src/tool/lsp.txt +19 -0
  295. package/src/tool/multiedit.ts +46 -0
  296. package/src/tool/multiedit.txt +41 -0
  297. package/src/tool/plan-enter.txt +14 -0
  298. package/src/tool/plan-exit.txt +13 -0
  299. package/src/tool/plan.ts +130 -0
  300. package/src/tool/question.ts +33 -0
  301. package/src/tool/question.txt +10 -0
  302. package/src/tool/read.ts +202 -0
  303. package/src/tool/read.txt +12 -0
  304. package/src/tool/registry.ts +163 -0
  305. package/src/tool/skill.ts +75 -0
  306. package/src/tool/task.ts +188 -0
  307. package/src/tool/task.txt +60 -0
  308. package/src/tool/todo.ts +53 -0
  309. package/src/tool/todoread.txt +14 -0
  310. package/src/tool/todowrite.txt +167 -0
  311. package/src/tool/tool.ts +88 -0
  312. package/src/tool/truncation.ts +106 -0
  313. package/src/tool/webfetch.ts +182 -0
  314. package/src/tool/webfetch.txt +13 -0
  315. package/src/tool/websearch.ts +150 -0
  316. package/src/tool/websearch.txt +14 -0
  317. package/src/tool/write.ts +80 -0
  318. package/src/tool/write.txt +8 -0
  319. package/src/util/archive.ts +16 -0
  320. package/src/util/color.ts +19 -0
  321. package/src/util/context.ts +25 -0
  322. package/src/util/defer.ts +12 -0
  323. package/src/util/eventloop.ts +20 -0
  324. package/src/util/filesystem.ts +93 -0
  325. package/src/util/fn.ts +11 -0
  326. package/src/util/format.ts +20 -0
  327. package/src/util/iife.ts +3 -0
  328. package/src/util/keybind.ts +103 -0
  329. package/src/util/lazy.ts +18 -0
  330. package/src/util/locale.ts +81 -0
  331. package/src/util/lock.ts +98 -0
  332. package/src/util/log.ts +180 -0
  333. package/src/util/queue.ts +32 -0
  334. package/src/util/rpc.ts +66 -0
  335. package/src/util/scrap.ts +10 -0
  336. package/src/util/signal.ts +12 -0
  337. package/src/util/timeout.ts +14 -0
  338. package/src/util/token.ts +7 -0
  339. package/src/util/wildcard.ts +56 -0
  340. package/src/worktree/index.ts +424 -0
  341. package/sst-env.d.ts +9 -0
  342. package/test/acp/event-subscription.test.ts +436 -0
  343. package/test/agent/agent.test.ts +638 -0
  344. package/test/bun.test.ts +53 -0
  345. package/test/cli/github-action.test.ts +129 -0
  346. package/test/cli/github-remote.test.ts +80 -0
  347. package/test/cli/tui/transcript.test.ts +297 -0
  348. package/test/config/agent-color.test.ts +66 -0
  349. package/test/config/config.test.ts +1414 -0
  350. package/test/config/fixtures/empty-frontmatter.md +4 -0
  351. package/test/config/fixtures/frontmatter.md +28 -0
  352. package/test/config/fixtures/no-frontmatter.md +1 -0
  353. package/test/config/markdown.test.ts +192 -0
  354. package/test/file/ignore.test.ts +10 -0
  355. package/test/file/path-traversal.test.ts +198 -0
  356. package/test/fixture/fixture.ts +45 -0
  357. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  358. package/test/ide/ide.test.ts +82 -0
  359. package/test/keybind.test.ts +421 -0
  360. package/test/lsp/client.test.ts +95 -0
  361. package/test/mcp/headers.test.ts +153 -0
  362. package/test/mcp/oauth-browser.test.ts +261 -0
  363. package/test/patch/patch.test.ts +348 -0
  364. package/test/permission/arity.test.ts +33 -0
  365. package/test/permission/next.test.ts +652 -0
  366. package/test/permission-task.test.ts +319 -0
  367. package/test/plugin/codex.test.ts +123 -0
  368. package/test/preload.ts +65 -0
  369. package/test/project/project.test.ts +120 -0
  370. package/test/provider/amazon-bedrock.test.ts +268 -0
  371. package/test/provider/gitlab-duo.test.ts +286 -0
  372. package/test/provider/provider.test.ts +2149 -0
  373. package/test/provider/transform.test.ts +1596 -0
  374. package/test/question/question.test.ts +300 -0
  375. package/test/scheduler.test.ts +73 -0
  376. package/test/server/session-list.test.ts +39 -0
  377. package/test/server/session-select.test.ts +78 -0
  378. package/test/session/compaction.test.ts +293 -0
  379. package/test/session/llm.test.ts +90 -0
  380. package/test/session/message-v2.test.ts +662 -0
  381. package/test/session/retry.test.ts +131 -0
  382. package/test/session/revert-compact.test.ts +285 -0
  383. package/test/session/session.test.ts +71 -0
  384. package/test/skill/skill.test.ts +185 -0
  385. package/test/snapshot/snapshot.test.ts +939 -0
  386. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  387. package/test/tool/apply_patch.test.ts +499 -0
  388. package/test/tool/bash.test.ts +320 -0
  389. package/test/tool/external-directory.test.ts +126 -0
  390. package/test/tool/fixtures/large-image.png +0 -0
  391. package/test/tool/fixtures/models-api.json +33453 -0
  392. package/test/tool/grep.test.ts +109 -0
  393. package/test/tool/question.test.ts +105 -0
  394. package/test/tool/read.test.ts +332 -0
  395. package/test/tool/registry.test.ts +76 -0
  396. package/test/tool/truncation.test.ts +159 -0
  397. package/test/util/filesystem.test.ts +39 -0
  398. package/test/util/format.test.ts +59 -0
  399. package/test/util/iife.test.ts +36 -0
  400. package/test/util/lazy.test.ts +50 -0
  401. package/test/util/lock.test.ts +72 -0
  402. package/test/util/timeout.test.ts +21 -0
  403. package/test/util/wildcard.test.ts +75 -0
  404. package/tsconfig.json +16 -0
@@ -0,0 +1,653 @@
1
+ import type { BoxRenderable, TextareaRenderable, KeyEvent, ScrollBoxRenderable } from "@opentui/core"
2
+ import fuzzysort from "fuzzysort"
3
+ import { firstBy } from "remeda"
4
+ import { createMemo, createResource, createEffect, onMount, onCleanup, Index, Show, createSignal } from "solid-js"
5
+ import { createStore } from "solid-js/store"
6
+ import { useSDK } from "@tui/context/sdk"
7
+ import { useSync } from "@tui/context/sync"
8
+ import { useTheme, selectedForeground } from "@tui/context/theme"
9
+ import { SplitBorder } from "@tui/component/border"
10
+ import { useCommandDialog } from "@tui/component/dialog-command"
11
+ import { useTerminalDimensions } from "@opentui/solid"
12
+ import { Locale } from "@/util/locale"
13
+ import type { PromptInfo } from "./history"
14
+ import { useFrecency } from "./frecency"
15
+
16
+ function removeLineRange(input: string) {
17
+ const hashIndex = input.lastIndexOf("#")
18
+ return hashIndex !== -1 ? input.substring(0, hashIndex) : input
19
+ }
20
+
21
+ function extractLineRange(input: string) {
22
+ const hashIndex = input.lastIndexOf("#")
23
+ if (hashIndex === -1) {
24
+ return { baseQuery: input }
25
+ }
26
+
27
+ const baseName = input.substring(0, hashIndex)
28
+ const linePart = input.substring(hashIndex + 1)
29
+ const lineMatch = linePart.match(/^(\d+)(?:-(\d*))?$/)
30
+
31
+ if (!lineMatch) {
32
+ return { baseQuery: baseName }
33
+ }
34
+
35
+ const startLine = Number(lineMatch[1])
36
+ const endLine = lineMatch[2] && startLine < Number(lineMatch[2]) ? Number(lineMatch[2]) : undefined
37
+
38
+ return {
39
+ lineRange: {
40
+ baseName,
41
+ startLine,
42
+ endLine,
43
+ },
44
+ baseQuery: baseName,
45
+ }
46
+ }
47
+
48
+ export type AutocompleteRef = {
49
+ onInput: (value: string) => void
50
+ onKeyDown: (e: KeyEvent) => void
51
+ visible: false | "@" | "/"
52
+ }
53
+
54
+ export type AutocompleteOption = {
55
+ display: string
56
+ value?: string
57
+ aliases?: string[]
58
+ disabled?: boolean
59
+ description?: string
60
+ isDirectory?: boolean
61
+ onSelect?: () => void
62
+ path?: string
63
+ }
64
+
65
+ export function Autocomplete(props: {
66
+ value: string
67
+ sessionID?: string
68
+ setPrompt: (input: (prompt: PromptInfo) => void) => void
69
+ setExtmark: (partIndex: number, extmarkId: number) => void
70
+ anchor: () => BoxRenderable
71
+ input: () => TextareaRenderable
72
+ ref: (ref: AutocompleteRef) => void
73
+ fileStyleId: number
74
+ agentStyleId: number
75
+ promptPartTypeId: () => number
76
+ }) {
77
+ const sdk = useSDK()
78
+ const sync = useSync()
79
+ const command = useCommandDialog()
80
+ const { theme } = useTheme()
81
+ const dimensions = useTerminalDimensions()
82
+ const frecency = useFrecency()
83
+
84
+ const [store, setStore] = createStore({
85
+ index: 0,
86
+ selected: 0,
87
+ visible: false as AutocompleteRef["visible"],
88
+ input: "keyboard" as "keyboard" | "mouse",
89
+ })
90
+
91
+ const [positionTick, setPositionTick] = createSignal(0)
92
+
93
+ createEffect(() => {
94
+ if (store.visible) {
95
+ let lastPos = { x: 0, y: 0, width: 0 }
96
+ const interval = setInterval(() => {
97
+ const anchor = props.anchor()
98
+ if (anchor.x !== lastPos.x || anchor.y !== lastPos.y || anchor.width !== lastPos.width) {
99
+ lastPos = { x: anchor.x, y: anchor.y, width: anchor.width }
100
+ setPositionTick((t) => t + 1)
101
+ }
102
+ }, 50)
103
+
104
+ onCleanup(() => clearInterval(interval))
105
+ }
106
+ })
107
+
108
+ const position = createMemo(() => {
109
+ if (!store.visible) return { x: 0, y: 0, width: 0 }
110
+ const dims = dimensions()
111
+ positionTick()
112
+ const anchor = props.anchor()
113
+ const parent = anchor.parent
114
+ const parentX = parent?.x ?? 0
115
+ const parentY = parent?.y ?? 0
116
+
117
+ return {
118
+ x: anchor.x - parentX,
119
+ y: anchor.y - parentY,
120
+ width: anchor.width,
121
+ }
122
+ })
123
+
124
+ const filter = createMemo(() => {
125
+ if (!store.visible) return
126
+ // Track props.value to make memo reactive to text changes
127
+ props.value // <- there surely is a better way to do this, like making .input() reactive
128
+
129
+ return props.input().getTextRange(store.index + 1, props.input().cursorOffset)
130
+ })
131
+
132
+ // When the filter changes due to how TUI works, the mousemove might still be triggered
133
+ // via a synthetic event as the layout moves underneath the cursor. This is a workaround to make sure the input mode remains keyboard so
134
+ // that the mouseover event doesn't trigger when filtering.
135
+ createEffect(() => {
136
+ filter()
137
+ setStore("input", "keyboard")
138
+ })
139
+
140
+ function insertPart(text: string, part: PromptInfo["parts"][number]) {
141
+ const input = props.input()
142
+ const currentCursorOffset = input.cursorOffset
143
+
144
+ const charAfterCursor = props.value.at(currentCursorOffset)
145
+ const needsSpace = charAfterCursor !== " "
146
+ const append = "@" + text + (needsSpace ? " " : "")
147
+
148
+ input.cursorOffset = store.index
149
+ const startCursor = input.logicalCursor
150
+ input.cursorOffset = currentCursorOffset
151
+ const endCursor = input.logicalCursor
152
+
153
+ input.deleteRange(startCursor.row, startCursor.col, endCursor.row, endCursor.col)
154
+ input.insertText(append)
155
+
156
+ const virtualText = "@" + text
157
+ const extmarkStart = store.index
158
+ const extmarkEnd = extmarkStart + Bun.stringWidth(virtualText)
159
+
160
+ const styleId = part.type === "file" ? props.fileStyleId : part.type === "agent" ? props.agentStyleId : undefined
161
+
162
+ const extmarkId = input.extmarks.create({
163
+ start: extmarkStart,
164
+ end: extmarkEnd,
165
+ virtual: true,
166
+ styleId,
167
+ typeId: props.promptPartTypeId(),
168
+ })
169
+
170
+ props.setPrompt((draft) => {
171
+ if (part.type === "file") {
172
+ const existingIndex = draft.parts.findIndex((p) => p.type === "file" && "url" in p && p.url === part.url)
173
+ if (existingIndex !== -1) {
174
+ const existing = draft.parts[existingIndex]
175
+ if (
176
+ part.source?.text &&
177
+ existing &&
178
+ "source" in existing &&
179
+ existing.source &&
180
+ "text" in existing.source &&
181
+ existing.source.text
182
+ ) {
183
+ existing.source.text.start = extmarkStart
184
+ existing.source.text.end = extmarkEnd
185
+ existing.source.text.value = virtualText
186
+ }
187
+ return
188
+ }
189
+ }
190
+
191
+ if (part.type === "file" && part.source?.text) {
192
+ part.source.text.start = extmarkStart
193
+ part.source.text.end = extmarkEnd
194
+ part.source.text.value = virtualText
195
+ } else if (part.type === "agent" && part.source) {
196
+ part.source.start = extmarkStart
197
+ part.source.end = extmarkEnd
198
+ part.source.value = virtualText
199
+ }
200
+ const partIndex = draft.parts.length
201
+ draft.parts.push(part)
202
+ props.setExtmark(partIndex, extmarkId)
203
+ })
204
+
205
+ if (part.type === "file" && part.source && part.source.type === "file") {
206
+ frecency.updateFrecency(part.source.path)
207
+ }
208
+ }
209
+
210
+ const [files] = createResource(
211
+ () => filter(),
212
+ async (query) => {
213
+ if (!store.visible || store.visible === "/") return []
214
+
215
+ const { lineRange, baseQuery } = extractLineRange(query ?? "")
216
+
217
+ // Get files from SDK
218
+ const result = await sdk.client.find.files({
219
+ query: baseQuery,
220
+ })
221
+
222
+ const options: AutocompleteOption[] = []
223
+
224
+ // Add file options
225
+ if (!result.error && result.data) {
226
+ const sortedFiles = result.data.sort((a, b) => {
227
+ const aScore = frecency.getFrecency(a)
228
+ const bScore = frecency.getFrecency(b)
229
+ if (aScore !== bScore) return bScore - aScore
230
+ const aDepth = a.split("/").length
231
+ const bDepth = b.split("/").length
232
+ if (aDepth !== bDepth) return aDepth - bDepth
233
+ return a.localeCompare(b)
234
+ })
235
+
236
+ const width = props.anchor().width - 4
237
+ options.push(
238
+ ...sortedFiles.map((item): AutocompleteOption => {
239
+ let url = `file://${process.cwd()}/${item}`
240
+ let filename = item
241
+ if (lineRange && !item.endsWith("/")) {
242
+ filename = `${item}#${lineRange.startLine}${lineRange.endLine ? `-${lineRange.endLine}` : ""}`
243
+ const urlObj = new URL(url)
244
+ urlObj.searchParams.set("start", String(lineRange.startLine))
245
+ if (lineRange.endLine !== undefined) {
246
+ urlObj.searchParams.set("end", String(lineRange.endLine))
247
+ }
248
+ url = urlObj.toString()
249
+ }
250
+
251
+ const isDir = item.endsWith("/")
252
+ return {
253
+ display: Locale.truncateMiddle(filename, width),
254
+ value: filename,
255
+ isDirectory: isDir,
256
+ path: item,
257
+ onSelect: () => {
258
+ insertPart(filename, {
259
+ type: "file",
260
+ mime: "text/plain",
261
+ filename,
262
+ url,
263
+ source: {
264
+ type: "file",
265
+ text: {
266
+ start: 0,
267
+ end: 0,
268
+ value: "",
269
+ },
270
+ path: item,
271
+ },
272
+ })
273
+ },
274
+ }
275
+ }),
276
+ )
277
+ }
278
+
279
+ return options
280
+ },
281
+ {
282
+ initialValue: [],
283
+ },
284
+ )
285
+
286
+ const mcpResources = createMemo(() => {
287
+ if (!store.visible || store.visible === "/") return []
288
+
289
+ const options: AutocompleteOption[] = []
290
+ const width = props.anchor().width - 4
291
+
292
+ for (const res of Object.values(sync.data.mcp_resource)) {
293
+ const text = `${res.name} (${res.uri})`
294
+ options.push({
295
+ display: Locale.truncateMiddle(text, width),
296
+ value: text,
297
+ description: res.description,
298
+ onSelect: () => {
299
+ insertPart(res.name, {
300
+ type: "file",
301
+ mime: res.mimeType ?? "text/plain",
302
+ filename: res.name,
303
+ url: res.uri,
304
+ source: {
305
+ type: "resource",
306
+ text: {
307
+ start: 0,
308
+ end: 0,
309
+ value: "",
310
+ },
311
+ clientName: res.client,
312
+ uri: res.uri,
313
+ },
314
+ })
315
+ },
316
+ })
317
+ }
318
+
319
+ return options
320
+ })
321
+
322
+ const agents = createMemo(() => {
323
+ const agents = sync.data.agent
324
+ return agents
325
+ .filter((agent) => !agent.hidden && agent.mode !== "primary")
326
+ .map(
327
+ (agent): AutocompleteOption => ({
328
+ display: "@" + agent.name,
329
+ onSelect: () => {
330
+ insertPart(agent.name, {
331
+ type: "agent",
332
+ name: agent.name,
333
+ source: {
334
+ start: 0,
335
+ end: 0,
336
+ value: "",
337
+ },
338
+ })
339
+ },
340
+ }),
341
+ )
342
+ })
343
+
344
+ const commands = createMemo((): AutocompleteOption[] => {
345
+ const results: AutocompleteOption[] = [...command.slashes()]
346
+
347
+ for (const serverCommand of sync.data.command) {
348
+ results.push({
349
+ display: "/" + serverCommand.name + (serverCommand.mcp ? " (MCP)" : ""),
350
+ description: serverCommand.description,
351
+ onSelect: () => {
352
+ const newText = "/" + serverCommand.name + " "
353
+ const cursor = props.input().logicalCursor
354
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
355
+ props.input().insertText(newText)
356
+ props.input().cursorOffset = Bun.stringWidth(newText)
357
+ },
358
+ })
359
+ }
360
+
361
+ results.sort((a, b) => a.display.localeCompare(b.display))
362
+
363
+ const max = firstBy(results, [(x) => x.display.length, "desc"])?.display.length
364
+ if (!max) return results
365
+ return results.map((item) => ({
366
+ ...item,
367
+ display: item.display.padEnd(max + 2),
368
+ }))
369
+ })
370
+
371
+ const options = createMemo((prev: AutocompleteOption[] | undefined) => {
372
+ const filesValue = files()
373
+ const agentsValue = agents()
374
+ const commandsValue = commands()
375
+
376
+ const mixed: AutocompleteOption[] =
377
+ store.visible === "@" ? [...agentsValue, ...(filesValue || []), ...mcpResources()] : [...commandsValue]
378
+
379
+ const currentFilter = filter()
380
+
381
+ if (!currentFilter) {
382
+ return mixed
383
+ }
384
+
385
+ if (files.loading && prev && prev.length > 0) {
386
+ return prev
387
+ }
388
+
389
+ const result = fuzzysort.go(removeLineRange(currentFilter), mixed, {
390
+ keys: [
391
+ (obj) => removeLineRange((obj.value ?? obj.display).trimEnd()),
392
+ "description",
393
+ (obj) => obj.aliases?.join(" ") ?? "",
394
+ ],
395
+ limit: 10,
396
+ scoreFn: (objResults) => {
397
+ const displayResult = objResults[0]
398
+ let score = objResults.score
399
+ if (displayResult && displayResult.target.startsWith(store.visible + currentFilter)) {
400
+ score *= 2
401
+ }
402
+ const frecencyScore = objResults.obj.path ? frecency.getFrecency(objResults.obj.path) : 0
403
+ return score * (1 + frecencyScore)
404
+ },
405
+ })
406
+
407
+ return result.map((arr) => arr.obj)
408
+ })
409
+
410
+ createEffect(() => {
411
+ filter()
412
+ setStore("selected", 0)
413
+ })
414
+
415
+ function move(direction: -1 | 1) {
416
+ if (!store.visible) return
417
+ if (!options().length) return
418
+ let next = store.selected + direction
419
+ if (next < 0) next = options().length - 1
420
+ if (next >= options().length) next = 0
421
+ moveTo(next)
422
+ }
423
+
424
+ function moveTo(next: number) {
425
+ setStore("selected", next)
426
+ if (!scroll) return
427
+ const viewportHeight = Math.min(height(), options().length)
428
+ const scrollBottom = scroll.scrollTop + viewportHeight
429
+ if (next < scroll.scrollTop) {
430
+ scroll.scrollBy(next - scroll.scrollTop)
431
+ } else if (next + 1 > scrollBottom) {
432
+ scroll.scrollBy(next + 1 - scrollBottom)
433
+ }
434
+ }
435
+
436
+ function select() {
437
+ const selected = options()[store.selected]
438
+ if (!selected) return
439
+ hide()
440
+ selected.onSelect?.()
441
+ }
442
+
443
+ function expandDirectory() {
444
+ const selected = options()[store.selected]
445
+ if (!selected) return
446
+
447
+ const input = props.input()
448
+ const currentCursorOffset = input.cursorOffset
449
+
450
+ const displayText = selected.display.trimEnd()
451
+ const path = displayText.startsWith("@") ? displayText.slice(1) : displayText
452
+
453
+ input.cursorOffset = store.index
454
+ const startCursor = input.logicalCursor
455
+ input.cursorOffset = currentCursorOffset
456
+ const endCursor = input.logicalCursor
457
+
458
+ input.deleteRange(startCursor.row, startCursor.col, endCursor.row, endCursor.col)
459
+ input.insertText("@" + path)
460
+
461
+ setStore("selected", 0)
462
+ }
463
+
464
+ function show(mode: "@" | "/") {
465
+ command.keybinds(false)
466
+ setStore({
467
+ visible: mode,
468
+ index: props.input().cursorOffset,
469
+ })
470
+ }
471
+
472
+ function hide() {
473
+ const text = props.input().plainText
474
+ if (store.visible === "/" && !text.endsWith(" ") && text.startsWith("/")) {
475
+ const cursor = props.input().logicalCursor
476
+ props.input().deleteRange(0, 0, cursor.row, cursor.col)
477
+ // Sync the prompt store immediately since onContentChange is async
478
+ props.setPrompt((draft) => {
479
+ draft.input = props.input().plainText
480
+ })
481
+ }
482
+ command.keybinds(true)
483
+ setStore("visible", false)
484
+ }
485
+
486
+ onMount(() => {
487
+ props.ref({
488
+ get visible() {
489
+ return store.visible
490
+ },
491
+ onInput(value) {
492
+ if (store.visible) {
493
+ if (
494
+ // Typed text before the trigger
495
+ props.input().cursorOffset <= store.index ||
496
+ // There is a space between the trigger and the cursor
497
+ props.input().getTextRange(store.index, props.input().cursorOffset).match(/\s/) ||
498
+ // "/<command>" is not the sole content
499
+ (store.visible === "/" && value.match(/^\S+\s+\S+\s*$/))
500
+ ) {
501
+ hide()
502
+ }
503
+ return
504
+ }
505
+
506
+ // Check if autocomplete should reopen (e.g., after backspace deleted a space)
507
+ const offset = props.input().cursorOffset
508
+ if (offset === 0) return
509
+
510
+ // Check for "/" at position 0 - reopen slash commands
511
+ if (value.startsWith("/") && !value.slice(0, offset).match(/\s/)) {
512
+ show("/")
513
+ setStore("index", 0)
514
+ return
515
+ }
516
+
517
+ // Check for "@" trigger - find the nearest "@" before cursor with no whitespace between
518
+ const text = value.slice(0, offset)
519
+ const idx = text.lastIndexOf("@")
520
+ if (idx === -1) return
521
+
522
+ const between = text.slice(idx)
523
+ const before = idx === 0 ? undefined : value[idx - 1]
524
+ if ((before === undefined || /\s/.test(before)) && !between.match(/\s/)) {
525
+ show("@")
526
+ setStore("index", idx)
527
+ }
528
+ },
529
+ onKeyDown(e: KeyEvent) {
530
+ if (store.visible) {
531
+ const name = e.name?.toLowerCase()
532
+ const ctrlOnly = e.ctrl && !e.meta && !e.shift
533
+ const isNavUp = name === "up" || (ctrlOnly && name === "p")
534
+ const isNavDown = name === "down" || (ctrlOnly && name === "n")
535
+
536
+ if (isNavUp) {
537
+ setStore("input", "keyboard")
538
+ move(-1)
539
+ e.preventDefault()
540
+ return
541
+ }
542
+ if (isNavDown) {
543
+ setStore("input", "keyboard")
544
+ move(1)
545
+ e.preventDefault()
546
+ return
547
+ }
548
+ if (name === "escape") {
549
+ hide()
550
+ e.preventDefault()
551
+ return
552
+ }
553
+ if (name === "return") {
554
+ select()
555
+ e.preventDefault()
556
+ return
557
+ }
558
+ if (name === "tab") {
559
+ const selected = options()[store.selected]
560
+ if (selected?.isDirectory) {
561
+ expandDirectory()
562
+ } else {
563
+ select()
564
+ }
565
+ e.preventDefault()
566
+ return
567
+ }
568
+ }
569
+ if (!store.visible) {
570
+ if (e.name === "@") {
571
+ const cursorOffset = props.input().cursorOffset
572
+ const charBeforeCursor =
573
+ cursorOffset === 0 ? undefined : props.input().getTextRange(cursorOffset - 1, cursorOffset)
574
+ const canTrigger = charBeforeCursor === undefined || charBeforeCursor === "" || /\s/.test(charBeforeCursor)
575
+ if (canTrigger) show("@")
576
+ }
577
+
578
+ if (e.name === "/") {
579
+ if (props.input().cursorOffset === 0) show("/")
580
+ }
581
+ }
582
+ },
583
+ })
584
+ })
585
+
586
+ const height = createMemo(() => {
587
+ const count = options().length || 1
588
+ if (!store.visible) return Math.min(10, count)
589
+ positionTick()
590
+ return Math.min(10, count, Math.max(1, props.anchor().y))
591
+ })
592
+
593
+ let scroll: ScrollBoxRenderable
594
+
595
+ return (
596
+ <box
597
+ visible={store.visible !== false}
598
+ position="absolute"
599
+ top={position().y - height()}
600
+ left={position().x}
601
+ width={position().width}
602
+ zIndex={100}
603
+ {...SplitBorder}
604
+ borderColor={theme.border}
605
+ >
606
+ <scrollbox
607
+ ref={(r: ScrollBoxRenderable) => (scroll = r)}
608
+ backgroundColor={theme.backgroundMenu}
609
+ height={height()}
610
+ scrollbarOptions={{ visible: false }}
611
+ >
612
+ <Index
613
+ each={options()}
614
+ fallback={
615
+ <box paddingLeft={1} paddingRight={1}>
616
+ <text fg={theme.textMuted}>No matching items</text>
617
+ </box>
618
+ }
619
+ >
620
+ {(option, index) => (
621
+ <box
622
+ paddingLeft={1}
623
+ paddingRight={1}
624
+ backgroundColor={index === store.selected ? theme.primary : undefined}
625
+ flexDirection="row"
626
+ onMouseMove={() => {
627
+ setStore("input", "mouse")
628
+ }}
629
+ onMouseOver={() => {
630
+ if (store.input !== "mouse") return
631
+ moveTo(index)
632
+ }}
633
+ onMouseDown={() => {
634
+ setStore("input", "mouse")
635
+ moveTo(index)
636
+ }}
637
+ onMouseUp={() => select()}
638
+ >
639
+ <text fg={index === store.selected ? selectedForeground(theme) : theme.text} flexShrink={0}>
640
+ {option().display}
641
+ </text>
642
+ <Show when={option().description}>
643
+ <text fg={index === store.selected ? selectedForeground(theme) : theme.textMuted} wrapMode="none">
644
+ {option().description}
645
+ </text>
646
+ </Show>
647
+ </box>
648
+ )}
649
+ </Index>
650
+ </scrollbox>
651
+ </box>
652
+ )
653
+ }