@vikasitai/vikasit-code 2.0.3 → 2.0.5

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 (711) hide show
  1. package/AGENTS.md +69 -0
  2. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +15 -0
  5. package/bunfig.toml +7 -0
  6. package/drizzle.config.ts +10 -0
  7. package/git +0 -0
  8. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  9. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  10. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  11. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  12. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  13. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  14. package/migration/20260225215848_workspace/migration.sql +7 -0
  15. package/migration/20260225215848_workspace/snapshot.json +959 -0
  16. package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
  17. package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
  18. package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
  19. package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
  20. package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
  21. package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
  22. package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
  23. package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
  24. package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
  25. package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
  26. package/migration/20260323234822_events/migration.sql +13 -0
  27. package/migration/20260323234822_events/snapshot.json +1271 -0
  28. package/package.json +167 -18
  29. package/parsers-config.ts +290 -0
  30. package/script/build-all.sh +40 -0
  31. package/script/build-node.ts +54 -0
  32. package/script/build.ts +281 -0
  33. package/script/check-migrations.ts +16 -0
  34. package/script/fetch-models.ts +101 -0
  35. package/script/publish.ts +184 -0
  36. package/script/schema.ts +63 -0
  37. package/script/seed-e2e.ts +75 -0
  38. package/script/upgrade-opentui.ts +64 -0
  39. package/specs/claude-code-integration.md +563 -0
  40. package/specs/effect-migration.md +291 -0
  41. package/specs/tui-plugins.md +410 -0
  42. package/src/account/account.sql.ts +39 -0
  43. package/src/account/index.ts +397 -0
  44. package/src/account/repo.ts +163 -0
  45. package/src/account/schema.ts +91 -0
  46. package/src/acp/README.md +174 -0
  47. package/src/acp/agent.ts +1743 -0
  48. package/src/acp/session.ts +116 -0
  49. package/src/acp/types.ts +24 -0
  50. package/src/agent/agent-display.ts +82 -0
  51. package/src/agent/agent.ts +475 -0
  52. package/src/agent/extract-names.cjs +42 -0
  53. package/src/agent/generate.txt +75 -0
  54. package/src/agent/mailbox.ts +124 -0
  55. package/src/agent/orchestrator.ts +170 -0
  56. package/src/agent/prompt/code-reviewer.txt +11 -0
  57. package/src/agent/prompt/compaction.txt +14 -0
  58. package/src/agent/prompt/explore.txt +18 -0
  59. package/src/agent/prompt/general.txt +17 -0
  60. package/src/agent/prompt/summary.txt +11 -0
  61. package/src/agent/prompt/title.txt +44 -0
  62. package/src/agent/prompt/verification.txt +11 -0
  63. package/src/auth/index.ts +109 -0
  64. package/src/bun/index.ts +129 -0
  65. package/src/bun/registry.ts +50 -0
  66. package/src/bus/bus-event.ts +40 -0
  67. package/src/bus/global.ts +10 -0
  68. package/src/bus/index.ts +184 -0
  69. package/src/cli/bootstrap.ts +17 -0
  70. package/src/cli/cmd/account.ts +257 -0
  71. package/src/cli/cmd/acp.ts +70 -0
  72. package/src/cli/cmd/agent.ts +245 -0
  73. package/src/cli/cmd/auto-updater.ts +71 -0
  74. package/src/cli/cmd/cmd.ts +7 -0
  75. package/src/cli/cmd/db.ts +119 -0
  76. package/src/cli/cmd/debug/agent.ts +167 -0
  77. package/src/cli/cmd/debug/config.ts +16 -0
  78. package/src/cli/cmd/debug/file.ts +97 -0
  79. package/src/cli/cmd/debug/index.ts +48 -0
  80. package/src/cli/cmd/debug/lsp.ts +53 -0
  81. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  82. package/src/cli/cmd/debug/scrap.ts +16 -0
  83. package/src/cli/cmd/debug/skill.ts +16 -0
  84. package/src/cli/cmd/debug/snapshot.ts +52 -0
  85. package/src/cli/cmd/export.ts +89 -0
  86. package/src/cli/cmd/generate.ts +38 -0
  87. package/src/cli/cmd/github.ts +1646 -0
  88. package/src/cli/cmd/import.ts +207 -0
  89. package/src/cli/cmd/mcp.ts +754 -0
  90. package/src/cli/cmd/models.ts +78 -0
  91. package/src/cli/cmd/plug.ts +231 -0
  92. package/src/cli/cmd/pr.ts +127 -0
  93. package/src/cli/cmd/providers.ts +484 -0
  94. package/src/cli/cmd/run.ts +676 -0
  95. package/src/cli/cmd/search.ts +137 -0
  96. package/src/cli/cmd/serve.ts +24 -0
  97. package/src/cli/cmd/session.ts +159 -0
  98. package/src/cli/cmd/stats.ts +405 -0
  99. package/src/cli/cmd/tui/app.tsx +996 -0
  100. package/src/cli/cmd/tui/attach.ts +88 -0
  101. package/src/cli/cmd/tui/component/border.tsx +21 -0
  102. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  103. package/src/cli/cmd/tui/component/dialog-command.tsx +171 -0
  104. package/src/cli/cmd/tui/component/dialog-login.tsx +156 -0
  105. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  106. package/src/cli/cmd/tui/component/dialog-model.tsx +179 -0
  107. package/src/cli/cmd/tui/component/dialog-provider.tsx +329 -0
  108. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  109. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  110. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  111. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  112. package/src/cli/cmd/tui/component/dialog-status.tsx +168 -0
  113. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  114. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  115. package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
  116. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +320 -0
  117. package/src/cli/cmd/tui/component/error-component.tsx +91 -0
  118. package/src/cli/cmd/tui/component/login-gate.tsx +125 -0
  119. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  120. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  121. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +667 -0
  122. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  123. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  124. package/src/cli/cmd/tui/component/prompt/index.tsx +1224 -0
  125. package/src/cli/cmd/tui/component/prompt/part.ts +16 -0
  126. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  127. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  128. package/src/cli/cmd/tui/component/startup-loading.tsx +63 -0
  129. package/src/cli/cmd/tui/component/status-bar.ts +54 -0
  130. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  131. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  132. package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +151 -0
  133. package/src/cli/cmd/tui/context/args.tsx +15 -0
  134. package/src/cli/cmd/tui/context/directory.ts +13 -0
  135. package/src/cli/cmd/tui/context/exit.tsx +60 -0
  136. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  137. package/src/cli/cmd/tui/context/keybind.tsx +105 -0
  138. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  139. package/src/cli/cmd/tui/context/local.tsx +412 -0
  140. package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
  141. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  142. package/src/cli/cmd/tui/context/route.tsx +52 -0
  143. package/src/cli/cmd/tui/context/sdk.tsx +128 -0
  144. package/src/cli/cmd/tui/context/sync.tsx +504 -0
  145. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  146. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  147. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  148. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  149. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  150. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  151. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  152. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  153. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  154. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  155. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  156. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  157. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  158. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  159. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  160. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  161. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  162. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  163. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  164. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  165. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  166. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  167. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  168. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  169. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  170. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  171. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  172. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  173. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  174. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  175. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  176. package/src/cli/cmd/tui/context/theme/vikasit.json +245 -0
  177. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  178. package/src/cli/cmd/tui/context/theme.tsx +1236 -0
  179. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  180. package/src/cli/cmd/tui/event.ts +52 -0
  181. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
  182. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +152 -0
  183. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
  184. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +63 -0
  185. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
  186. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +99 -0
  187. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
  188. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +96 -0
  189. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +48 -0
  190. package/src/cli/cmd/tui/feature-plugins/system/invite.tsx +148 -0
  191. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +270 -0
  192. package/src/cli/cmd/tui/plugin/api.tsx +420 -0
  193. package/src/cli/cmd/tui/plugin/index.ts +3 -0
  194. package/src/cli/cmd/tui/plugin/internal.ts +29 -0
  195. package/src/cli/cmd/tui/plugin/runtime.ts +998 -0
  196. package/src/cli/cmd/tui/plugin/slots.tsx +61 -0
  197. package/src/cli/cmd/tui/routes/home.tsx +84 -0
  198. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +65 -0
  199. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +110 -0
  200. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  201. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  202. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  203. package/src/cli/cmd/tui/routes/session/index.tsx +2259 -0
  204. package/src/cli/cmd/tui/routes/session/permission.tsx +685 -0
  205. package/src/cli/cmd/tui/routes/session/question.tsx +467 -0
  206. package/src/cli/cmd/tui/routes/session/sidebar.tsx +68 -0
  207. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +124 -0
  208. package/src/cli/cmd/tui/thread.ts +232 -0
  209. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  210. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +89 -0
  211. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +208 -0
  212. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  213. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +106 -0
  214. package/src/cli/cmd/tui/ui/dialog-select.tsx +402 -0
  215. package/src/cli/cmd/tui/ui/dialog.tsx +192 -0
  216. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  217. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  218. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  219. package/src/cli/cmd/tui/util/clipboard.ts +192 -0
  220. package/src/cli/cmd/tui/util/editor.ts +37 -0
  221. package/src/cli/cmd/tui/util/selection.ts +25 -0
  222. package/src/cli/cmd/tui/util/signal.ts +7 -0
  223. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  224. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  225. package/src/cli/cmd/tui/win32.ts +129 -0
  226. package/src/cli/cmd/tui/worker.ts +172 -0
  227. package/src/cli/cmd/uninstall.ts +353 -0
  228. package/src/cli/cmd/upgrade.ts +73 -0
  229. package/src/cli/cmd/web.ts +81 -0
  230. package/src/cli/effect/prompt.ts +25 -0
  231. package/src/cli/error.ts +46 -0
  232. package/src/cli/logo.ts +20 -0
  233. package/src/cli/network.ts +60 -0
  234. package/src/cli/ui.ts +116 -0
  235. package/src/cli/upgrade.ts +31 -0
  236. package/src/command/index.ts +251 -0
  237. package/src/command/template/branch.txt +1 -0
  238. package/src/command/template/commit.txt +7 -0
  239. package/src/command/template/diff.txt +1 -0
  240. package/src/command/template/initialize.txt +10 -0
  241. package/src/command/template/memory.txt +1 -0
  242. package/src/command/template/review.txt +101 -0
  243. package/src/command/template/summary.txt +1 -0
  244. package/src/config/config.ts +1619 -0
  245. package/src/config/markdown.ts +99 -0
  246. package/src/config/migrate-tui-config.ts +155 -0
  247. package/src/config/paths.ts +174 -0
  248. package/src/config/tui-schema.ts +36 -0
  249. package/src/config/tui.ts +222 -0
  250. package/src/context/contextOptimizer.ts +277 -0
  251. package/src/control-plane/adaptors/index.ts +20 -0
  252. package/src/control-plane/adaptors/worktree.ts +38 -0
  253. package/src/control-plane/schema.ts +17 -0
  254. package/src/control-plane/sse.ts +66 -0
  255. package/src/control-plane/types.ts +21 -0
  256. package/src/control-plane/workspace.sql.ts +17 -0
  257. package/src/control-plane/workspace.ts +154 -0
  258. package/src/cost/index.ts +173 -0
  259. package/src/cost/schema.ts +43 -0
  260. package/src/cron/index.ts +200 -0
  261. package/src/effect/cross-spawn-spawner.ts +479 -0
  262. package/src/effect/instance-registry.ts +12 -0
  263. package/src/effect/instance-state.ts +47 -0
  264. package/src/effect/run-service.ts +19 -0
  265. package/src/effect/runner.ts +216 -0
  266. package/src/env/index.ts +28 -0
  267. package/src/file/ignore.ts +82 -0
  268. package/src/file/index.ts +698 -0
  269. package/src/file/protected.ts +59 -0
  270. package/src/file/ripgrep.ts +376 -0
  271. package/src/file/time.ts +128 -0
  272. package/src/file/watcher.ts +171 -0
  273. package/src/filesystem/index.ts +226 -0
  274. package/src/flag/flag.ts +175 -0
  275. package/src/format/formatter.ts +396 -0
  276. package/src/format/index.ts +199 -0
  277. package/src/global/index.ts +54 -0
  278. package/src/hook/index.ts +132 -0
  279. package/src/hook/schema.ts +31 -0
  280. package/src/id/id.ts +85 -0
  281. package/src/ide/index.ts +74 -0
  282. package/src/index.ts +249 -0
  283. package/src/installation/index.ts +363 -0
  284. package/src/lsp/client.ts +252 -0
  285. package/src/lsp/index.ts +558 -0
  286. package/src/lsp/language.ts +120 -0
  287. package/src/lsp/launch.ts +21 -0
  288. package/src/lsp/server.ts +2093 -0
  289. package/src/mcp/auth.ts +181 -0
  290. package/src/mcp/index.ts +926 -0
  291. package/src/mcp/oauth-callback.ts +215 -0
  292. package/src/mcp/oauth-provider.ts +185 -0
  293. package/src/memory/memory.ts +140 -0
  294. package/src/node.ts +1 -0
  295. package/src/patch/index.ts +680 -0
  296. package/src/permission/arity.ts +163 -0
  297. package/src/permission/auto-classify.ts +114 -0
  298. package/src/permission/evaluate.ts +15 -0
  299. package/src/permission/index.ts +322 -0
  300. package/src/permission/schema.ts +17 -0
  301. package/src/plugin/codex.ts +596 -0
  302. package/src/plugin/copilot.ts +343 -0
  303. package/src/plugin/index.ts +322 -0
  304. package/src/plugin/install.ts +417 -0
  305. package/src/plugin/loader.ts +137 -0
  306. package/src/plugin/meta.ts +188 -0
  307. package/src/plugin/shared.ts +272 -0
  308. package/src/project/bootstrap.ts +31 -0
  309. package/src/project/instance.ts +167 -0
  310. package/src/project/project.sql.ts +16 -0
  311. package/src/project/project.ts +519 -0
  312. package/src/project/schema.ts +16 -0
  313. package/src/project/state.ts +70 -0
  314. package/src/project/vcs.ts +124 -0
  315. package/src/provider/auth.ts +252 -0
  316. package/src/provider/error.ts +197 -0
  317. package/src/provider/github-models.ts +53 -0
  318. package/src/provider/models.ts +139 -0
  319. package/src/provider/provider.ts +1650 -0
  320. package/src/provider/schema.ts +38 -0
  321. package/src/provider/sdk/copilot/README.md +5 -0
  322. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -0
  323. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  324. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +19 -0
  325. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  326. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +815 -0
  327. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  328. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  329. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +83 -0
  330. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  331. package/src/provider/sdk/copilot/index.ts +2 -0
  332. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  333. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +335 -0
  334. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  335. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  336. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  337. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +214 -0
  338. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1769 -0
  339. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
  340. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  341. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
  342. package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
  343. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
  344. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
  345. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
  346. package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
  347. package/src/provider/transform.ts +1045 -0
  348. package/src/pty/index.ts +397 -0
  349. package/src/pty/schema.ts +17 -0
  350. package/src/question/index.ts +221 -0
  351. package/src/question/schema.ts +17 -0
  352. package/src/server/error.ts +36 -0
  353. package/src/server/event.ts +7 -0
  354. package/src/server/instance.ts +285 -0
  355. package/src/server/mdns.ts +60 -0
  356. package/src/server/middleware.ts +33 -0
  357. package/src/server/projectors.ts +28 -0
  358. package/src/server/router.ts +99 -0
  359. package/src/server/routes/config.ts +92 -0
  360. package/src/server/routes/event.ts +83 -0
  361. package/src/server/routes/experimental.ts +271 -0
  362. package/src/server/routes/file.ts +197 -0
  363. package/src/server/routes/global.ts +312 -0
  364. package/src/server/routes/instance/httpapi/app.ts +21 -0
  365. package/src/server/routes/mcp.ts +225 -0
  366. package/src/server/routes/permission.ts +69 -0
  367. package/src/server/routes/project.ts +118 -0
  368. package/src/server/routes/provider.ts +171 -0
  369. package/src/server/routes/pty.ts +211 -0
  370. package/src/server/routes/question.ts +99 -0
  371. package/src/server/routes/session.ts +1031 -0
  372. package/src/server/routes/tui.ts +379 -0
  373. package/src/server/routes/workspace.ts +94 -0
  374. package/src/server/server.ts +314 -0
  375. package/src/session/compaction.ts +432 -0
  376. package/src/session/index.ts +885 -0
  377. package/src/session/instruction.ts +192 -0
  378. package/src/session/llm.ts +365 -0
  379. package/src/session/memory.ts +126 -0
  380. package/src/session/message-v2.ts +1030 -0
  381. package/src/session/message.ts +191 -0
  382. package/src/session/overflow.ts +22 -0
  383. package/src/session/processor.ts +697 -0
  384. package/src/session/projectors.ts +135 -0
  385. package/src/session/prompt/anthropic.txt +125 -0
  386. package/src/session/prompt/beast.txt +161 -0
  387. package/src/session/prompt/build-switch.txt +5 -0
  388. package/src/session/prompt/codex.txt +90 -0
  389. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  390. package/src/session/prompt/default.txt +116 -0
  391. package/src/session/prompt/gemini.txt +173 -0
  392. package/src/session/prompt/gpt.txt +120 -0
  393. package/src/session/prompt/kimi.txt +131 -0
  394. package/src/session/prompt/max-steps.txt +16 -0
  395. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  396. package/src/session/prompt/plan.txt +26 -0
  397. package/src/session/prompt/trinity.txt +109 -0
  398. package/src/session/prompt.ts +1898 -0
  399. package/src/session/rate-limit.ts +176 -0
  400. package/src/session/retry.ts +107 -0
  401. package/src/session/revert.ts +135 -0
  402. package/src/session/schema.ts +38 -0
  403. package/src/session/session.sql.ts +103 -0
  404. package/src/session/status.ts +102 -0
  405. package/src/session/summary.ts +169 -0
  406. package/src/session/system.ts +76 -0
  407. package/src/session/todo.ts +57 -0
  408. package/src/share/share-next.ts +287 -0
  409. package/src/share/share.sql.ts +13 -0
  410. package/src/shell/shell.ts +110 -0
  411. package/src/skill/discovery.ts +116 -0
  412. package/src/skill/index.ts +277 -0
  413. package/src/snapshot/index.ts +489 -0
  414. package/src/sql.d.ts +4 -0
  415. package/src/storage/db.bun.ts +8 -0
  416. package/src/storage/db.node.ts +8 -0
  417. package/src/storage/db.ts +177 -0
  418. package/src/storage/json-migration.ts +425 -0
  419. package/src/storage/schema.sql.ts +10 -0
  420. package/src/storage/schema.ts +5 -0
  421. package/src/storage/storage.ts +353 -0
  422. package/src/sync/README.md +179 -0
  423. package/src/sync/event.sql.ts +16 -0
  424. package/src/sync/index.ts +263 -0
  425. package/src/sync/schema.ts +14 -0
  426. package/src/task/index.ts +89 -0
  427. package/src/task/schema.ts +39 -0
  428. package/src/tool/apply_patch.ts +281 -0
  429. package/src/tool/apply_patch.txt +33 -0
  430. package/src/tool/bash-security.ts +214 -0
  431. package/src/tool/bash.ts +526 -0
  432. package/src/tool/bash.txt +129 -0
  433. package/src/tool/batch.ts +183 -0
  434. package/src/tool/batch.txt +24 -0
  435. package/src/tool/codesearch.ts +132 -0
  436. package/src/tool/codesearch.txt +12 -0
  437. package/src/tool/cron-create.ts +38 -0
  438. package/src/tool/cron-create.txt +10 -0
  439. package/src/tool/cron-delete.ts +29 -0
  440. package/src/tool/cron-delete.txt +1 -0
  441. package/src/tool/cron-list.ts +37 -0
  442. package/src/tool/cron-list.txt +1 -0
  443. package/src/tool/edit.ts +667 -0
  444. package/src/tool/edit.txt +10 -0
  445. package/src/tool/external-directory.ts +37 -0
  446. package/src/tool/glob.ts +78 -0
  447. package/src/tool/glob.txt +6 -0
  448. package/src/tool/grep.ts +156 -0
  449. package/src/tool/grep.txt +8 -0
  450. package/src/tool/invalid.ts +17 -0
  451. package/src/tool/ls.ts +121 -0
  452. package/src/tool/ls.txt +1 -0
  453. package/src/tool/lsp.ts +97 -0
  454. package/src/tool/lsp.txt +19 -0
  455. package/src/tool/multiedit.ts +46 -0
  456. package/src/tool/multiedit.txt +41 -0
  457. package/src/tool/notebook-edit.ts +136 -0
  458. package/src/tool/notebook-edit.txt +10 -0
  459. package/src/tool/plan-enter.txt +14 -0
  460. package/src/tool/plan-exit.txt +13 -0
  461. package/src/tool/plan.ts +130 -0
  462. package/src/tool/question.ts +33 -0
  463. package/src/tool/question.txt +10 -0
  464. package/src/tool/read.ts +296 -0
  465. package/src/tool/read.txt +14 -0
  466. package/src/tool/registry.ts +333 -0
  467. package/src/tool/schema.ts +17 -0
  468. package/src/tool/sendmessage.ts +57 -0
  469. package/src/tool/sendmessage.txt +6 -0
  470. package/src/tool/skill.ts +105 -0
  471. package/src/tool/task-create.ts +25 -0
  472. package/src/tool/task-create.txt +1 -0
  473. package/src/tool/task-get.ts +27 -0
  474. package/src/tool/task-get.txt +1 -0
  475. package/src/tool/task-list.ts +45 -0
  476. package/src/tool/task-list.txt +1 -0
  477. package/src/tool/task-output.ts +28 -0
  478. package/src/tool/task-output.txt +1 -0
  479. package/src/tool/task-stop.ts +27 -0
  480. package/src/tool/task-stop.txt +1 -0
  481. package/src/tool/task-update.ts +38 -0
  482. package/src/tool/task-update.txt +1 -0
  483. package/src/tool/task.ts +246 -0
  484. package/src/tool/task.txt +65 -0
  485. package/src/tool/todo.ts +31 -0
  486. package/src/tool/todowrite.txt +167 -0
  487. package/src/tool/tool.ts +98 -0
  488. package/src/tool/toolsearch.ts +117 -0
  489. package/src/tool/toolsearch.txt +10 -0
  490. package/src/tool/truncate.ts +144 -0
  491. package/src/tool/truncation-dir.ts +4 -0
  492. package/src/tool/webfetch.ts +206 -0
  493. package/src/tool/webfetch.txt +13 -0
  494. package/src/tool/websearch.ts +150 -0
  495. package/src/tool/websearch.txt +14 -0
  496. package/src/tool/worktree-enter.ts +38 -0
  497. package/src/tool/worktree-enter.txt +8 -0
  498. package/src/tool/worktree-exit.ts +43 -0
  499. package/src/tool/worktree-exit.txt +3 -0
  500. package/src/tool/write.ts +84 -0
  501. package/src/tool/write.txt +8 -0
  502. package/src/util/abort.ts +35 -0
  503. package/src/util/archive.ts +17 -0
  504. package/src/util/color.ts +19 -0
  505. package/src/util/context.ts +25 -0
  506. package/src/util/data-url.ts +9 -0
  507. package/src/util/defer.ts +12 -0
  508. package/src/util/effect-http-client.ts +11 -0
  509. package/src/util/effect-zod.ts +98 -0
  510. package/src/util/error.ts +77 -0
  511. package/src/util/filesystem.ts +220 -0
  512. package/src/util/flock.ts +333 -0
  513. package/src/util/fn.ts +21 -0
  514. package/src/util/format.ts +20 -0
  515. package/src/util/git.ts +35 -0
  516. package/src/util/glob.ts +34 -0
  517. package/src/util/hash.ts +7 -0
  518. package/src/util/iife.ts +3 -0
  519. package/src/util/keybind.ts +103 -0
  520. package/src/util/lazy.ts +23 -0
  521. package/src/util/locale.ts +81 -0
  522. package/src/util/lock.ts +98 -0
  523. package/src/util/log.ts +182 -0
  524. package/src/util/network.ts +9 -0
  525. package/src/util/process.ts +172 -0
  526. package/src/util/queue.ts +32 -0
  527. package/src/util/record.ts +3 -0
  528. package/src/util/rpc.ts +66 -0
  529. package/src/util/schema.ts +53 -0
  530. package/src/util/scrap.ts +10 -0
  531. package/src/util/signal.ts +12 -0
  532. package/src/util/timeout.ts +14 -0
  533. package/src/util/token.ts +7 -0
  534. package/src/util/update-schema.ts +13 -0
  535. package/src/util/which.ts +14 -0
  536. package/src/util/wildcard.ts +59 -0
  537. package/src/vim/agent-namer.ts +86 -0
  538. package/src/vim/types.ts +29 -0
  539. package/src/worktree/index.ts +638 -0
  540. package/sst-env.d.ts +10 -0
  541. package/test/AGENTS.md +81 -0
  542. package/test/account/repo.test.ts +326 -0
  543. package/test/account/service.test.ts +282 -0
  544. package/test/acp/agent-interface.test.ts +51 -0
  545. package/test/acp/event-subscription.test.ts +685 -0
  546. package/test/agent/agent.test.ts +717 -0
  547. package/test/auth/auth.test.ts +58 -0
  548. package/test/bun.test.ts +137 -0
  549. package/test/bus/bus-effect.test.ts +164 -0
  550. package/test/bus/bus-integration.test.ts +87 -0
  551. package/test/bus/bus.test.ts +219 -0
  552. package/test/cli/account.test.ts +26 -0
  553. package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
  554. package/test/cli/github-action.test.ts +198 -0
  555. package/test/cli/github-remote.test.ts +80 -0
  556. package/test/cli/import.test.ts +54 -0
  557. package/test/cli/plugin-auth-picker.test.ts +120 -0
  558. package/test/cli/tui/keybind-plugin.test.ts +90 -0
  559. package/test/cli/tui/plugin-add.test.ts +61 -0
  560. package/test/cli/tui/plugin-install.test.ts +89 -0
  561. package/test/cli/tui/plugin-lifecycle.test.ts +225 -0
  562. package/test/cli/tui/plugin-loader-entrypoint.test.ts +492 -0
  563. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  564. package/test/cli/tui/plugin-loader.test.ts +752 -0
  565. package/test/cli/tui/plugin-toggle.test.ts +159 -0
  566. package/test/cli/tui/theme-store.test.ts +51 -0
  567. package/test/cli/tui/thread.test.ts +128 -0
  568. package/test/cli/tui/transcript.test.ts +322 -0
  569. package/test/config/agent-color.test.ts +71 -0
  570. package/test/config/config.test.ts +2218 -0
  571. package/test/config/fixtures/empty-frontmatter.md +4 -0
  572. package/test/config/fixtures/frontmatter.md +28 -0
  573. package/test/config/fixtures/markdown-header.md +11 -0
  574. package/test/config/fixtures/no-frontmatter.md +1 -0
  575. package/test/config/fixtures/weird-model-id.md +13 -0
  576. package/test/config/markdown.test.ts +228 -0
  577. package/test/config/tui.test.ts +673 -0
  578. package/test/control-plane/sse.test.ts +56 -0
  579. package/test/effect/cross-spawn-spawner.test.ts +402 -0
  580. package/test/effect/instance-state.test.ts +384 -0
  581. package/test/effect/run-service.test.ts +46 -0
  582. package/test/effect/runner.test.ts +523 -0
  583. package/test/file/fsmonitor.test.ts +62 -0
  584. package/test/file/ignore.test.ts +10 -0
  585. package/test/file/index.test.ts +946 -0
  586. package/test/file/path-traversal.test.ts +198 -0
  587. package/test/file/ripgrep.test.ts +54 -0
  588. package/test/file/time.test.ts +354 -0
  589. package/test/file/watcher.test.ts +247 -0
  590. package/test/filesystem/filesystem.test.ts +319 -0
  591. package/test/fixture/db.ts +11 -0
  592. package/test/fixture/fixture.test.ts +26 -0
  593. package/test/fixture/fixture.ts +141 -0
  594. package/test/fixture/flock-worker.ts +72 -0
  595. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  596. package/test/fixture/plug-worker.ts +93 -0
  597. package/test/fixture/plugin-meta-worker.ts +26 -0
  598. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  599. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  600. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  601. package/test/fixture/skills/index.json +6 -0
  602. package/test/fixture/tui-plugin.ts +335 -0
  603. package/test/fixture/tui-runtime.ts +27 -0
  604. package/test/format/format.test.ts +179 -0
  605. package/test/ide/ide.test.ts +82 -0
  606. package/test/installation/installation.test.ts +151 -0
  607. package/test/keybind.test.ts +421 -0
  608. package/test/lib/effect.ts +37 -0
  609. package/test/lib/filesystem.ts +10 -0
  610. package/test/lsp/client.test.ts +95 -0
  611. package/test/lsp/index.test.ts +55 -0
  612. package/test/lsp/launch.test.ts +22 -0
  613. package/test/lsp/lifecycle.test.ts +147 -0
  614. package/test/mcp/headers.test.ts +153 -0
  615. package/test/mcp/lifecycle.test.ts +750 -0
  616. package/test/mcp/oauth-auto-connect.test.ts +199 -0
  617. package/test/mcp/oauth-browser.test.ts +249 -0
  618. package/test/memory/abort-leak.test.ts +137 -0
  619. package/test/patch/patch.test.ts +348 -0
  620. package/test/permission/arity.test.ts +33 -0
  621. package/test/permission/next.test.ts +1148 -0
  622. package/test/permission-task.test.ts +323 -0
  623. package/test/plugin/auth-override.test.ts +74 -0
  624. package/test/plugin/codex.test.ts +123 -0
  625. package/test/plugin/install-concurrency.test.ts +140 -0
  626. package/test/plugin/install.test.ts +531 -0
  627. package/test/plugin/loader-shared.test.ts +836 -0
  628. package/test/plugin/meta.test.ts +137 -0
  629. package/test/plugin/trigger.test.ts +111 -0
  630. package/test/preload.ts +90 -0
  631. package/test/project/migrate-global.test.ts +140 -0
  632. package/test/project/project.test.ts +459 -0
  633. package/test/project/state.test.ts +115 -0
  634. package/test/project/vcs.test.ts +116 -0
  635. package/test/project/worktree-remove.test.ts +96 -0
  636. package/test/project/worktree.test.ts +173 -0
  637. package/test/provider/amazon-bedrock.test.ts +447 -0
  638. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  639. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  640. package/test/provider/gitlab-duo.test.ts +412 -0
  641. package/test/provider/provider.test.ts +2284 -0
  642. package/test/provider/transform.test.ts +2758 -0
  643. package/test/pty/pty-output-isolation.test.ts +141 -0
  644. package/test/pty/pty-session.test.ts +92 -0
  645. package/test/pty/pty-shell.test.ts +59 -0
  646. package/test/question/question.test.ts +453 -0
  647. package/test/server/global-session-list.test.ts +89 -0
  648. package/test/server/project-init-git.test.ts +121 -0
  649. package/test/server/session-actions.test.ts +83 -0
  650. package/test/server/session-list.test.ts +98 -0
  651. package/test/server/session-messages.test.ts +159 -0
  652. package/test/server/session-select.test.ts +84 -0
  653. package/test/session/compaction.test.ts +1202 -0
  654. package/test/session/instruction.test.ts +170 -0
  655. package/test/session/llm.test.ts +1098 -0
  656. package/test/session/message-v2.test.ts +957 -0
  657. package/test/session/messages-pagination.test.ts +115 -0
  658. package/test/session/processor-effect.test.ts +872 -0
  659. package/test/session/prompt-concurrency.test.ts +247 -0
  660. package/test/session/prompt-effect.test.ts +1206 -0
  661. package/test/session/prompt.test.ts +518 -0
  662. package/test/session/retry.test.ts +232 -0
  663. package/test/session/revert-compact.test.ts +286 -0
  664. package/test/session/session.test.ts +142 -0
  665. package/test/session/structured-output-integration.test.ts +233 -0
  666. package/test/session/structured-output.test.ts +391 -0
  667. package/test/session/system.test.ts +59 -0
  668. package/test/share/share-next.test.ts +76 -0
  669. package/test/shell/shell.test.ts +73 -0
  670. package/test/skill/discovery.test.ts +116 -0
  671. package/test/skill/skill.test.ts +392 -0
  672. package/test/snapshot/snapshot.test.ts +1235 -0
  673. package/test/storage/db.test.ts +14 -0
  674. package/test/storage/json-migration.test.ts +849 -0
  675. package/test/storage/storage.test.ts +295 -0
  676. package/test/sync/index.test.ts +191 -0
  677. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  678. package/test/tool/apply_patch.test.ts +567 -0
  679. package/test/tool/bash.test.ts +984 -0
  680. package/test/tool/edit.test.ts +681 -0
  681. package/test/tool/external-directory.test.ts +198 -0
  682. package/test/tool/fixtures/large-image.png +0 -0
  683. package/test/tool/fixtures/models-api.json +65179 -0
  684. package/test/tool/grep.test.ts +111 -0
  685. package/test/tool/question.test.ts +108 -0
  686. package/test/tool/read.test.ts +546 -0
  687. package/test/tool/registry.test.ts +126 -0
  688. package/test/tool/skill.test.ts +167 -0
  689. package/test/tool/task.test.ts +49 -0
  690. package/test/tool/truncation.test.ts +161 -0
  691. package/test/tool/webfetch.test.ts +101 -0
  692. package/test/tool/write.test.ts +353 -0
  693. package/test/util/data-url.test.ts +14 -0
  694. package/test/util/effect-zod.test.ts +61 -0
  695. package/test/util/error.test.ts +38 -0
  696. package/test/util/filesystem.test.ts +567 -0
  697. package/test/util/flock.test.ts +383 -0
  698. package/test/util/format.test.ts +59 -0
  699. package/test/util/glob.test.ts +164 -0
  700. package/test/util/iife.test.ts +36 -0
  701. package/test/util/lazy.test.ts +50 -0
  702. package/test/util/lock.test.ts +72 -0
  703. package/test/util/module.test.ts +59 -0
  704. package/test/util/process.test.ts +128 -0
  705. package/test/util/timeout.test.ts +21 -0
  706. package/test/util/which.test.ts +100 -0
  707. package/test/util/wildcard.test.ts +90 -0
  708. package/tsconfig.json +23 -0
  709. package/vikasitai-vikasit-code-2.0.5.tgz +0 -0
  710. package/LICENSE +0 -21
  711. /package/{postinstall.mjs → script/postinstall.mjs} +0 -0
