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,56 @@
1
+ import { sortBy, pipe } from "remeda"
2
+
3
+ export namespace Wildcard {
4
+ export function match(str: string, pattern: string) {
5
+ let escaped = pattern
6
+ .replace(/[.+^${}()|[\]\\]/g, "\\$&") // escape special regex chars
7
+ .replace(/\*/g, ".*") // * becomes .*
8
+ .replace(/\?/g, ".") // ? becomes .
9
+
10
+ // If pattern ends with " *" (space + wildcard), make the trailing part optional
11
+ // This allows "ls *" to match both "ls" and "ls -la"
12
+ if (escaped.endsWith(" .*")) {
13
+ escaped = escaped.slice(0, -3) + "( .*)?"
14
+ }
15
+
16
+ return new RegExp("^" + escaped + "$", "s").test(str)
17
+ }
18
+
19
+ export function all(input: string, patterns: Record<string, any>) {
20
+ const sorted = pipe(patterns, Object.entries, sortBy([([key]) => key.length, "asc"], [([key]) => key, "asc"]))
21
+ let result = undefined
22
+ for (const [pattern, value] of sorted) {
23
+ if (match(input, pattern)) {
24
+ result = value
25
+ continue
26
+ }
27
+ }
28
+ return result
29
+ }
30
+
31
+ export function allStructured(input: { head: string; tail: string[] }, patterns: Record<string, any>) {
32
+ const sorted = pipe(patterns, Object.entries, sortBy([([key]) => key.length, "asc"], [([key]) => key, "asc"]))
33
+ let result = undefined
34
+ for (const [pattern, value] of sorted) {
35
+ const parts = pattern.split(/\s+/)
36
+ if (!match(input.head, parts[0])) continue
37
+ if (parts.length === 1 || matchSequence(input.tail, parts.slice(1))) {
38
+ result = value
39
+ continue
40
+ }
41
+ }
42
+ return result
43
+ }
44
+
45
+ function matchSequence(items: string[], patterns: string[]): boolean {
46
+ if (patterns.length === 0) return true
47
+ const [pattern, ...rest] = patterns
48
+ if (pattern === "*") return matchSequence(items, rest)
49
+ for (let i = 0; i < items.length; i++) {
50
+ if (match(items[i], pattern) && matchSequence(items.slice(i + 1), rest)) {
51
+ return true
52
+ }
53
+ }
54
+ return false
55
+ }
56
+ }
@@ -0,0 +1,271 @@
1
+ import fs from "fs/promises"
2
+ import path from "path"
3
+ import { createServer } from "node:net"
4
+ import { fileURLToPath } from "node:url"
5
+ import { Global } from "@/global"
6
+ import type { Config } from "../config"
7
+ import type { Adaptor } from "./types"
8
+ import { WorktreeAdaptor } from "./worktree"
9
+
10
+ type ContainerConfig = Extract<Config, { type: "container" }>
11
+
12
+ const CONFIGURED_CONTAINER_IMAGE = process.env.NIKCLI_MOBILE_CONTAINER_IMAGE?.trim()
13
+ const LOCAL_CONTAINER_IMAGE = "nikcli-mobile-workspace:local"
14
+ const LOCAL_CONTAINER_DOCKERFILE = fileURLToPath(new URL("../../../../../Dockerfile.serve", import.meta.url))
15
+ const LOCAL_CONTAINER_CONTEXT = path.dirname(LOCAL_CONTAINER_DOCKERFILE)
16
+
17
+ type ContainerImageSource =
18
+ | {
19
+ image: string
20
+ source: "configured"
21
+ }
22
+ | {
23
+ image: string
24
+ source: "local-build"
25
+ dockerfile: string
26
+ context: string
27
+ }
28
+
29
+ type ContainerRuntimeInfo = {
30
+ available: boolean
31
+ runtime?: ContainerConfig["runtime"]
32
+ image: string
33
+ }
34
+
35
+ async function hasLocalContainerBuildContext() {
36
+ return fs
37
+ .stat(LOCAL_CONTAINER_DOCKERFILE)
38
+ .then((entry) => entry.isFile())
39
+ .catch(() => false)
40
+ }
41
+
42
+ async function resolveContainerImageSource(): Promise<ContainerImageSource | undefined> {
43
+ if (CONFIGURED_CONTAINER_IMAGE) {
44
+ return {
45
+ image: CONFIGURED_CONTAINER_IMAGE,
46
+ source: "configured",
47
+ }
48
+ }
49
+
50
+ if (await hasLocalContainerBuildContext()) {
51
+ return {
52
+ image: LOCAL_CONTAINER_IMAGE,
53
+ source: "local-build",
54
+ dockerfile: LOCAL_CONTAINER_DOCKERFILE,
55
+ context: LOCAL_CONTAINER_CONTEXT,
56
+ }
57
+ }
58
+
59
+ return undefined
60
+ }
61
+
62
+ function containerRuntime(): ContainerConfig["runtime"] | undefined {
63
+ if (Bun.which("docker")) return "docker"
64
+ if (Bun.which("podman")) return "podman"
65
+ return undefined
66
+ }
67
+
68
+ async function isRuntimeHealthy(runtime: ContainerConfig["runtime"]) {
69
+ const result = await runContainerCommand(runtime, ["info"], true)
70
+ return result.exitCode === 0
71
+ }
72
+
73
+ async function ensureContainerImage(runtime: ContainerConfig["runtime"], source: ContainerImageSource) {
74
+ if (source.source === "configured") return source.image
75
+
76
+ const inspect = await runContainerCommand(runtime, ["image", "inspect", source.image], true)
77
+ if (inspect.exitCode === 0) return source.image
78
+
79
+ await runContainerCommand(runtime, ["build", "-f", source.dockerfile, "-t", source.image, source.context])
80
+ return source.image
81
+ }
82
+
83
+ export async function getContainerRuntimeInfo(): Promise<ContainerRuntimeInfo> {
84
+ const runtime = containerRuntime()
85
+ const imageSource = await resolveContainerImageSource()
86
+
87
+ if (!runtime) {
88
+ return {
89
+ available: false,
90
+ runtime: undefined,
91
+ image: imageSource?.image ?? LOCAL_CONTAINER_IMAGE,
92
+ }
93
+ }
94
+
95
+ const healthy = await isRuntimeHealthy(runtime)
96
+
97
+ return {
98
+ available: healthy && Boolean(imageSource),
99
+ runtime,
100
+ image: imageSource?.image ?? LOCAL_CONTAINER_IMAGE,
101
+ }
102
+ }
103
+
104
+ async function reservePort(): Promise<number> {
105
+ return new Promise((resolve, reject) => {
106
+ const server = createServer()
107
+ server.unref()
108
+ server.once("error", reject)
109
+ server.listen(0, "127.0.0.1", () => {
110
+ const address = server.address()
111
+ if (!address || typeof address === "string") {
112
+ server.close()
113
+ reject(new Error("Failed to reserve container workspace port"))
114
+ return
115
+ }
116
+ server.close((error) => {
117
+ if (error) {
118
+ reject(error)
119
+ return
120
+ }
121
+ resolve(address.port)
122
+ })
123
+ })
124
+ })
125
+ }
126
+
127
+ async function runContainerCommand(runtime: ContainerConfig["runtime"], args: string[], allowFailure = false) {
128
+ const proc = Bun.spawn([runtime, ...args], {
129
+ stdout: "pipe",
130
+ stderr: "pipe",
131
+ })
132
+
133
+ const [exitCode, stdout, stderr] = await Promise.all([
134
+ proc.exited,
135
+ new Response(proc.stdout).text(),
136
+ new Response(proc.stderr).text(),
137
+ ])
138
+
139
+ if (exitCode !== 0 && !allowFailure) {
140
+ throw new Error(stderr.trim() || stdout.trim() || `${runtime} ${args.join(" ")} failed`)
141
+ }
142
+
143
+ return {
144
+ exitCode,
145
+ stdout: stdout.trim(),
146
+ stderr: stderr.trim(),
147
+ }
148
+ }
149
+
150
+ function sanitizeContainerName(input: string) {
151
+ return input.toLowerCase().replace(/[^a-z0-9_.-]+/g, "-")
152
+ }
153
+
154
+ async function waitForContainerServer(config: ContainerConfig) {
155
+ const deadline = Date.now() + 30_000
156
+ while (Date.now() < deadline) {
157
+ const controller = new AbortController()
158
+ const timeout = setTimeout(() => controller.abort(), 1_500)
159
+ try {
160
+ const response = await fetch(new URL("/event", config.serverUrl), {
161
+ headers: {
162
+ "x-nikcli-directory": config.directory,
163
+ },
164
+ signal: controller.signal,
165
+ })
166
+ await response.body?.cancel().catch(() => undefined)
167
+ if (response.ok) return
168
+ } catch {
169
+ // wait and retry while the container starts
170
+ } finally {
171
+ clearTimeout(timeout)
172
+ }
173
+ await Bun.sleep(500)
174
+ }
175
+
176
+ throw new Error(`Workspace container ${config.containerName} did not become ready in time`)
177
+ }
178
+
179
+ function proxyHeaders(config: ContainerConfig, headers?: HeadersInit) {
180
+ const next = new Headers(headers)
181
+ next.delete("host")
182
+ next.delete("content-length")
183
+ next.delete("x-nikcli-workspace")
184
+ if (!next.has("x-nikcli-directory")) next.set("x-nikcli-directory", config.directory)
185
+ return next
186
+ }
187
+
188
+ const ContainerAdaptor: Adaptor<ContainerConfig> = {
189
+ async create(from, _branch, workspaceID) {
190
+ const runtimeInfo = await getContainerRuntimeInfo()
191
+ if (!runtimeInfo.available || !runtimeInfo.runtime) {
192
+ throw new Error(
193
+ "Container sandbox requires a healthy Docker or Podman runtime and a compatible Nikcli workspace image",
194
+ )
195
+ }
196
+ const runtime = runtimeInfo.runtime
197
+ const imageSource = await resolveContainerImageSource()
198
+ if (!imageSource) throw new Error("No Nikcli workspace container image is configured")
199
+ const image = await ensureContainerImage(runtime, imageSource)
200
+
201
+ const port = await reservePort()
202
+ const containerName = sanitizeContainerName(workspaceID || `nikcli-mobile-${port}`)
203
+ const config: ContainerConfig = {
204
+ type: "container",
205
+ directory: from.directory,
206
+ runtime,
207
+ image: from.image || image,
208
+ containerName,
209
+ port,
210
+ serverUrl: `http://127.0.0.1:${port}`,
211
+ }
212
+
213
+ return {
214
+ config,
215
+ init: async () => {
216
+ await runContainerCommand(runtime, ["rm", "-f", containerName], true)
217
+ await runContainerCommand(runtime, [
218
+ "run",
219
+ "-d",
220
+ "--init",
221
+ "--name",
222
+ containerName,
223
+ "-p",
224
+ `127.0.0.1:${port}:${port}`,
225
+ "-v",
226
+ `${config.directory}:${config.directory}`,
227
+ "-v",
228
+ `${Global.Path.data}:${Global.Path.data}`,
229
+ "-v",
230
+ `${Global.Path.config}:${Global.Path.config}`,
231
+ "-v",
232
+ `${Global.Path.state}:${Global.Path.state}`,
233
+ "-w",
234
+ config.directory,
235
+ config.image,
236
+ "workspace-serve",
237
+ "--hostname",
238
+ "0.0.0.0",
239
+ "--port",
240
+ String(port),
241
+ ])
242
+ try {
243
+ await waitForContainerServer(config)
244
+ } catch (error) {
245
+ await runContainerCommand(runtime, ["rm", "-f", containerName], true)
246
+ throw error
247
+ }
248
+ },
249
+ }
250
+ },
251
+ async remove(config) {
252
+ await runContainerCommand(config.runtime, ["rm", "-f", config.containerName], true)
253
+ },
254
+ async request(config, method, url, data, signal, headers) {
255
+ return fetch(new URL(url, config.serverUrl), {
256
+ method,
257
+ body: data,
258
+ headers: proxyHeaders(config, headers),
259
+ signal,
260
+ })
261
+ },
262
+ }
263
+
264
+ export function getAdaptor(config: Config): Adaptor {
265
+ switch (config.type) {
266
+ case "worktree":
267
+ return WorktreeAdaptor
268
+ case "container":
269
+ return ContainerAdaptor
270
+ }
271
+ }
@@ -0,0 +1,14 @@
1
+ import type { Config } from "../config"
2
+
3
+ export type Adaptor<T extends Config = Config> = {
4
+ create(from: T, branch?: string | null, workspaceID?: string): Promise<{ config: T; init: () => Promise<void> }>
5
+ remove(from: T): Promise<void>
6
+ request(
7
+ from: T,
8
+ method: string,
9
+ url: string,
10
+ data?: BodyInit,
11
+ signal?: AbortSignal,
12
+ headers?: HeadersInit,
13
+ ): Promise<Response | undefined>
14
+ }
@@ -0,0 +1,31 @@
1
+ import { Worktree } from "@/worktree"
2
+ import type { Config } from "../config"
3
+ import type { Adaptor } from "./types"
4
+
5
+ type WorktreeConfig = Extract<Config, { type: "worktree" }>
6
+
7
+ export const WorktreeAdaptor: Adaptor<WorktreeConfig> = {
8
+ async create(_from: WorktreeConfig, branch: string | null | undefined, _workspaceID?: string) {
9
+ const next = await Worktree.create(branch ? { branch } : undefined)
10
+ return {
11
+ config: {
12
+ type: "worktree",
13
+ directory: next.directory,
14
+ },
15
+ init: async () => {},
16
+ }
17
+ },
18
+ async remove(config: WorktreeConfig) {
19
+ await Worktree.remove({ directory: config.directory })
20
+ },
21
+ async request(
22
+ _from: WorktreeConfig,
23
+ _method: string,
24
+ _url: string,
25
+ _data?: BodyInit,
26
+ _signal?: AbortSignal,
27
+ _headers?: HeadersInit,
28
+ ) {
29
+ throw new Error("worktree does not support request")
30
+ },
31
+ }
@@ -0,0 +1,19 @@
1
+ import z from "zod"
2
+
3
+ export const Config = z.discriminatedUnion("type", [
4
+ z.object({
5
+ directory: z.string(),
6
+ type: z.literal("worktree"),
7
+ }),
8
+ z.object({
9
+ directory: z.string(),
10
+ type: z.literal("container"),
11
+ runtime: z.enum(["docker", "podman"]),
12
+ image: z.string(),
13
+ containerName: z.string(),
14
+ port: z.number().int().positive(),
15
+ serverUrl: z.string().url(),
16
+ }),
17
+ ])
18
+
19
+ export type Config = z.infer<typeof Config>
@@ -0,0 +1,223 @@
1
+ import z from "zod"
2
+ import { BusEvent } from "@/bus/bus-event"
3
+ import { GlobalBus } from "@/bus/global"
4
+ import { Identifier } from "@/id/id"
5
+ import { PermissionNext } from "@/permission/next"
6
+ import { Project } from "@/project/project"
7
+ import { InstanceBootstrap } from "@/project/bootstrap"
8
+ import { Instance } from "@/project/instance"
9
+ import { SessionStatus } from "@/session/status"
10
+ import { Storage } from "@/storage/storage"
11
+ import { fn } from "@/util/fn"
12
+ import { Log } from "@/util/log"
13
+ import { getAdaptor } from "./adaptors"
14
+ import { Config } from "./config"
15
+ import { parseSSE } from "./sse"
16
+
17
+ export namespace Workspace {
18
+ export const Event = {
19
+ Ready: BusEvent.define(
20
+ "workspace.ready",
21
+ z.object({
22
+ name: z.string(),
23
+ }),
24
+ ),
25
+ Failed: BusEvent.define(
26
+ "workspace.failed",
27
+ z.object({
28
+ message: z.string(),
29
+ }),
30
+ ),
31
+ }
32
+
33
+ export const Info = z
34
+ .object({
35
+ id: Identifier.schema("workspace"),
36
+ branch: z.string().nullable(),
37
+ projectID: z.string(),
38
+ config: Config,
39
+ })
40
+ .meta({
41
+ ref: "Workspace",
42
+ })
43
+ export type Info = z.infer<typeof Info>
44
+
45
+ function fromStorage(row: Partial<Info>): Info {
46
+ return Info.parse({
47
+ ...row,
48
+ branch: row.branch ?? null,
49
+ })
50
+ }
51
+
52
+ const syncControllers = new Map<string, AbortController>()
53
+
54
+ function syncDirectory(space: Info) {
55
+ if (space.config.type === "worktree") return
56
+ return space.config.directory
57
+ }
58
+
59
+ async function mirrorWorkspaceEvent(space: Info, event: { type?: string; properties?: any }) {
60
+ const directory = syncDirectory(space)
61
+ if (!directory || !event?.type) return
62
+
63
+ await Instance.provide({
64
+ directory,
65
+ init: InstanceBootstrap,
66
+ async fn() {
67
+ if (event.type === "session.status" && event.properties?.sessionID && event.properties?.status) {
68
+ SessionStatus.hydrate(event.properties.sessionID, event.properties.status)
69
+ }
70
+
71
+ if (event.type === "session.idle" && event.properties?.sessionID) {
72
+ SessionStatus.hydrate(event.properties.sessionID, { type: "idle" })
73
+ }
74
+
75
+ if (event.type === "permission.asked" && event.properties?.id) {
76
+ await PermissionNext.hydrateAsk(event.properties)
77
+ }
78
+
79
+ if (event.type === "permission.replied" && event.properties?.requestID) {
80
+ await PermissionNext.hydrateReply(event.properties.requestID)
81
+ }
82
+ },
83
+ })
84
+ }
85
+
86
+ function startSpaceSync(space: Info) {
87
+ if (space.config.type === "worktree") return
88
+ if (syncControllers.has(space.id)) return
89
+
90
+ const stop = new AbortController()
91
+ syncControllers.set(space.id, stop)
92
+
93
+ void workspaceEventLoop(space, stop.signal)
94
+ .catch((error) => {
95
+ log.warn("workspace sync listener failed", {
96
+ workspaceID: space.id,
97
+ error,
98
+ })
99
+ })
100
+ .finally(() => {
101
+ if (syncControllers.get(space.id) === stop) syncControllers.delete(space.id)
102
+ })
103
+ }
104
+
105
+ function stopSpaceSync(id: string) {
106
+ const controller = syncControllers.get(id)
107
+ if (!controller) return
108
+ controller.abort()
109
+ syncControllers.delete(id)
110
+ }
111
+
112
+ export const create = fn(
113
+ z.object({
114
+ id: Identifier.schema("workspace").optional(),
115
+ projectID: Info.shape.projectID,
116
+ branch: Info.shape.branch,
117
+ config: Info.shape.config,
118
+ }),
119
+ async (input) => {
120
+ const id = Identifier.ascending("workspace", input.id)
121
+
122
+ const { config, init } = await getAdaptor(input.config).create(input.config, input.branch, id)
123
+
124
+ const info: Info = {
125
+ id,
126
+ projectID: input.projectID,
127
+ branch: input.branch,
128
+ config,
129
+ }
130
+
131
+ await init()
132
+ await Storage.write(["workspace", info.projectID, info.id], info)
133
+ startSpaceSync(info)
134
+
135
+ GlobalBus.emit("event", {
136
+ directory: id,
137
+ payload: {
138
+ type: Event.Ready.type,
139
+ properties: {},
140
+ },
141
+ })
142
+
143
+ return info
144
+ },
145
+ )
146
+
147
+ export async function list(project: Project.Info) {
148
+ const rows = await Storage.list(["workspace", project.id])
149
+ const result = await Promise.all(rows.map((row) => Storage.read<Info>(row).catch(() => undefined)))
150
+ return result
151
+ .filter((row): row is Info => !!row)
152
+ .map(fromStorage)
153
+ .sort((a, b) => a.id.localeCompare(b.id))
154
+ }
155
+
156
+ export const get = fn(Identifier.schema("workspace"), async (id) => {
157
+ const rows = await Storage.list(["workspace"])
158
+ for (const row of rows) {
159
+ const result = await Storage.read<Info>(row).catch(() => undefined)
160
+ if (!result || result.id !== id) continue
161
+ return fromStorage(result)
162
+ }
163
+ return undefined
164
+ })
165
+
166
+ export const remove = fn(Identifier.schema("workspace"), async (id) => {
167
+ const info = await get(id)
168
+ if (info) {
169
+ stopSpaceSync(id)
170
+ await getAdaptor(info.config).remove(info.config)
171
+ await Storage.remove(["workspace", info.projectID, id])
172
+ return info
173
+ }
174
+ })
175
+ const log = Log.create({ service: "workspace-sync" })
176
+
177
+ async function workspaceEventLoop(space: Info, stop: AbortSignal) {
178
+ while (!stop.aborted) {
179
+ const res = await getAdaptor(space.config)
180
+ .request(space.config, "GET", "/event", undefined, stop)
181
+ .catch(() => undefined)
182
+ if (!res || !res.ok || !res.body) {
183
+ await Bun.sleep(1000)
184
+ continue
185
+ }
186
+ await parseSSE(res.body, stop, (event) => {
187
+ const payload = event as { type?: string; properties?: any }
188
+ void mirrorWorkspaceEvent(space, payload).catch((error) => {
189
+ log.warn("workspace event mirror failed", {
190
+ workspaceID: space.id,
191
+ error,
192
+ type: payload?.type,
193
+ })
194
+ })
195
+ GlobalBus.emit("event", {
196
+ directory: space.id,
197
+ payload,
198
+ })
199
+ })
200
+ await Bun.sleep(250)
201
+ }
202
+ }
203
+
204
+ export function startSyncing(project: Project.Info) {
205
+ void (async () => {
206
+ const spaces = (await list(project)).filter((space) => space.config.type !== "worktree")
207
+ spaces.forEach(startSpaceSync)
208
+ })()
209
+
210
+ return {
211
+ async stop() {
212
+ const spaces = await list(project)
213
+ spaces.forEach((space) => stopSpaceSync(space.id))
214
+ },
215
+ }
216
+ }
217
+
218
+ export function stopAllSyncing() {
219
+ for (const id of [...syncControllers.keys()]) {
220
+ stopSpaceSync(id)
221
+ }
222
+ }
223
+ }