nikcli 0.0.6

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 (602) hide show
  1. package/.turbo/turbo-typecheck.log +1 -0
  2. package/AGENTS.md +27 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +15 -0
  5. package/bin/nikcli +84 -0
  6. package/config.json +13 -0
  7. package/docs/tailscale-mobile/01-tailscale-setup.md +94 -0
  8. package/docs/tailscale-mobile/02-host-setup.md +115 -0
  9. package/docs/tailscale-mobile/03-phone-and-serve.md +134 -0
  10. package/docs/tailscale-mobile/README.md +59 -0
  11. package/examples/README.md +54 -0
  12. package/package.json +147 -0
  13. package/parsers-config.ts +253 -0
  14. package/script/build.ts +179 -0
  15. package/script/postinstall.mjs +125 -0
  16. package/script/publish-registries.ts +187 -0
  17. package/script/publish.ts +100 -0
  18. package/script/schema.ts +47 -0
  19. package/script/seed-e2e.ts +50 -0
  20. package/sequential-prancing-forest.md +373 -0
  21. package/src/acp/README.md +164 -0
  22. package/src/acp/agent.ts +1303 -0
  23. package/src/acp/session.ts +105 -0
  24. package/src/acp/types.ts +22 -0
  25. package/src/agent/agent.ts +528 -0
  26. package/src/agent/generate.txt +32 -0
  27. package/src/agent/prompt/compaction.txt +14 -0
  28. package/src/agent/prompt/explore.txt +18 -0
  29. package/src/agent/prompt/summary.txt +11 -0
  30. package/src/agent/prompt/title.txt +44 -0
  31. package/src/auth/index.ts +73 -0
  32. package/src/bun/index.ts +119 -0
  33. package/src/bun/registry.ts +54 -0
  34. package/src/bus/bus-event.ts +43 -0
  35. package/src/bus/global.ts +10 -0
  36. package/src/bus/index.ts +105 -0
  37. package/src/chatbot/handlers.ts +150 -0
  38. package/src/chatbot/index.ts +132 -0
  39. package/src/cli/bootstrap.ts +17 -0
  40. package/src/cli/cmd/acp.ts +69 -0
  41. package/src/cli/cmd/ads.ts +377 -0
  42. package/src/cli/cmd/agent.ts +259 -0
  43. package/src/cli/cmd/auth.ts +400 -0
  44. package/src/cli/cmd/chatbot.ts +420 -0
  45. package/src/cli/cmd/cmd.ts +7 -0
  46. package/src/cli/cmd/companion.ts +81 -0
  47. package/src/cli/cmd/connectors.ts +593 -0
  48. package/src/cli/cmd/debug/agent.ts +166 -0
  49. package/src/cli/cmd/debug/config.ts +16 -0
  50. package/src/cli/cmd/debug/file.ts +97 -0
  51. package/src/cli/cmd/debug/index.ts +48 -0
  52. package/src/cli/cmd/debug/lsp.ts +52 -0
  53. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  54. package/src/cli/cmd/debug/scrap.ts +16 -0
  55. package/src/cli/cmd/debug/skill.ts +16 -0
  56. package/src/cli/cmd/debug/snapshot.ts +52 -0
  57. package/src/cli/cmd/export.ts +88 -0
  58. package/src/cli/cmd/generate.ts +38 -0
  59. package/src/cli/cmd/github.ts +412 -0
  60. package/src/cli/cmd/image-model.ts +128 -0
  61. package/src/cli/cmd/import.ts +201 -0
  62. package/src/cli/cmd/lovable.ts +128 -0
  63. package/src/cli/cmd/mcp.ts +738 -0
  64. package/src/cli/cmd/mobile.ts +223 -0
  65. package/src/cli/cmd/models.ts +77 -0
  66. package/src/cli/cmd/plug.ts +231 -0
  67. package/src/cli/cmd/pr.ts +104 -0
  68. package/src/cli/cmd/rag-model.ts +167 -0
  69. package/src/cli/cmd/remote.ts +416 -0
  70. package/src/cli/cmd/run.ts +589 -0
  71. package/src/cli/cmd/serve.ts +51 -0
  72. package/src/cli/cmd/session.ts +133 -0
  73. package/src/cli/cmd/speak-model.ts +204 -0
  74. package/src/cli/cmd/stats.ts +402 -0
  75. package/src/cli/cmd/tui/app.tsx +841 -0
  76. package/src/cli/cmd/tui/attach.ts +31 -0
  77. package/src/cli/cmd/tui/component/border.tsx +75 -0
  78. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  79. package/src/cli/cmd/tui/component/dialog-command.tsx +172 -0
  80. package/src/cli/cmd/tui/component/dialog-config.tsx +291 -0
  81. package/src/cli/cmd/tui/component/dialog-connectors.tsx +440 -0
  82. package/src/cli/cmd/tui/component/dialog-image-model.tsx +97 -0
  83. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  84. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  85. package/src/cli/cmd/tui/component/dialog-provider.tsx +260 -0
  86. package/src/cli/cmd/tui/component/dialog-rag-model.tsx +217 -0
  87. package/src/cli/cmd/tui/component/dialog-remote.tsx +489 -0
  88. package/src/cli/cmd/tui/component/dialog-session-list.tsx +170 -0
  89. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  90. package/src/cli/cmd/tui/component/dialog-settings/index.tsx +59 -0
  91. package/src/cli/cmd/tui/component/dialog-settings/prompt.tsx +40 -0
  92. package/src/cli/cmd/tui/component/dialog-settings/sidebar.tsx +39 -0
  93. package/src/cli/cmd/tui/component/dialog-settings/spinner.tsx +62 -0
  94. package/src/cli/cmd/tui/component/dialog-settings/ui.tsx +58 -0
  95. package/src/cli/cmd/tui/component/dialog-skills.tsx +117 -0
  96. package/src/cli/cmd/tui/component/dialog-speak-model.tsx +304 -0
  97. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  98. package/src/cli/cmd/tui/component/dialog-status.tsx +165 -0
  99. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  100. package/src/cli/cmd/tui/component/dialog-theme-create.tsx +717 -0
  101. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +52 -0
  102. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +350 -0
  103. package/src/cli/cmd/tui/component/error-component.tsx +91 -0
  104. package/src/cli/cmd/tui/component/logo.tsx +103 -0
  105. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  106. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +669 -0
  107. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  108. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  109. package/src/cli/cmd/tui/component/prompt/index.tsx +2165 -0
  110. package/src/cli/cmd/tui/component/prompt/stash.tsx +63 -0
  111. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  112. package/src/cli/cmd/tui/component/startup-loading.tsx +63 -0
  113. package/src/cli/cmd/tui/component/table/markdown-table.tsx +267 -0
  114. package/src/cli/cmd/tui/component/table-db/db/connections.ts +75 -0
  115. package/src/cli/cmd/tui/component/table-db/db/db-connection.ts +223 -0
  116. package/src/cli/cmd/tui/component/table-db/db/db-preview.ts +202 -0
  117. package/src/cli/cmd/tui/component/table-db/db/factory.ts +77 -0
  118. package/src/cli/cmd/tui/component/table-db/db/index.ts +9 -0
  119. package/src/cli/cmd/tui/component/table-db/db/mysql-connection.ts +330 -0
  120. package/src/cli/cmd/tui/component/table-db/db/postgres-connection.ts +338 -0
  121. package/src/cli/cmd/tui/component/table-db/db/sqlite-connection.ts +302 -0
  122. package/src/cli/cmd/tui/component/table-db/db/types.ts +108 -0
  123. package/src/cli/cmd/tui/component/table-db/table/dbedit-hooks.ts +74 -0
  124. package/src/cli/cmd/tui/component/table-db/table/index.ts +15 -0
  125. package/src/cli/cmd/tui/component/table-db/table/table-events.ts +54 -0
  126. package/src/cli/cmd/tui/component/table-db/table/table-formatters.ts +191 -0
  127. package/src/cli/cmd/tui/component/table-db/table/table-hooks.ts +105 -0
  128. package/src/cli/cmd/tui/component/table-db/table/table-keyboard-handler.ts +255 -0
  129. package/src/cli/cmd/tui/component/table-db/table/table-layout-engine.ts +208 -0
  130. package/src/cli/cmd/tui/component/table-db/table/table-renderable.ts +486 -0
  131. package/src/cli/cmd/tui/component/table-db/table/table-selection-manager.ts +136 -0
  132. package/src/cli/cmd/tui/component/table-db/table/table-state.ts +198 -0
  133. package/src/cli/cmd/tui/component/table-db/table/types.ts +69 -0
  134. package/src/cli/cmd/tui/component/table-db/ui/db-visualizer.tsx +71 -0
  135. package/src/cli/cmd/tui/component/table-db/ui/index.ts +2 -0
  136. package/src/cli/cmd/tui/component/table-db/ui/table-renderer.ts +607 -0
  137. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  138. package/src/cli/cmd/tui/component/tips.tsx +195 -0
  139. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  140. package/src/cli/cmd/tui/context/args.tsx +14 -0
  141. package/src/cli/cmd/tui/context/directory.ts +13 -0
  142. package/src/cli/cmd/tui/context/exit.tsx +24 -0
  143. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  144. package/src/cli/cmd/tui/context/keybind.tsx +102 -0
  145. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  146. package/src/cli/cmd/tui/context/local.tsx +458 -0
  147. package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
  148. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  149. package/src/cli/cmd/tui/context/route.tsx +54 -0
  150. package/src/cli/cmd/tui/context/sdk.tsx +128 -0
  151. package/src/cli/cmd/tui/context/server.tsx +8 -0
  152. package/src/cli/cmd/tui/context/sync.tsx +510 -0
  153. package/src/cli/cmd/tui/context/theme/abyss.json +233 -0
  154. package/src/cli/cmd/tui/context/theme/apple.json +235 -0
  155. package/src/cli/cmd/tui/context/theme/arctic.json +232 -0
  156. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  157. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  158. package/src/cli/cmd/tui/context/theme/ayuai.json +229 -0
  159. package/src/cli/cmd/tui/context/theme/blood.json +229 -0
  160. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  161. package/src/cli/cmd/tui/context/theme/catmoe.json +235 -0
  162. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  163. package/src/cli/cmd/tui/context/theme/catppuccin-latte.json +233 -0
  164. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  165. package/src/cli/cmd/tui/context/theme/catppuccin.json +259 -0
  166. package/src/cli/cmd/tui/context/theme/charcoal.json +230 -0
  167. package/src/cli/cmd/tui/context/theme/chromatic.json +235 -0
  168. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  169. package/src/cli/cmd/tui/context/theme/cosmic.json +234 -0
  170. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  171. package/src/cli/cmd/tui/context/theme/cyber.json +235 -0
  172. package/src/cli/cmd/tui/context/theme/dawnfox.json +229 -0
  173. package/src/cli/cmd/tui/context/theme/dimension.json +235 -0
  174. package/src/cli/cmd/tui/context/theme/dracula-official.json +222 -0
  175. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  176. package/src/cli/cmd/tui/context/theme/dream.json +235 -0
  177. package/src/cli/cmd/tui/context/theme/duo.json +235 -0
  178. package/src/cli/cmd/tui/context/theme/dusk.json +235 -0
  179. package/src/cli/cmd/tui/context/theme/ebony.json +232 -0
  180. package/src/cli/cmd/tui/context/theme/equilibrium.json +232 -0
  181. package/src/cli/cmd/tui/context/theme/ethereal.json +235 -0
  182. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  183. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  184. package/src/cli/cmd/tui/context/theme/fusion.json +235 -0
  185. package/src/cli/cmd/tui/context/theme/ghost.json +235 -0
  186. package/src/cli/cmd/tui/context/theme/github-dark.json +229 -0
  187. package/src/cli/cmd/tui/context/theme/github-dimmed.json +231 -0
  188. package/src/cli/cmd/tui/context/theme/github-light.json +229 -0
  189. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  190. package/src/cli/cmd/tui/context/theme/glass.json +235 -0
  191. package/src/cli/cmd/tui/context/theme/gold.json +235 -0
  192. package/src/cli/cmd/tui/context/theme/gone.json +234 -0
  193. package/src/cli/cmd/tui/context/theme/greyscale.json +229 -0
  194. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  195. package/src/cli/cmd/tui/context/theme/hacker.json +229 -0
  196. package/src/cli/cmd/tui/context/theme/holo.json +235 -0
  197. package/src/cli/cmd/tui/context/theme/ink.json +235 -0
  198. package/src/cli/cmd/tui/context/theme/jet.json +233 -0
  199. package/src/cli/cmd/tui/context/theme/kanagawa.json +227 -0
  200. package/src/cli/cmd/tui/context/theme/lavender.json +236 -0
  201. package/src/cli/cmd/tui/context/theme/lightph.json +235 -0
  202. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  203. package/src/cli/cmd/tui/context/theme/material-ocean.json +230 -0
  204. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  205. package/src/cli/cmd/tui/context/theme/matrix.json +227 -0
  206. package/src/cli/cmd/tui/context/theme/mercury.json +245 -0
  207. package/src/cli/cmd/tui/context/theme/midnight.json +235 -0
  208. package/src/cli/cmd/tui/context/theme/modern.json +235 -0
  209. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  210. package/src/cli/cmd/tui/context/theme/muted.json +229 -0
  211. package/src/cli/cmd/tui/context/theme/neon.json +229 -0
  212. package/src/cli/cmd/tui/context/theme/neonfusion.json +235 -0
  213. package/src/cli/cmd/tui/context/theme/neutral.json +235 -0
  214. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  215. package/src/cli/cmd/tui/context/theme/nikcli.json +245 -0
  216. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  217. package/src/cli/cmd/tui/context/theme/nordic.json +235 -0
  218. package/src/cli/cmd/tui/context/theme/nova.json +235 -0
  219. package/src/cli/cmd/tui/context/theme/obsidian.json +234 -0
  220. package/src/cli/cmd/tui/context/theme/one-dark.json +231 -0
  221. package/src/cli/cmd/tui/context/theme/one-pro.json +229 -0
  222. package/src/cli/cmd/tui/context/theme/onyx.json +233 -0
  223. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  224. package/src/cli/cmd/tui/context/theme/osaka-jade.json +240 -0
  225. package/src/cli/cmd/tui/context/theme/oxocarbon.json +229 -0
  226. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  227. package/src/cli/cmd/tui/context/theme/poimandres.json +230 -0
  228. package/src/cli/cmd/tui/context/theme/prism.json +235 -0
  229. package/src/cli/cmd/tui/context/theme/radiant.json +235 -0
  230. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  231. package/src/cli/cmd/tui/context/theme/shadow.json +235 -0
  232. package/src/cli/cmd/tui/context/theme/silicon.json +235 -0
  233. package/src/cli/cmd/tui/context/theme/slate.json +233 -0
  234. package/src/cli/cmd/tui/context/theme/soft.json +235 -0
  235. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  236. package/src/cli/cmd/tui/context/theme/spectrum.json +235 -0
  237. package/src/cli/cmd/tui/context/theme/starlight.json +233 -0
  238. package/src/cli/cmd/tui/context/theme/sunrise.json +235 -0
  239. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  240. package/src/cli/cmd/tui/context/theme/tech.json +235 -0
  241. package/src/cli/cmd/tui/context/theme/tokyonight-storm.json +245 -0
  242. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  243. package/src/cli/cmd/tui/context/theme/vapor.json +235 -0
  244. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  245. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  246. package/src/cli/cmd/tui/context/theme/vivid.json +232 -0
  247. package/src/cli/cmd/tui/context/theme/void.json +235 -0
  248. package/src/cli/cmd/tui/context/theme/vscode.json +235 -0
  249. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  250. package/src/cli/cmd/tui/context/theme/zinc.json +236 -0
  251. package/src/cli/cmd/tui/context/theme.tsx +1303 -0
  252. package/src/cli/cmd/tui/event.ts +48 -0
  253. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +152 -0
  254. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
  255. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +63 -0
  256. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
  257. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
  258. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
  259. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +96 -0
  260. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +48 -0
  261. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +288 -0
  262. package/src/cli/cmd/tui/plugin/api.tsx +407 -0
  263. package/src/cli/cmd/tui/plugin/index.ts +3 -0
  264. package/src/cli/cmd/tui/plugin/internal.ts +25 -0
  265. package/src/cli/cmd/tui/plugin/runtime.ts +1048 -0
  266. package/src/cli/cmd/tui/plugin/slots.tsx +61 -0
  267. package/src/cli/cmd/tui/routes/home.tsx +153 -0
  268. package/src/cli/cmd/tui/routes/session/dbedit.tsx +474 -0
  269. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +65 -0
  270. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +110 -0
  271. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +105 -0
  272. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  273. package/src/cli/cmd/tui/routes/session/footer.tsx +75 -0
  274. package/src/cli/cmd/tui/routes/session/header.tsx +177 -0
  275. package/src/cli/cmd/tui/routes/session/index.tsx +2280 -0
  276. package/src/cli/cmd/tui/routes/session/permission.tsx +540 -0
  277. package/src/cli/cmd/tui/routes/session/question.tsx +435 -0
  278. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  279. package/src/cli/cmd/tui/thread.ts +174 -0
  280. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  281. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  282. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  283. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  284. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +102 -0
  285. package/src/cli/cmd/tui/ui/dialog-select.tsx +389 -0
  286. package/src/cli/cmd/tui/ui/dialog.tsx +180 -0
  287. package/src/cli/cmd/tui/ui/link.tsx +34 -0
  288. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  289. package/src/cli/cmd/tui/ui/toast.tsx +138 -0
  290. package/src/cli/cmd/tui/util/clipboard.ts +154 -0
  291. package/src/cli/cmd/tui/util/editor.ts +32 -0
  292. package/src/cli/cmd/tui/util/signal.ts +7 -0
  293. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  294. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  295. package/src/cli/cmd/tui/win32.ts +110 -0
  296. package/src/cli/cmd/tui/worker.ts +156 -0
  297. package/src/cli/cmd/uninstall.ts +357 -0
  298. package/src/cli/cmd/upgrade.ts +72 -0
  299. package/src/cli/cmd/web.ts +87 -0
  300. package/src/cli/cmd/workspace-serve.ts +16 -0
  301. package/src/cli/error.ts +57 -0
  302. package/src/cli/network.ts +55 -0
  303. package/src/cli/remote/index.ts +36 -0
  304. package/src/cli/remote/notifications.ts +104 -0
  305. package/src/cli/remote/qr-renderer.ts +86 -0
  306. package/src/cli/remote/remote-service.ts +757 -0
  307. package/src/cli/remote/session-manager.ts +284 -0
  308. package/src/cli/remote/subagent-hooks.ts +151 -0
  309. package/src/cli/remote/types.ts +121 -0
  310. package/src/cli/ui.ts +96 -0
  311. package/src/cli/upgrade.ts +25 -0
  312. package/src/command/index.ts +174 -0
  313. package/src/command/template/initialize.txt +10 -0
  314. package/src/command/template/review.txt +99 -0
  315. package/src/config/config.ts +1760 -0
  316. package/src/config/markdown.ts +88 -0
  317. package/src/config/migrate-tui-config.ts +155 -0
  318. package/src/config/paths.ts +174 -0
  319. package/src/config/tui-schema.ts +36 -0
  320. package/src/config/tui.ts +209 -0
  321. package/src/connectors/api/base.ts +75 -0
  322. package/src/connectors/api/figma.ts +103 -0
  323. package/src/connectors/api/github.ts +247 -0
  324. package/src/connectors/api/lovable.ts +126 -0
  325. package/src/connectors/api/slack.ts +137 -0
  326. package/src/connectors/auth.ts +68 -0
  327. package/src/connectors/cache.ts +119 -0
  328. package/src/connectors/credentials.ts +81 -0
  329. package/src/connectors/index.ts +202 -0
  330. package/src/connectors/registry.ts +358 -0
  331. package/src/docs/context.ts +120 -0
  332. package/src/docs/library.ts +189 -0
  333. package/src/env/index.ts +26 -0
  334. package/src/file/ignore.ts +83 -0
  335. package/src/file/index.ts +411 -0
  336. package/src/file/ripgrep.ts +402 -0
  337. package/src/file/time.ts +65 -0
  338. package/src/file/watcher.ts +127 -0
  339. package/src/flag/flag.ts +128 -0
  340. package/src/format/formatter.ts +356 -0
  341. package/src/format/index.ts +137 -0
  342. package/src/global/index.ts +57 -0
  343. package/src/id/id.ts +83 -0
  344. package/src/ide/index.ts +76 -0
  345. package/src/index.ts +184 -0
  346. package/src/installation/index.ts +246 -0
  347. package/src/lsp/client.ts +250 -0
  348. package/src/lsp/index.ts +483 -0
  349. package/src/lsp/language.ts +119 -0
  350. package/src/lsp/server.ts +2046 -0
  351. package/src/mcp/auth.ts +121 -0
  352. package/src/mcp/index.ts +860 -0
  353. package/src/mcp/oauth-callback.ts +198 -0
  354. package/src/mcp/oauth-provider.ts +148 -0
  355. package/src/mobile/auth.ts +97 -0
  356. package/src/mobile/github-repo.ts +185 -0
  357. package/src/patch/index.ts +631 -0
  358. package/src/permission/arity.ts +150 -0
  359. package/src/permission/dbedit.ts +236 -0
  360. package/src/permission/index.ts +210 -0
  361. package/src/permission/next.ts +287 -0
  362. package/src/plugin/codex.ts +493 -0
  363. package/src/plugin/copilot.ts +261 -0
  364. package/src/plugin/index.ts +714 -0
  365. package/src/plugin/install.ts +379 -0
  366. package/src/plugin/meta.ts +165 -0
  367. package/src/plugin/shared.ts +188 -0
  368. package/src/project/bootstrap.ts +35 -0
  369. package/src/project/instance.ts +84 -0
  370. package/src/project/project.ts +373 -0
  371. package/src/project/state.ts +66 -0
  372. package/src/project/vcs.ts +76 -0
  373. package/src/prompt/stash-store.ts +93 -0
  374. package/src/provider/auth.ts +147 -0
  375. package/src/provider/models-macro.ts +22 -0
  376. package/src/provider/models.ts +216 -0
  377. package/src/provider/provider.ts +1483 -0
  378. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  379. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  380. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  381. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  382. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  383. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  384. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  385. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  386. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
  387. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  388. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  389. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  390. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  391. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  392. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  393. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  394. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  395. package/src/provider/transform.ts +828 -0
  396. package/src/pty/index.ts +241 -0
  397. package/src/question/index.ts +171 -0
  398. package/src/rag/chunk.ts +43 -0
  399. package/src/rag/embed.ts +179 -0
  400. package/src/rag/index.ts +376 -0
  401. package/src/rag/storage.ts +76 -0
  402. package/src/scheduler/index.ts +61 -0
  403. package/src/server/error.ts +36 -0
  404. package/src/server/event.ts +7 -0
  405. package/src/server/mdns.ts +59 -0
  406. package/src/server/routes/chatbot.ts +205 -0
  407. package/src/server/routes/companion.ts +729 -0
  408. package/src/server/routes/config.ts +92 -0
  409. package/src/server/routes/connectors.ts +121 -0
  410. package/src/server/routes/dbedit.ts +76 -0
  411. package/src/server/routes/experimental.ts +210 -0
  412. package/src/server/routes/file.ts +197 -0
  413. package/src/server/routes/global.ts +135 -0
  414. package/src/server/routes/mcp.ts +225 -0
  415. package/src/server/routes/mobile.ts +2044 -0
  416. package/src/server/routes/permission.ts +68 -0
  417. package/src/server/routes/project.ts +82 -0
  418. package/src/server/routes/provider.ts +235 -0
  419. package/src/server/routes/pty.ts +169 -0
  420. package/src/server/routes/question.ts +98 -0
  421. package/src/server/routes/session.ts +968 -0
  422. package/src/server/routes/tui.ts +379 -0
  423. package/src/server/routes/workspace.ts +104 -0
  424. package/src/server/server.ts +761 -0
  425. package/src/server/ssh.ts +207 -0
  426. package/src/session/auth.ts +402 -0
  427. package/src/session/compaction.ts +253 -0
  428. package/src/session/generate.ts +38 -0
  429. package/src/session/index.ts +598 -0
  430. package/src/session/llm.ts +273 -0
  431. package/src/session/message-v2.ts +836 -0
  432. package/src/session/message.ts +189 -0
  433. package/src/session/processor.ts +408 -0
  434. package/src/session/prompt/anthropic-20250930.txt +165 -0
  435. package/src/session/prompt/anthropic.txt +105 -0
  436. package/src/session/prompt/anthropic_spoof.txt +1 -0
  437. package/src/session/prompt/beast.txt +147 -0
  438. package/src/session/prompt/build-switch.txt +5 -0
  439. package/src/session/prompt/codex_header.txt +79 -0
  440. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  441. package/src/session/prompt/gemini.txt +155 -0
  442. package/src/session/prompt/max-steps.txt +16 -0
  443. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  444. package/src/session/prompt/plan.txt +25 -0
  445. package/src/session/prompt/qwen.txt +108 -0
  446. package/src/session/prompt.ts +1942 -0
  447. package/src/session/retry.ts +90 -0
  448. package/src/session/revert.ts +120 -0
  449. package/src/session/stats.ts +404 -0
  450. package/src/session/status.ts +84 -0
  451. package/src/session/summary.ts +184 -0
  452. package/src/session/system.ts +195 -0
  453. package/src/session/toast.tsx +105 -0
  454. package/src/session/todo.ts +258 -0
  455. package/src/session/uninstall.ts +357 -0
  456. package/src/share/share-next.ts +421 -0
  457. package/src/share/share.ts +92 -0
  458. package/src/shell/shell.ts +65 -0
  459. package/src/skill/index.ts +1 -0
  460. package/src/skill/skill.ts +232 -0
  461. package/src/snapshot/index.ts +297 -0
  462. package/src/storage/storage.ts +227 -0
  463. package/src/tool/apply_patch.ts +288 -0
  464. package/src/tool/apply_patch.txt +33 -0
  465. package/src/tool/bash.ts +252 -0
  466. package/src/tool/bash.txt +115 -0
  467. package/src/tool/batch.ts +175 -0
  468. package/src/tool/batch.txt +24 -0
  469. package/src/tool/codesearch.ts +132 -0
  470. package/src/tool/codesearch.txt +12 -0
  471. package/src/tool/context_collect.ts +152 -0
  472. package/src/tool/context_collect.txt +9 -0
  473. package/src/tool/context_diagnostics.ts +81 -0
  474. package/src/tool/context_diagnostics.txt +5 -0
  475. package/src/tool/context_related.ts +117 -0
  476. package/src/tool/context_related.txt +5 -0
  477. package/src/tool/context_search.ts +108 -0
  478. package/src/tool/context_search.txt +8 -0
  479. package/src/tool/db-diff.ts +434 -0
  480. package/src/tool/db-table.txt +15 -0
  481. package/src/tool/docs_add.ts +50 -0
  482. package/src/tool/docs_add.txt +5 -0
  483. package/src/tool/docs_context.ts +56 -0
  484. package/src/tool/docs_context.txt +4 -0
  485. package/src/tool/docs_gap_report.ts +79 -0
  486. package/src/tool/docs_gap_report.txt +7 -0
  487. package/src/tool/docs_load.ts +41 -0
  488. package/src/tool/docs_load.txt +4 -0
  489. package/src/tool/docs_request.ts +129 -0
  490. package/src/tool/docs_request.txt +7 -0
  491. package/src/tool/docs_search.ts +51 -0
  492. package/src/tool/docs_search.txt +6 -0
  493. package/src/tool/docs_unload.ts +38 -0
  494. package/src/tool/docs_unload.txt +5 -0
  495. package/src/tool/edit.ts +614 -0
  496. package/src/tool/edit.txt +10 -0
  497. package/src/tool/external-directory.ts +32 -0
  498. package/src/tool/generate_image.ts +174 -0
  499. package/src/tool/generate_image.txt +12 -0
  500. package/src/tool/glob.ts +79 -0
  501. package/src/tool/glob.txt +6 -0
  502. package/src/tool/grep.ts +153 -0
  503. package/src/tool/grep.txt +8 -0
  504. package/src/tool/invalid.ts +17 -0
  505. package/src/tool/ls.ts +116 -0
  506. package/src/tool/ls.txt +1 -0
  507. package/src/tool/lsp.ts +96 -0
  508. package/src/tool/lsp.txt +19 -0
  509. package/src/tool/memory_search.ts +141 -0
  510. package/src/tool/memory_search.txt +8 -0
  511. package/src/tool/multiedit.ts +46 -0
  512. package/src/tool/multiedit.txt +41 -0
  513. package/src/tool/plan-enter.txt +14 -0
  514. package/src/tool/plan-exit.txt +13 -0
  515. package/src/tool/plan.ts +130 -0
  516. package/src/tool/question.ts +33 -0
  517. package/src/tool/question.txt +10 -0
  518. package/src/tool/rag_index.ts +77 -0
  519. package/src/tool/rag_index.txt +10 -0
  520. package/src/tool/rag_reset.ts +26 -0
  521. package/src/tool/rag_reset.txt +4 -0
  522. package/src/tool/rag_search.ts +62 -0
  523. package/src/tool/rag_search.txt +6 -0
  524. package/src/tool/rag_status.ts +45 -0
  525. package/src/tool/rag_status.txt +4 -0
  526. package/src/tool/read.ts +203 -0
  527. package/src/tool/read.txt +12 -0
  528. package/src/tool/registry.ts +214 -0
  529. package/src/tool/skill.ts +169 -0
  530. package/src/tool/skill.txt +3 -0
  531. package/src/tool/smart_docs.ts +74 -0
  532. package/src/tool/smart_docs.txt +7 -0
  533. package/src/tool/speak/elevenlabs.ts +201 -0
  534. package/src/tool/speak/openrouter.ts +240 -0
  535. package/src/tool/speak/provider.ts +83 -0
  536. package/src/tool/speak.ts +440 -0
  537. package/src/tool/task.ts +194 -0
  538. package/src/tool/task.txt +60 -0
  539. package/src/tool/todo.ts +53 -0
  540. package/src/tool/todoread.txt +14 -0
  541. package/src/tool/todowrite.txt +167 -0
  542. package/src/tool/tool.ts +87 -0
  543. package/src/tool/tree.ts +218 -0
  544. package/src/tool/tree.txt +8 -0
  545. package/src/tool/truncation.ts +106 -0
  546. package/src/tool/use-connector.ts +47 -0
  547. package/src/tool/voice.ts +188 -0
  548. package/src/tool/webfetch.ts +205 -0
  549. package/src/tool/webfetch.txt +13 -0
  550. package/src/tool/websearch.ts +150 -0
  551. package/src/tool/websearch.txt +14 -0
  552. package/src/tool/write.ts +80 -0
  553. package/src/tool/write.txt +8 -0
  554. package/src/util/archive.ts +16 -0
  555. package/src/util/color.ts +19 -0
  556. package/src/util/context.ts +25 -0
  557. package/src/util/defer.ts +12 -0
  558. package/src/util/error.ts +77 -0
  559. package/src/util/eventloop.ts +20 -0
  560. package/src/util/filesystem.ts +125 -0
  561. package/src/util/flock.ts +329 -0
  562. package/src/util/fn.ts +11 -0
  563. package/src/util/format.ts +20 -0
  564. package/src/util/hash.ts +7 -0
  565. package/src/util/iife.ts +3 -0
  566. package/src/util/keybind.ts +103 -0
  567. package/src/util/lazy.ts +18 -0
  568. package/src/util/locale.ts +81 -0
  569. package/src/util/lock.ts +98 -0
  570. package/src/util/log.ts +180 -0
  571. package/src/util/network.ts +9 -0
  572. package/src/util/process.ts +15 -0
  573. package/src/util/queue.ts +32 -0
  574. package/src/util/record.ts +3 -0
  575. package/src/util/rpc.ts +66 -0
  576. package/src/util/scrap.ts +10 -0
  577. package/src/util/signal.ts +12 -0
  578. package/src/util/timeout.ts +14 -0
  579. package/src/util/token.ts +7 -0
  580. package/src/util/wildcard.ts +56 -0
  581. package/src/workspace/adaptors/index.ts +271 -0
  582. package/src/workspace/adaptors/types.ts +14 -0
  583. package/src/workspace/adaptors/worktree.ts +31 -0
  584. package/src/workspace/config.ts +19 -0
  585. package/src/workspace/index.ts +223 -0
  586. package/src/workspace/session-proxy-middleware.ts +97 -0
  587. package/src/workspace/sse.ts +66 -0
  588. package/src/workspace/workspace-context.ts +23 -0
  589. package/src/workspace/workspace-server/routes.ts +33 -0
  590. package/src/workspace/workspace-server/server.ts +47 -0
  591. package/src/worktree/index.ts +487 -0
  592. package/sst-env.d.ts +10 -0
  593. package/test/benchmark.test.ts +121 -0
  594. package/test/build-optimizations.test.ts +124 -0
  595. package/test/id-benchmark.test.ts +132 -0
  596. package/test/optimizations.test.ts +302 -0
  597. package/test/preload.ts +1 -0
  598. package/test/solidjs-benchmark.test.ts +262 -0
  599. package/test/solidjs-optimizations.test.ts +259 -0
  600. package/test/tui-benchmark.test.ts +230 -0
  601. package/test/wildcard-benchmark.test.ts +180 -0
  602. package/tsconfig.json +26 -0