@@ -0,0 +1,188 @@
1
+ import path from "path"
2
+ import { fileURLToPath } from "url"
3
+
4
+ import { Flag } from "@/flag/flag"
5
+ import { Global } from "@/global"
6
+ import { Filesystem } from "@/util/filesystem"
7
+ import { Flock } from "@/util/flock"
8
+
9
+ import { parsePluginSpecifier, pluginSource } from "./shared"
10
+
11
+ export namespace PluginMeta {
12
+ type Source = "file" | "npm"
13
+
14
+ export type Theme = {
15
+ src: string
16
+ dest: string
17
+ mtime?: number
18
+ size?: number
19
+ }
20
+
21
+ export type Entry = {
22
+ id: string
23
+ source: Source
24
+ spec: string
25
+ target: string
26
+ requested?: string
27
+ version?: string
28
+ modified?: number
29
+ first_time: number
30
+ last_time: number
31
+ time_changed: number
32
+ load_count: number
33
+ fingerprint: string
34
+ themes?: Record<string, Theme>
35
+ }
36
+
37
+ export type State = "first" | "updated" | "same"
38
+
39
+ export type Touch = {
40
+ spec: string
41
+ target: string
42
+ id: string
43
+ }
44
+
45
+ type Store = Record<string, Entry>
46
+ type Core = Omit<Entry, "first_time" | "last_time" | "time_changed" | "load_count" | "fingerprint" | "themes">
47
+ type Row = Touch & { core: Core }
48
+
49
+ function storePath() {
50
+ return Flag.VIKASIT_PLUGIN_META_FILE ?? path.join(Global.Path.state, "plugin-meta.json")
51
+ }
52
+
53
+ function lock(file: string) {
54
+ return `plugin-meta:${file}`
55
+ }
56
+
57
+ function fileTarget(spec: string, target: string) {
58
+ if (spec.startsWith("file://")) return fileURLToPath(spec)
59
+ if (target.startsWith("file://")) return fileURLToPath(target)
60
+ return
61
+ }
62
+
63
+ async function modifiedAt(file: string) {
64
+ const stat = await Filesystem.statAsync(file)
65
+ if (!stat) return
66
+ const mtime = stat.mtimeMs
67
+ return Math.floor(typeof mtime === "bigint" ? Number(mtime) : mtime)
68
+ }
69
+
70
+ function resolvedTarget(target: string) {
71
+ if (target.startsWith("file://")) return fileURLToPath(target)
72
+ return target
73
+ }
74
+
75
+ async function npmVersion(target: string) {
76
+ const resolved = resolvedTarget(target)
77
+ const stat = await Filesystem.statAsync(resolved)
78
+ const dir = stat?.isDirectory() ? resolved : path.dirname(resolved)
79
+ return Filesystem.readJson<{ version?: string }>(path.join(dir, "package.json"))
80
+ .then((item) => item.version)
81
+ .catch(() => undefined)
82
+ }
83
+
84
+ async function entryCore(item: Touch): Promise<Core> {
85
+ const spec = item.spec
86
+ const target = item.target
87
+ const source = pluginSource(spec)
88
+ if (source === "file") {
89
+ const file = fileTarget(spec, target)
90
+ return {
91
+ id: item.id,
92
+ source,
93
+ spec,
94
+ target,
95
+ modified: file ? await modifiedAt(file) : undefined,
96
+ }
97
+ }
98
+
99
+ return {
100
+ id: item.id,
101
+ source,
102
+ spec,
103
+ target,
104
+ requested: parsePluginSpecifier(spec).version,
105
+ version: await npmVersion(target),
106
+ }
107
+ }
108
+
109
+ function fingerprint(value: Core) {
110
+ if (value.source === "file") return [value.target, value.modified ?? ""].join("|")
111
+ return [value.target, value.requested ?? "", value.version ?? ""].join("|")
112
+ }
113
+
114
+ async function read(file: string): Promise<Store> {
115
+ return Filesystem.readJson<Store>(file).catch(() => ({}) as Store)
116
+ }
117
+
118
+ async function row(item: Touch): Promise<Row> {
119
+ return {
120
+ ...item,
121
+ core: await entryCore(item),
122
+ }
123
+ }
124
+
125
+ function next(prev: Entry | undefined, core: Core, now: number): { state: State; entry: Entry } {
126
+ const entry: Entry = {
127
+ ...core,
128
+ first_time: prev?.first_time ?? now,
129
+ last_time: now,
130
+ time_changed: prev?.time_changed ?? now,
131
+ load_count: (prev?.load_count ?? 0) + 1,
132
+ fingerprint: fingerprint(core),
133
+ themes: prev?.themes,
134
+ }
135
+ const state: State = !prev ? "first" : prev.fingerprint === entry.fingerprint ? "same" : "updated"
136
+ if (state === "updated") entry.time_changed = now
137
+ return {
138
+ state,
139
+ entry,
140
+ }
141
+ }
142
+
143
+ export async function touchMany(items: Touch[]): Promise<Array<{ state: State; entry: Entry }>> {
144
+ if (!items.length) return []
145
+ const file = storePath()
146
+ const rows = await Promise.all(items.map((item) => row(item)))
147
+
148
+ return Flock.withLock(lock(file), async () => {
149
+ const store = await read(file)
150
+ const now = Date.now()
151
+ const out: Array<{ state: State; entry: Entry }> = []
152
+ for (const item of rows) {
153
+ const hit = next(store[item.id], item.core, now)
154
+ store[item.id] = hit.entry
155
+ out.push(hit)
156
+ }
157
+ await Filesystem.writeJson(file, store)
158
+ return out
159
+ })
160
+ }
161
+
162
+ export async function touch(spec: string, target: string, id: string): Promise<{ state: State; entry: Entry }> {
163
+ return touchMany([{ spec, target, id }]).then((item) => {
164
+ const hit = item[0]
165
+ if (hit) return hit
166
+ throw new Error("Failed to touch plugin metadata.")
167
+ })
168
+ }
169
+
170
+ export async function setTheme(id: string, name: string, theme: Theme): Promise<void> {
171
+ const file = storePath()
172
+ await Flock.withLock(lock(file), async () => {
173
+ const store = await read(file)
174
+ const entry = store[id]
175
+ if (!entry) return
176
+ entry.themes = {
177
+ ...(entry.themes ?? {}),
178
+ [name]: theme,
179
+ }
180
+ await Filesystem.writeJson(file, store)
181
+ })
182
+ }
183
+
184
+ export async function list(): Promise<Store> {
185
+ const file = storePath()
186
+ return Flock.withLock(lock(file), async () => read(file))
187
+ }
188
+ }
@@ -0,0 +1,272 @@
1
+ import path from "path"
2
+ import { fileURLToPath, pathToFileURL } from "url"
3
+ import semver from "semver"
4
+ import { BunProc } from "@/bun"
5
+ import { Filesystem } from "@/util/filesystem"
6
+ import { isRecord } from "@/util/record"
7
+
8
+ // Old npm package names for plugins that are now built-in
9
+ export const DEPRECATED_PLUGIN_PACKAGES = ["opencode-openai-codex-auth", "opencode-copilot-auth"]
10
+
11
+ export function isDeprecatedPlugin(spec: string) {
12
+ return DEPRECATED_PLUGIN_PACKAGES.some((pkg) => spec.includes(pkg))
13
+ }
14
+
15
+ export function parsePluginSpecifier(spec: string) {
16
+ const lastAt = spec.lastIndexOf("@")
17
+ const pkg = lastAt > 0 ? spec.substring(0, lastAt) : spec
18
+ const version = lastAt > 0 ? spec.substring(lastAt + 1) : "latest"
19
+ return { pkg, version }
20
+ }
21
+
22
+ export type PluginSource = "file" | "npm"
23
+ export type PluginKind = "server" | "tui"
24
+ type PluginMode = "strict" | "detect"
25
+
26
+ export type PluginPackage = {
27
+ dir: string
28
+ pkg: string
29
+ json: Record<string, unknown>
30
+ }
31
+
32
+ export type PluginEntry = {
33
+ spec: string
34
+ source: PluginSource
35
+ target: string
36
+ pkg?: PluginPackage
37
+ entry?: string
38
+ }
39
+
40
+ const INDEX_FILES = ["index.ts", "index.tsx", "index.js", "index.mjs", "index.cjs"]
41
+
42
+ export function pluginSource(spec: string): PluginSource {
43
+ if (isPathPluginSpec(spec)) return "file"
44
+ return "npm"
45
+ }
46
+
47
+ function resolveExportPath(raw: string, dir: string) {
48
+ if (raw.startsWith("file://")) return fileURLToPath(raw)
49
+ if (path.isAbsolute(raw)) return raw
50
+ return path.resolve(dir, raw)
51
+ }
52
+
53
+ function extractExportValue(value: unknown): string | undefined {
54
+ if (typeof value === "string") return value
55
+ if (!isRecord(value)) return undefined
56
+ for (const key of ["import", "default"]) {
57
+ const nested = value[key]
58
+ if (typeof nested === "string") return nested
59
+ }
60
+ return undefined
61
+ }
62
+
63
+ function packageMain(pkg: PluginPackage) {
64
+ const value = pkg.json.main
65
+ if (typeof value !== "string") return
66
+ const next = value.trim()
67
+ if (!next) return
68
+ return next
69
+ }
70
+
71
+ function resolvePackagePath(spec: string, raw: string, kind: PluginKind, pkg: PluginPackage) {
72
+ const resolved = resolveExportPath(raw, pkg.dir)
73
+ const root = Filesystem.resolve(pkg.dir)
74
+ const next = Filesystem.resolve(resolved)
75
+ if (!Filesystem.contains(root, next)) {
76
+ throw new Error(`Plugin ${spec} resolved ${kind} entry outside plugin directory`)
77
+ }
78
+ return pathToFileURL(next).href
79
+ }
80
+
81
+ function resolvePackageEntrypoint(spec: string, kind: PluginKind, pkg: PluginPackage) {
82
+ const exports = pkg.json.exports
83
+ if (isRecord(exports)) {
84
+ const raw = extractExportValue(exports[`./${kind}`])
85
+ if (raw) return resolvePackagePath(spec, raw, kind, pkg)
86
+ }
87
+
88
+ if (kind !== "server") return
89
+ const main = packageMain(pkg)
90
+ if (!main) return
91
+ return resolvePackagePath(spec, main, kind, pkg)
92
+ }
93
+
94
+ function targetPath(target: string) {
95
+ if (target.startsWith("file://")) return fileURLToPath(target)
96
+ if (path.isAbsolute(target)) return target
97
+ }
98
+
99
+ async function resolveDirectoryIndex(dir: string) {
100
+ for (const name of INDEX_FILES) {
101
+ const file = path.join(dir, name)
102
+ if (await Filesystem.exists(file)) return file
103
+ }
104
+ }
105
+
106
+ async function resolveTargetDirectory(target: string) {
107
+ const file = targetPath(target)
108
+ if (!file) return
109
+ const stat = await Filesystem.stat(file)
110
+ if (!stat?.isDirectory()) return
111
+ return file
112
+ }
113
+
114
+ async function resolvePluginEntrypoint(spec: string, target: string, kind: PluginKind, pkg?: PluginPackage) {
115
+ const source = pluginSource(spec)
116
+ const hit =
117
+ pkg ?? (source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined))
118
+ if (!hit) return target
119
+
120
+ const entry = resolvePackageEntrypoint(spec, kind, hit)
121
+ if (entry) return entry
122
+
123
+ const dir = await resolveTargetDirectory(target)
124
+
125
+ if (kind === "tui") {
126
+ if (source === "file" && dir) {
127
+ const index = await resolveDirectoryIndex(dir)
128
+ if (index) return pathToFileURL(index).href
129
+ }
130
+
131
+ if (source === "npm") return
132
+ if (dir) return
133
+
134
+ return target
135
+ }
136
+
137
+ if (dir && isRecord(hit.json.exports)) {
138
+ if (source === "file") {
139
+ const index = await resolveDirectoryIndex(dir)
140
+ if (index) return pathToFileURL(index).href
141
+ }
142
+
143
+ return
144
+ }
145
+
146
+ return target
147
+ }
148
+
149
+ export function isPathPluginSpec(spec: string) {
150
+ return spec.startsWith("file://") || spec.startsWith(".") || path.isAbsolute(spec) || /^[A-Za-z]:[\\/]/.test(spec)
151
+ }
152
+
153
+ export async function resolvePathPluginTarget(spec: string) {
154
+ const raw = spec.startsWith("file://") ? fileURLToPath(spec) : spec
155
+ const file = path.isAbsolute(raw) || /^[A-Za-z]:[\\/]/.test(raw) ? raw : path.resolve(raw)
156
+ const stat = await Filesystem.stat(file)
157
+ if (!stat?.isDirectory()) {
158
+ if (spec.startsWith("file://")) return spec
159
+ return pathToFileURL(file).href
160
+ }
161
+
162
+ if (await Filesystem.exists(path.join(file, "package.json"))) {
163
+ return pathToFileURL(file).href
164
+ }
165
+
166
+ const index = await resolveDirectoryIndex(file)
167
+ if (index) return pathToFileURL(index).href
168
+
169
+ throw new Error(`Plugin directory ${file} is missing package.json or index file`)
170
+ }
171
+
172
+ export async function checkPluginCompatibility(target: string, opencodeVersion: string, pkg?: PluginPackage) {
173
+ if (!semver.valid(opencodeVersion) || semver.major(opencodeVersion) === 0) return
174
+ const hit = pkg ?? (await readPluginPackage(target).catch(() => undefined))
175
+ if (!hit) return
176
+ const engines = hit.json.engines
177
+ if (!isRecord(engines)) return
178
+ const range = engines.opencode
179
+ if (typeof range !== "string") return
180
+ if (!semver.satisfies(opencodeVersion, range)) {
181
+ throw new Error(`Plugin requires opencode ${range} but running ${opencodeVersion}`)
182
+ }
183
+ }
184
+
185
+ export async function resolvePluginTarget(spec: string, parsed = parsePluginSpecifier(spec)) {
186
+ if (isPathPluginSpec(spec)) return resolvePathPluginTarget(spec)
187
+ return BunProc.install(parsed.pkg, parsed.version, { ignoreScripts: true })
188
+ }
189
+
190
+ export async function readPluginPackage(target: string): Promise<PluginPackage> {
191
+ const file = target.startsWith("file://") ? fileURLToPath(target) : target
192
+ const stat = await Filesystem.stat(file)
193
+ const dir = stat?.isDirectory() ? file : path.dirname(file)
194
+ const pkg = path.join(dir, "package.json")
195
+ const json = await Filesystem.readJson<Record<string, unknown>>(pkg)
196
+ return { dir, pkg, json }
197
+ }
198
+
199
+ export async function createPluginEntry(spec: string, target: string, kind: PluginKind): Promise<PluginEntry> {
200
+ const source = pluginSource(spec)
201
+ const pkg =
202
+ source === "npm" ? await readPluginPackage(target) : await readPluginPackage(target).catch(() => undefined)
203
+ const entry = await resolvePluginEntrypoint(spec, target, kind, pkg)
204
+ return {
205
+ spec,
206
+ source,
207
+ target,
208
+ pkg,
209
+ entry,
210
+ }
211
+ }
212
+
213
+ export function readPluginId(id: unknown, spec: string) {
214
+ if (id === undefined) return
215
+ if (typeof id !== "string") throw new TypeError(`Plugin ${spec} has invalid id type ${typeof id}`)
216
+ const value = id.trim()
217
+ if (!value) throw new TypeError(`Plugin ${spec} has an empty id`)
218
+ return value
219
+ }
220
+
221
+ export function readV1Plugin(
222
+ mod: Record<string, unknown>,
223
+ spec: string,
224
+ kind: PluginKind,
225
+ mode: PluginMode = "strict",
226
+ ) {
227
+ const value = mod.default
228
+ if (!isRecord(value)) {
229
+ if (mode === "detect") return
230
+ throw new TypeError(`Plugin ${spec} must default export an object with ${kind}()`)
231
+ }
232
+ if (mode === "detect" && !("id" in value) && !("server" in value) && !("tui" in value)) return
233
+
234
+ const server = "server" in value ? value.server : undefined
235
+ const tui = "tui" in value ? value.tui : undefined
236
+ if (server !== undefined && typeof server !== "function") {
237
+ throw new TypeError(`Plugin ${spec} has invalid server export`)
238
+ }
239
+ if (tui !== undefined && typeof tui !== "function") {
240
+ throw new TypeError(`Plugin ${spec} has invalid tui export`)
241
+ }
242
+ if (server !== undefined && tui !== undefined) {
243
+ throw new TypeError(`Plugin ${spec} must default export either server() or tui(), not both`)
244
+ }
245
+ if (kind === "server" && server === undefined) {
246
+ throw new TypeError(`Plugin ${spec} must default export an object with server()`)
247
+ }
248
+ if (kind === "tui" && tui === undefined) {
249
+ throw new TypeError(`Plugin ${spec} must default export an object with tui()`)
250
+ }
251
+
252
+ return value
253
+ }
254
+
255
+ export async function resolvePluginId(
256
+ source: PluginSource,
257
+ spec: string,
258
+ target: string,
259
+ id: string | undefined,
260
+ pkg?: PluginPackage,
261
+ ) {
262
+ if (source === "file") {
263
+ if (id) return id
264
+ throw new TypeError(`Path plugin ${spec} must export id`)
265
+ }
266
+ if (id) return id
267
+ const hit = pkg ?? (await readPluginPackage(target))
268
+ if (typeof hit.json.name !== "string" || !hit.json.name.trim()) {
269
+ throw new TypeError(`Plugin package ${hit.pkg} is missing name`)
270
+ }
271
+ return hit.json.name.trim()
272
+ }
@@ -0,0 +1,31 @@
1
+ import { Plugin } from "../plugin"
2
+ import { Format } from "../format"
3
+ import { LSP } from "../lsp"
4
+ import { File } from "../file"
5
+ import { FileWatcher } from "../file/watcher"
6
+ import { Snapshot } from "../snapshot"
7
+ import { Project } from "./project"
8
+ import { Vcs } from "./vcs"
9
+ import { Bus } from "../bus"
10
+ import { Command } from "../command"
11
+ import { Instance } from "./instance"
12
+ import { Log } from "@/util/log"
13
+ import { ShareNext } from "@/share/share-next"
14
+
15
+ export async function InstanceBootstrap() {
16
+ Log.Default.info("bootstrapping", { directory: Instance.directory })
17
+ await Plugin.init()
18
+ ShareNext.init()
19
+ Format.init()
20
+ await LSP.init()
21
+ File.init()
22
+ FileWatcher.init()
23
+ Vcs.init()
24
+ Snapshot.init()
25
+
26
+ Bus.subscribe(Command.Event.Executed, async (payload) => {
27
+ if (payload.properties.name === Command.Default.INIT) {
28
+ Project.setInitialized(Instance.project.id)
29
+ }
30
+ })
31
+ }
@@ -0,0 +1,167 @@
1
+ import { GlobalBus } from "@/bus/global"
2
+ import { disposeInstance } from "@/effect/instance-registry"
3
+ import { Filesystem } from "@/util/filesystem"
4
+ import { iife } from "@/util/iife"
5
+ import { Log } from "@/util/log"
6
+ import { Context } from "../util/context"
7
+ import { Project } from "./project"
8
+ import { State } from "./state"
9
+
10
+ export interface InstanceContext {
11
+ directory: string
12
+ worktree: string
13
+ project: Project.Info
14
+ }
15
+
16
+ const context = Context.create<InstanceContext>("instance")
17
+ const cache = new Map<string, Promise<InstanceContext>>()
18
+
19
+ const disposal = {
20
+ all: undefined as Promise<void> | undefined,
21
+ }
22
+
23
+ function emit(directory: string) {
24
+ GlobalBus.emit("event", {
25
+ directory,
26
+ payload: {
27
+ type: "server.instance.disposed",
28
+ properties: {
29
+ directory,
30
+ },
31
+ },
32
+ })
33
+ }
34
+
35
+ function boot(input: { directory: string; init?: () => Promise<any>; project?: Project.Info; worktree?: string }) {
36
+ return iife(async () => {
37
+ const ctx =
38
+ input.project && input.worktree
39
+ ? {
40
+ directory: input.directory,
41
+ worktree: input.worktree,
42
+ project: input.project,
43
+ }
44
+ : await Project.fromDirectory(input.directory).then(({ project, sandbox }) => ({
45
+ directory: input.directory,
46
+ worktree: sandbox,
47
+ project,
48
+ }))
49
+ await context.provide(ctx, async () => {
50
+ await input.init?.()
51
+ })
52
+ return ctx
53
+ })
54
+ }
55
+
56
+ function track(directory: string, next: Promise<InstanceContext>) {
57
+ const task = next.catch((error) => {
58
+ if (cache.get(directory) === task) cache.delete(directory)
59
+ throw error
60
+ })
61
+ cache.set(directory, task)
62
+ return task
63
+ }
64
+
65
+ export const Instance = {
66
+ async provide<R>(input: { directory: string; init?: () => Promise<any>; fn: () => R }): Promise<R> {
67
+ const directory = Filesystem.resolve(input.directory)
68
+ let existing = cache.get(directory)
69
+ if (!existing) {
70
+ Log.Default.info("creating instance", { directory })
71
+ existing = track(
72
+ directory,
73
+ boot({
74
+ directory,
75
+ init: input.init,
76
+ }),
77
+ )
78
+ }
79
+ const ctx = await existing
80
+ return context.provide(ctx, async () => {
81
+ return input.fn()
82
+ })
83
+ },
84
+ get current() {
85
+ return context.use()
86
+ },
87
+ get directory() {
88
+ return context.use().directory
89
+ },
90
+ get worktree() {
91
+ return context.use().worktree
92
+ },
93
+ get project() {
94
+ return context.use().project
95
+ },
96
+ /**
97
+ * Check if a path is within the project boundary.
98
+ * Returns true if path is inside Instance.directory OR Instance.worktree.
99
+ * Paths within the worktree but outside the working directory should not trigger external_directory permission.
100
+ */
101
+ containsPath(filepath: string) {
102
+ if (Filesystem.contains(Instance.directory, filepath)) return true
103
+ // Non-git projects set worktree to "/" which would match ANY absolute path.
104
+ // Skip worktree check in this case to preserve external_directory permissions.
105
+ if (Instance.worktree === "/") return false
106
+ return Filesystem.contains(Instance.worktree, filepath)
107
+ },
108
+ /**
109
+ * Captures the current instance ALS context and returns a wrapper that
110
+ * restores it when called. Use this for callbacks that fire outside the
111
+ * instance async context (native addons, event emitters, timers, etc.).
112
+ */
113
+ bind<F extends (...args: any[]) => any>(fn: F): F {
114
+ const ctx = context.use()
115
+ return ((...args: any[]) => context.provide(ctx, () => fn(...args))) as F
116
+ },
117
+ state<S>(init: () => S, dispose?: (state: Awaited<S>) => Promise<void>): () => S {
118
+ return State.create(() => Instance.directory, init, dispose)
119
+ },
120
+ async reload(input: { directory: string; init?: () => Promise<any>; project?: Project.Info; worktree?: string }) {
121
+ const directory = Filesystem.resolve(input.directory)
122
+ Log.Default.info("reloading instance", { directory })
123
+ await Promise.all([State.dispose(directory), disposeInstance(directory)])
124
+ cache.delete(directory)
125
+ const next = track(directory, boot({ ...input, directory }))
126
+ emit(directory)
127
+ return await next
128
+ },
129
+ async dispose() {
130
+ const directory = Instance.directory
131
+ Log.Default.info("disposing instance", { directory })
132
+ await Promise.all([State.dispose(directory), disposeInstance(directory)])
133
+ cache.delete(directory)
134
+ emit(directory)
135
+ },
136
+ async disposeAll() {
137
+ if (disposal.all) return disposal.all
138
+
139
+ disposal.all = iife(async () => {
140
+ Log.Default.info("disposing all instances")
141
+ const entries = [...cache.entries()]
142
+ for (const [key, value] of entries) {
143
+ if (cache.get(key) !== value) continue
144
+
145
+ const ctx = await value.catch((error) => {
146
+ Log.Default.warn("instance dispose failed", { key, error })
147
+ return undefined
148
+ })
149
+
150
+ if (!ctx) {
151
+ if (cache.get(key) === value) cache.delete(key)
152
+ continue
153
+ }
154
+
155
+ if (cache.get(key) !== value) continue
156
+
157
+ await context.provide(ctx, async () => {
158
+ await Instance.dispose()
159
+ })
160
+ }
161
+ }).finally(() => {
162
+ disposal.all = undefined
163
+ })
164
+
165
+ return disposal.all
166
+ },
167
+ }
@@ -0,0 +1,16 @@
1
+ import { sqliteTable, text, integer } from "drizzle-orm/sqlite-core"
2
+ import { Timestamps } from "../storage/schema.sql"
3
+ import type { ProjectID } from "./schema"
4
+
5
+ export const ProjectTable = sqliteTable("project", {
6
+ id: text().$type<ProjectID>().primaryKey(),
7
+ worktree: text().notNull(),
8
+ vcs: text(),
9
+ name: text(),
10
+ icon_url: text(),
11
+ icon_color: text(),
12
+ ...Timestamps,
13
+ time_initialized: integer(),
14
+ sandboxes: text({ mode: "json" }).notNull().$type<string[]>(),
15
+ commands: text({ mode: "json" }).$type<{ start?: string }>(),
16
+ })