nuwaxcode 1.1.25

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