@@ -0,0 +1,88 @@
1
+ import { NamedError } from "@nikcli-ai/util/error"
2
+ import matter from "gray-matter"
3
+ import { z } from "zod"
4
+
5
+ export namespace ConfigMarkdown {
6
+ export const FILE_REGEX = /(?<![\w`])@(\.?[^\s`,.]*(?:\.[^\s`,.]+)*)/g
7
+ export const SHELL_REGEX = /!`([^`]+)`/g
8
+
9
+ export function files(template: string) {
10
+ return Array.from(template.matchAll(FILE_REGEX))
11
+ }
12
+
13
+ export function shell(template: string) {
14
+ return Array.from(template.matchAll(SHELL_REGEX))
15
+ }
16
+
17
+ export function preprocessFrontmatter(content: string): string {
18
+ const match = content.match(/^---\r?\n([\s\S]*?)\r?\n---/)
19
+ if (!match) return content
20
+
21
+ const frontmatter = match[1]
22
+ const lines = frontmatter.split("\n")
23
+ const result: string[] = []
24
+
25
+ for (const line of lines) {
26
+ if (line.trim().startsWith("#") || line.trim() === "") {
27
+ result.push(line)
28
+ continue
29
+ }
30
+
31
+ if (line.match(/^\s+/)) {
32
+ result.push(line)
33
+ continue
34
+ }
35
+
36
+ const kvMatch = line.match(/^([a-zA-Z_][a-zA-Z0-9_]*)\s*:\s*(.*)$/)
37
+ if (!kvMatch) {
38
+ result.push(line)
39
+ continue
40
+ }
41
+
42
+ const key = kvMatch[1]
43
+ const value = kvMatch[2].trim()
44
+
45
+ if (value === "" || value === ">" || value === "|" || value.startsWith('"') || value.startsWith("'")) {
46
+ result.push(line)
47
+ continue
48
+ }
49
+
50
+ if (value.includes(":")) {
51
+ result.push(`${key}: |`)
52
+ result.push(` ${value}`)
53
+ continue
54
+ }
55
+
56
+ result.push(line)
57
+ }
58
+
59
+ const processed = result.join("\n")
60
+ return content.replace(frontmatter, () => processed)
61
+ }
62
+
63
+ export async function parse(filePath: string) {
64
+ const raw = await Bun.file(filePath).text()
65
+ const template = preprocessFrontmatter(raw)
66
+
67
+ try {
68
+ const md = matter(template)
69
+ return md
70
+ } catch (err) {
71
+ throw new FrontmatterError(
72
+ {
73
+ path: filePath,
74
+ message: `${filePath}: Failed to parse YAML frontmatter: ${err instanceof Error ? err.message : String(err)}`,
75
+ },
76
+ { cause: err },
77
+ )
78
+ }
79
+ }
80
+
81
+ export const FrontmatterError = NamedError.create(
82
+ "ConfigFrontmatterError",
83
+ z.object({
84
+ path: z.string(),
85
+ message: z.string(),
86
+ }),
87
+ )
88
+ }
@@ -0,0 +1,155 @@
1
+ import path from "path"
2
+ import { type ParseError as JsoncParseError, applyEdits, modify, parse as parseJsonc } from "jsonc-parser"
3
+ import { unique } from "remeda"
4
+ import z from "zod"
5
+ import { ConfigPaths } from "./paths"
6
+ import { TuiInfo, TuiOptions } from "./tui-schema"
7
+ import { Instance } from "@/project/instance"
8
+ import { Flag } from "@/flag/flag"
9
+ import { Log } from "@/util/log"
10
+ import { Filesystem } from "@/util/filesystem"
11
+ import { Global } from "@/global"
12
+
13
+ const log = Log.create({ service: "tui.migrate" })
14
+
15
+ const TUI_SCHEMA_URL = "https://opencode.ai/tui.json"
16
+
17
+ const LegacyTheme = TuiInfo.shape.theme.optional()
18
+ const LegacyRecord = z.record(z.string(), z.unknown()).optional()
19
+
20
+ const TuiLegacy = z
21
+ .object({
22
+ scroll_speed: TuiOptions.shape.scroll_speed.catch(undefined),
23
+ scroll_acceleration: TuiOptions.shape.scroll_acceleration.catch(undefined),
24
+ diff_style: TuiOptions.shape.diff_style.catch(undefined),
25
+ })
26
+ .strip()
27
+
28
+ interface MigrateInput {
29
+ directories: string[]
30
+ custom?: string
31
+ managed: string
32
+ }
33
+
34
+ /**
35
+ * Migrates tui-specific keys (theme, keybinds, tui) from nikcli.json files
36
+ * into dedicated tui.json files. Migration is performed per-directory and
37
+ * skips only locations where a tui.json already exists.
38
+ */
39
+ export async function migrateTuiConfig(input: MigrateInput) {
40
+ const nikcli = await nikcliFiles(input)
41
+ for (const file of nikcli) {
42
+ const source = await Filesystem.readText(file).catch((error) => {
43
+ log.warn("failed to read config for tui migration", { path: file, error })
44
+ return undefined
45
+ })
46
+ if (!source) continue
47
+ const errors: JsoncParseError[] = []
48
+ const data = parseJsonc(source, errors, { allowTrailingComma: true })
49
+ if (errors.length || !data || typeof data !== "object" || Array.isArray(data)) continue
50
+
51
+ const theme = LegacyTheme.safeParse("theme" in data ? data.theme : undefined)
52
+ const keybinds = LegacyRecord.safeParse("keybinds" in data ? data.keybinds : undefined)
53
+ const legacyTui = LegacyRecord.safeParse("tui" in data ? data.tui : undefined)
54
+ const extracted = {
55
+ theme: theme.success ? theme.data : undefined,
56
+ keybinds: keybinds.success ? keybinds.data : undefined,
57
+ tui: legacyTui.success ? legacyTui.data : undefined,
58
+ }
59
+ const tui = extracted.tui ? normalizeTui(extracted.tui) : undefined
60
+ if (extracted.theme === undefined && extracted.keybinds === undefined && !tui) continue
61
+
62
+ const target = path.join(path.dirname(file), "tui.json")
63
+ const targetExists = await Filesystem.exists(target)
64
+ if (targetExists) continue
65
+
66
+ const payload: Record<string, unknown> = {
67
+ $schema: TUI_SCHEMA_URL,
68
+ }
69
+ if (extracted.theme !== undefined) payload.theme = extracted.theme
70
+ if (extracted.keybinds !== undefined) payload.keybinds = extracted.keybinds
71
+ if (tui) Object.assign(payload, tui)
72
+
73
+ const wrote = await Filesystem.write(target, JSON.stringify(payload, null, 2))
74
+ .then(() => true)
75
+ .catch((error) => {
76
+ log.warn("failed to write tui migration target", { from: file, to: target, error })
77
+ return false
78
+ })
79
+ if (!wrote) continue
80
+
81
+ const stripped = await backupAndStripLegacy(file, source)
82
+ if (!stripped) {
83
+ log.warn("tui config migrated but source file was not stripped", { from: file, to: target })
84
+ continue
85
+ }
86
+ log.info("migrated tui config", { from: file, to: target })
87
+ }
88
+ }
89
+
90
+ function normalizeTui(data: Record<string, unknown>) {
91
+ const parsed = TuiLegacy.parse(data)
92
+ if (
93
+ parsed.scroll_speed === undefined &&
94
+ parsed.diff_style === undefined &&
95
+ parsed.scroll_acceleration === undefined
96
+ ) {
97
+ return
98
+ }
99
+ return parsed
100
+ }
101
+
102
+ async function backupAndStripLegacy(file: string, source: string) {
103
+ const backup = file + ".tui-migration.bak"
104
+ const hasBackup = await Filesystem.exists(backup)
105
+ const backed = hasBackup
106
+ ? true
107
+ : await Filesystem.write(backup, source)
108
+ .then(() => true)
109
+ .catch((error) => {
110
+ log.warn("failed to backup source config during tui migration", { path: file, backup, error })
111
+ return false
112
+ })
113
+ if (!backed) return false
114
+
115
+ const text = ["theme", "keybinds", "tui"].reduce((acc, key) => {
116
+ const edits = modify(acc, [key], undefined, {
117
+ formattingOptions: {
118
+ insertSpaces: true,
119
+ tabSize: 2,
120
+ },
121
+ })
122
+ if (!edits.length) return acc
123
+ return applyEdits(acc, edits)
124
+ }, source)
125
+
126
+ return Filesystem.write(file, text)
127
+ .then(() => {
128
+ log.info("stripped tui keys from server config", { path: file, backup })
129
+ return true
130
+ })
131
+ .catch((error) => {
132
+ log.warn("failed to strip legacy tui keys from server config", { path: file, backup, error })
133
+ return false
134
+ })
135
+ }
136
+
137
+ async function nikcliFiles(input: { directories: string[]; managed: string }) {
138
+ const project = Flag.NIKCLI_DISABLE_PROJECT_CONFIG
139
+ ? []
140
+ : await ConfigPaths.projectFiles("nikcli", Instance.directory, Instance.worktree)
141
+ const files = [...project, ...ConfigPaths.fileInDirectory(Global.Path.config, "nikcli")]
142
+ for (const dir of unique(input.directories)) {
143
+ files.push(...ConfigPaths.fileInDirectory(dir, "nikcli"))
144
+ }
145
+ if (Flag.NIKCLI_CONFIG) files.push(Flag.NIKCLI_CONFIG)
146
+ files.push(...ConfigPaths.fileInDirectory(input.managed, "nikcli"))
147
+
148
+ const existing = await Promise.all(
149
+ unique(files).map(async (file) => {
150
+ const ok = await Filesystem.exists(file)
151
+ return ok ? file : undefined
152
+ }),
153
+ )
154
+ return existing.filter((file): file is string => !!file)
155
+ }
@@ -0,0 +1,174 @@
1
+ import path from "path"
2
+ import os from "os"
3
+ import z from "zod"
4
+ import { type ParseError as JsoncParseError, parse as parseJsonc, printParseErrorCode } from "jsonc-parser"
5
+ import { NamedError } from "@nikcli-ai/util/error"
6
+ import { Filesystem } from "@/util/filesystem"
7
+ import { Flag } from "@/flag/flag"
8
+ import { Global } from "@/global"
9
+
10
+ export namespace ConfigPaths {
11
+ export async function projectFiles(name: string, directory: string, worktree: string) {
12
+ const files: string[] = []
13
+ for (const file of [`${name}.jsonc`, `${name}.json`]) {
14
+ const found = await Filesystem.findUp(file, directory, worktree)
15
+ for (const resolved of found.toReversed()) {
16
+ files.push(resolved)
17
+ }
18
+ }
19
+ return files
20
+ }
21
+
22
+ export async function directories(directory: string, worktree: string) {
23
+ return [
24
+ Global.Path.config,
25
+ ...(!Flag.NIKCLI_DISABLE_PROJECT_CONFIG
26
+ ? await Array.fromAsync(
27
+ Filesystem.up({
28
+ targets: [".nikcli"],
29
+ start: directory,
30
+ stop: worktree,
31
+ }),
32
+ )
33
+ : []),
34
+ ...(await Array.fromAsync(
35
+ Filesystem.up({
36
+ targets: [".nikcli"],
37
+ start: Global.Path.home,
38
+ stop: Global.Path.home,
39
+ }),
40
+ )),
41
+ ...(Flag.NIKCLI_CONFIG_DIR ? [Flag.NIKCLI_CONFIG_DIR] : []),
42
+ ]
43
+ }
44
+
45
+ export function fileInDirectory(dir: string, name: string) {
46
+ return [path.join(dir, `${name}.jsonc`), path.join(dir, `${name}.json`)]
47
+ }
48
+
49
+ export const JsonError = NamedError.create(
50
+ "ConfigJsonError",
51
+ z.object({
52
+ path: z.string(),
53
+ message: z.string().optional(),
54
+ }),
55
+ )
56
+
57
+ export const InvalidError = NamedError.create(
58
+ "ConfigInvalidError",
59
+ z.object({
60
+ path: z.string(),
61
+ issues: z.custom<z.core.$ZodIssue[]>().optional(),
62
+ message: z.string().optional(),
63
+ }),
64
+ )
65
+
66
+ /** Read a config file, returning undefined for missing files and throwing JsonError for other failures. */
67
+ export async function readFile(filepath: string) {
68
+ return Filesystem.readText(filepath).catch((err: NodeJS.ErrnoException) => {
69
+ if (err.code === "ENOENT") return
70
+ throw new JsonError({ path: filepath }, { cause: err })
71
+ })
72
+ }
73
+
74
+ type ParseSource = string | { source: string; dir: string }
75
+
76
+ function source(input: ParseSource) {
77
+ return typeof input === "string" ? input : input.source
78
+ }
79
+
80
+ function dir(input: ParseSource) {
81
+ return typeof input === "string" ? path.dirname(input) : input.dir
82
+ }
83
+
84
+ /** Apply {env:VAR} and {file:path} substitutions to config text. */
85
+ async function substitute(text: string, input: ParseSource, missing: "error" | "empty" = "error") {
86
+ text = text.replace(/\{env:([^}]+)\}/g, (_, varName) => {
87
+ return process.env[varName] || ""
88
+ })
89
+
90
+ const fileMatches = Array.from(text.matchAll(/\{file:[^}]+\}/g))
91
+ if (!fileMatches.length) return text
92
+
93
+ const configDir = dir(input)
94
+ const configSource = source(input)
95
+ let out = ""
96
+ let cursor = 0
97
+
98
+ for (const match of fileMatches) {
99
+ const token = match[0]
100
+ const index = match.index!
101
+ out += text.slice(cursor, index)
102
+
103
+ const lineStart = text.lastIndexOf("\n", index - 1) + 1
104
+ const prefix = text.slice(lineStart, index).trimStart()
105
+ if (prefix.startsWith("//")) {
106
+ out += token
107
+ cursor = index + token.length
108
+ continue
109
+ }
110
+
111
+ let filePath = token.replace(/^\{file:/, "").replace(/\}$/, "")
112
+ if (filePath.startsWith("~/")) {
113
+ filePath = path.join(os.homedir(), filePath.slice(2))
114
+ }
115
+
116
+ const resolvedPath = path.isAbsolute(filePath) ? filePath : path.resolve(configDir, filePath)
117
+ const fileContent = (
118
+ await Filesystem.readText(resolvedPath).catch((error: NodeJS.ErrnoException) => {
119
+ if (missing === "empty") return ""
120
+
121
+ const errMsg = `bad file reference: "${token}"`
122
+ if (error.code === "ENOENT") {
123
+ throw new InvalidError(
124
+ {
125
+ path: configSource,
126
+ message: errMsg + ` ${resolvedPath} does not exist`,
127
+ },
128
+ { cause: error },
129
+ )
130
+ }
131
+ throw new InvalidError({ path: configSource, message: errMsg }, { cause: error })
132
+ })
133
+ ).trim()
134
+
135
+ out += JSON.stringify(fileContent).slice(1, -1)
136
+ cursor = index + token.length
137
+ }
138
+
139
+ out += text.slice(cursor)
140
+ return out
141
+ }
142
+
143
+ /** Substitute and parse JSONC text, throwing JsonError on syntax errors. */
144
+ export async function parseText(text: string, input: ParseSource, missing: "error" | "empty" = "error") {
145
+ const configSource = source(input)
146
+ text = await substitute(text, input, missing)
147
+
148
+ const errors: JsoncParseError[] = []
149
+ const data = parseJsonc(text, errors, { allowTrailingComma: true })
150
+ if (errors.length) {
151
+ const lines = text.split("\n")
152
+ const errorDetails = errors
153
+ .map((e) => {
154
+ const beforeOffset = text.substring(0, e.offset).split("\n")
155
+ const line = beforeOffset.length
156
+ const column = beforeOffset[beforeOffset.length - 1].length + 1
157
+ const problemLine = lines[line - 1]
158
+
159
+ const error = `${printParseErrorCode(e.error)} at line ${line}, column ${column}`
160
+ if (!problemLine) return error
161
+
162
+ return `${error}\n Line ${line}: ${problemLine}\n${"".padStart(column + 9)}^`
163
+ })
164
+ .join("\n")
165
+
166
+ throw new JsonError({
167
+ path: configSource,
168
+ message: `\n--- JSONC Input ---\n${text}\n--- Errors ---\n${errorDetails}\n--- End ---`,
169
+ })
170
+ }
171
+
172
+ return data
173
+ }
174
+ }
@@ -0,0 +1,36 @@
1
+ import z from "zod"
2
+ import { Config } from "./config"
3
+
4
+ const KeybindOverride = z
5
+ .object(
6
+ Object.fromEntries(Object.keys(Config.Keybinds.shape).map((key) => [key, z.string().optional()])) as Record<
7
+ string,
8
+ z.ZodOptional<z.ZodString>
9
+ >,
10
+ )
11
+ .strict()
12
+
13
+ export const TuiOptions = z.object({
14
+ scroll_speed: z.number().min(0.001).optional().describe("TUI scroll speed"),
15
+ scroll_acceleration: z
16
+ .object({
17
+ enabled: z.boolean().describe("Enable scroll acceleration"),
18
+ })
19
+ .optional()
20
+ .describe("Scroll acceleration settings"),
21
+ diff_style: z
22
+ .enum(["auto", "stacked"])
23
+ .optional()
24
+ .describe("Control diff rendering style: 'auto' adapts to terminal width, 'stacked' always shows single column"),
25
+ })
26
+
27
+ export const TuiInfo = z
28
+ .object({
29
+ $schema: z.string().optional(),
30
+ theme: z.string().optional(),
31
+ keybinds: KeybindOverride.optional(),
32
+ plugin: Config.PluginSpec.array().optional(),
33
+ plugin_enabled: z.record(z.string(), z.boolean()).optional(),
34
+ })
35
+ .extend(TuiOptions.shape)
36
+ .strict()
@@ -0,0 +1,209 @@
1
+ import { existsSync } from "fs"
2
+ import z from "zod"
3
+ import { mergeDeep, unique } from "remeda"
4
+ import { Config } from "./config"
5
+ import { ConfigPaths } from "./paths"
6
+ import { migrateTuiConfig } from "./migrate-tui-config"
7
+ import { TuiInfo } from "./tui-schema"
8
+ import { Instance } from "@/project/instance"
9
+ import { Flag } from "@/flag/flag"
10
+ import { Log } from "@/util/log"
11
+ import { isRecord } from "@/util/record"
12
+ import { Global } from "@/global"
13
+ import { parsePluginSpecifier } from "@/plugin/shared"
14
+
15
+ export namespace TuiConfig {
16
+ const log = Log.create({ service: "tui.config" })
17
+
18
+ export const Info = TuiInfo
19
+
20
+ export type PluginMeta = {
21
+ scope: "global" | "local"
22
+ source: string
23
+ }
24
+
25
+ type PluginEntry = {
26
+ item: Config.PluginSpec
27
+ meta: PluginMeta
28
+ }
29
+
30
+ type Acc = {
31
+ result: Info
32
+ entries: PluginEntry[]
33
+ }
34
+
35
+ export type Info = z.output<typeof Info> & {
36
+ plugin_meta?: Record<string, PluginMeta>
37
+ }
38
+
39
+ function pluginScope(file: string): PluginMeta["scope"] {
40
+ if (Instance.containsPath(file)) return "local"
41
+ return "global"
42
+ }
43
+
44
+ function dedupePlugins(list: PluginEntry[]) {
45
+ const seen = new Set<string>()
46
+ const result: PluginEntry[] = []
47
+ for (const item of list.toReversed()) {
48
+ const spec = Config.pluginSpecifier(item.item)
49
+ const name = spec.startsWith("file://") ? spec : parsePluginSpecifier(spec).pkg
50
+ if (seen.has(name)) continue
51
+ seen.add(name)
52
+ result.push(item)
53
+ }
54
+ return result.toReversed()
55
+ }
56
+
57
+ function mergeInfo(target: Info, source: Info): Info {
58
+ const merged = mergeDeep(target, source)
59
+ return merged
60
+ }
61
+
62
+ function customPath() {
63
+ return Flag.NIKCLI_TUI_CONFIG
64
+ }
65
+
66
+ function normalize(raw: Record<string, unknown>) {
67
+ const data = { ...raw }
68
+ if (!("tui" in data)) return data
69
+ if (!isRecord(data.tui)) {
70
+ delete data.tui
71
+ return data
72
+ }
73
+
74
+ const tui = data.tui
75
+ delete data.tui
76
+ return {
77
+ ...tui,
78
+ ...data,
79
+ }
80
+ }
81
+
82
+ function installDeps(dir: string): Promise<void> {
83
+ return Config.installDependencies(dir)
84
+ }
85
+
86
+ async function mergeFile(acc: Acc, file: string) {
87
+ const data = await loadFile(file)
88
+ acc.result = mergeInfo(acc.result, data)
89
+ if (!data.plugin?.length) return
90
+
91
+ const scope = pluginScope(file)
92
+ for (const item of data.plugin) {
93
+ acc.entries.push({
94
+ item,
95
+ meta: {
96
+ scope,
97
+ source: file,
98
+ },
99
+ })
100
+ }
101
+ }
102
+
103
+ const state = Instance.state(async () => {
104
+ let projectFiles = Flag.NIKCLI_DISABLE_PROJECT_CONFIG
105
+ ? []
106
+ : await ConfigPaths.projectFiles("tui", Instance.directory, Instance.worktree)
107
+ const directories = await ConfigPaths.directories(Instance.directory, Instance.worktree)
108
+ const custom = customPath()
109
+ const managed = Config.managedConfigDir()
110
+ await migrateTuiConfig({ directories, custom, managed })
111
+ // Re-compute after migration since migrateTuiConfig may have created new tui.json files
112
+ projectFiles = Flag.NIKCLI_DISABLE_PROJECT_CONFIG
113
+ ? []
114
+ : await ConfigPaths.projectFiles("tui", Instance.directory, Instance.worktree)
115
+
116
+ const acc: Acc = {
117
+ result: {},
118
+ entries: [],
119
+ }
120
+
121
+ for (const file of ConfigPaths.fileInDirectory(Global.Path.config, "tui")) {
122
+ await mergeFile(acc, file)
123
+ }
124
+
125
+ if (custom) {
126
+ await mergeFile(acc, custom)
127
+ log.debug("loaded custom tui config", { path: custom })
128
+ }
129
+
130
+ for (const file of projectFiles) {
131
+ await mergeFile(acc, file)
132
+ }
133
+
134
+ for (const dir of unique(directories)) {
135
+ if (!dir.endsWith(".nikcli") && dir !== Flag.NIKCLI_CONFIG_DIR) continue
136
+ for (const file of ConfigPaths.fileInDirectory(dir, "tui")) {
137
+ await mergeFile(acc, file)
138
+ }
139
+ }
140
+
141
+ if (existsSync(managed)) {
142
+ for (const file of ConfigPaths.fileInDirectory(managed, "tui")) {
143
+ await mergeFile(acc, file)
144
+ }
145
+ }
146
+
147
+ const merged = dedupePlugins(acc.entries)
148
+ acc.result.keybinds = Config.Keybinds.parse(acc.result.keybinds ?? {})
149
+ acc.result.plugin = merged.map((item) => item.item)
150
+ acc.result.plugin_meta = merged.length
151
+ ? Object.fromEntries(merged.map((item) => [Config.pluginSpecifier(item.item), item.meta]))
152
+ : undefined
153
+
154
+ const deps: Promise<void>[] = []
155
+ if (acc.result.plugin?.length) {
156
+ for (const dir of unique(directories)) {
157
+ if (!dir.endsWith(".nikcli") && dir !== Flag.NIKCLI_CONFIG_DIR) continue
158
+ deps.push(installDeps(dir))
159
+ }
160
+ }
161
+
162
+ return {
163
+ config: acc.result,
164
+ deps,
165
+ }
166
+ })
167
+
168
+ export async function get() {
169
+ return state().then((x) => x.config)
170
+ }
171
+
172
+ export async function waitForDependencies() {
173
+ const deps = await state().then((x) => x.deps)
174
+ await Promise.all(deps)
175
+ }
176
+
177
+ async function loadFile(filepath: string): Promise<Info> {
178
+ const text = await ConfigPaths.readFile(filepath)
179
+ if (!text) return {}
180
+ return load(text, filepath).catch((error) => {
181
+ log.warn("failed to load tui config", { path: filepath, error })
182
+ return {}
183
+ })
184
+ }
185
+
186
+ async function load(text: string, configFilepath: string): Promise<Info> {
187
+ const raw = await ConfigPaths.parseText(text, configFilepath, "empty")
188
+ if (!isRecord(raw)) return {}
189
+
190
+ // Flatten a nested "tui" key so users who wrote `{ "tui": { ... } }` inside tui.json
191
+ // (mirroring the old nikcli.json shape) still get their settings applied.
192
+ const normalized = normalize(raw)
193
+
194
+ const parsed = Info.safeParse(normalized)
195
+ if (!parsed.success) {
196
+ log.warn("invalid tui config", { path: configFilepath, issues: parsed.error.issues })
197
+ return {}
198
+ }
199
+
200
+ const data = parsed.data
201
+ if (data.plugin) {
202
+ for (let i = 0; i < data.plugin.length; i++) {
203
+ data.plugin[i] = await Config.resolvePluginSpec(data.plugin[i], configFilepath)
204
+ }
205
+ }
206
+
207
+ return data
208
+ }
209
+ }