@happy-creative/iroder 1.0.0

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 (697) hide show
  1. package/.qwen/settings.json +8 -0
  2. package/.qwen/settings.json.orig +7 -0
  3. package/AGENTS.md +69 -0
  4. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  5. package/Dockerfile +20 -0
  6. package/README.md +15 -0
  7. package/bin/iroder +182 -0
  8. package/docker-compose.iroder.yml +18 -0
  9. package/drizzle.config.ts +10 -0
  10. package/git +0 -0
  11. package/migration/20260127222353_familiar_lady_ursula/migration.sql +90 -0
  12. package/migration/20260127222353_familiar_lady_ursula/snapshot.json +796 -0
  13. package/migration/20260211171708_add_project_commands/migration.sql +1 -0
  14. package/migration/20260211171708_add_project_commands/snapshot.json +806 -0
  15. package/migration/20260213144116_wakeful_the_professor/migration.sql +11 -0
  16. package/migration/20260213144116_wakeful_the_professor/snapshot.json +897 -0
  17. package/migration/20260225215848_workspace/migration.sql +7 -0
  18. package/migration/20260225215848_workspace/snapshot.json +959 -0
  19. package/migration/20260227213759_add_session_workspace_id/migration.sql +2 -0
  20. package/migration/20260227213759_add_session_workspace_id/snapshot.json +983 -0
  21. package/migration/20260228203230_blue_harpoon/migration.sql +17 -0
  22. package/migration/20260228203230_blue_harpoon/snapshot.json +1102 -0
  23. package/migration/20260303231226_add_workspace_fields/migration.sql +5 -0
  24. package/migration/20260303231226_add_workspace_fields/snapshot.json +1013 -0
  25. package/migration/20260309230000_move_org_to_state/migration.sql +3 -0
  26. package/migration/20260309230000_move_org_to_state/snapshot.json +1156 -0
  27. package/migration/20260312043431_session_message_cursor/migration.sql +4 -0
  28. package/migration/20260312043431_session_message_cursor/snapshot.json +1168 -0
  29. package/migration/20260323234822_events/migration.sql +13 -0
  30. package/migration/20260323234822_events/snapshot.json +1271 -0
  31. package/package.json +180 -0
  32. package/parsers-config.ts +290 -0
  33. package/script/build-node.ts +60 -0
  34. package/script/build.ts +281 -0
  35. package/script/check-migrations.ts +16 -0
  36. package/script/e2e-local-real-key.ts +197 -0
  37. package/script/fix-node-pty.ts +28 -0
  38. package/script/postinstall.mjs +131 -0
  39. package/script/publish-all.sh +68 -0
  40. package/script/publish.ts +181 -0
  41. package/script/schema.ts +63 -0
  42. package/script/seed-e2e.ts +60 -0
  43. package/script/upgrade-opentui.ts +64 -0
  44. package/specs/effect-migration.md +310 -0
  45. package/specs/tui-plugins.md +436 -0
  46. package/specs/v2/keymappings.md +10 -0
  47. package/specs/v2/message-shape.md +136 -0
  48. package/src/account/account.sql.ts +39 -0
  49. package/src/account/index.ts +488 -0
  50. package/src/account/repo.ts +166 -0
  51. package/src/account/schema.ts +119 -0
  52. package/src/account/url.ts +8 -0
  53. package/src/acp/README.md +174 -0
  54. package/src/acp/agent.ts +1847 -0
  55. package/src/acp/session.ts +116 -0
  56. package/src/acp/types.ts +24 -0
  57. package/src/agent/agent.ts +422 -0
  58. package/src/agent/generate.txt +75 -0
  59. package/src/agent/prompt/compaction.txt +15 -0
  60. package/src/agent/prompt/explore.txt +18 -0
  61. package/src/agent/prompt/summary.txt +11 -0
  62. package/src/agent/prompt/title.txt +44 -0
  63. package/src/auth/index.ts +110 -0
  64. package/src/bus/bus-event.ts +40 -0
  65. package/src/bus/global.ts +10 -0
  66. package/src/bus/index.ts +185 -0
  67. package/src/cli/bootstrap.ts +17 -0
  68. package/src/cli/cmd/account.ts +257 -0
  69. package/src/cli/cmd/acp.ts +72 -0
  70. package/src/cli/cmd/agent.ts +245 -0
  71. package/src/cli/cmd/cmd.ts +7 -0
  72. package/src/cli/cmd/db.ts +120 -0
  73. package/src/cli/cmd/debug/agent.ts +170 -0
  74. package/src/cli/cmd/debug/config.ts +16 -0
  75. package/src/cli/cmd/debug/file.ts +97 -0
  76. package/src/cli/cmd/debug/index.ts +48 -0
  77. package/src/cli/cmd/debug/lsp.ts +53 -0
  78. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  79. package/src/cli/cmd/debug/scrap.ts +16 -0
  80. package/src/cli/cmd/debug/skill.ts +16 -0
  81. package/src/cli/cmd/debug/snapshot.ts +52 -0
  82. package/src/cli/cmd/export.ts +89 -0
  83. package/src/cli/cmd/generate.ts +38 -0
  84. package/src/cli/cmd/github.ts +1647 -0
  85. package/src/cli/cmd/import.ts +207 -0
  86. package/src/cli/cmd/mcp.ts +754 -0
  87. package/src/cli/cmd/models.ts +78 -0
  88. package/src/cli/cmd/plug.ts +233 -0
  89. package/src/cli/cmd/pr.ts +127 -0
  90. package/src/cli/cmd/providers.ts +480 -0
  91. package/src/cli/cmd/run.ts +692 -0
  92. package/src/cli/cmd/serve.ts +23 -0
  93. package/src/cli/cmd/session.ts +159 -0
  94. package/src/cli/cmd/stats.ts +410 -0
  95. package/src/cli/cmd/tui/app.tsx +940 -0
  96. package/src/cli/cmd/tui/attach.ts +88 -0
  97. package/src/cli/cmd/tui/component/border.tsx +21 -0
  98. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  99. package/src/cli/cmd/tui/component/dialog-command.tsx +171 -0
  100. package/src/cli/cmd/tui/component/dialog-console-org.tsx +103 -0
  101. package/src/cli/cmd/tui/component/dialog-go-upsell.tsx +100 -0
  102. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  103. package/src/cli/cmd/tui/component/dialog-model.tsx +183 -0
  104. package/src/cli/cmd/tui/component/dialog-provider.tsx +360 -0
  105. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  106. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  107. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  108. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  109. package/src/cli/cmd/tui/component/dialog-status.tsx +168 -0
  110. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  111. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  112. package/src/cli/cmd/tui/component/dialog-variant.tsx +39 -0
  113. package/src/cli/cmd/tui/component/dialog-workspace-list.tsx +320 -0
  114. package/src/cli/cmd/tui/component/error-component.tsx +93 -0
  115. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  116. package/src/cli/cmd/tui/component/plugin-route-missing.tsx +14 -0
  117. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +672 -0
  118. package/src/cli/cmd/tui/component/prompt/frecency.tsx +90 -0
  119. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  120. package/src/cli/cmd/tui/component/prompt/index.tsx +1261 -0
  121. package/src/cli/cmd/tui/component/prompt/part.ts +16 -0
  122. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  123. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  124. package/src/cli/cmd/tui/component/startup-loading.tsx +63 -0
  125. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  126. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  127. package/src/cli/cmd/tui/component/workspace/dialog-session-list.tsx +151 -0
  128. package/src/cli/cmd/tui/context/args.tsx +15 -0
  129. package/src/cli/cmd/tui/context/directory.ts +13 -0
  130. package/src/cli/cmd/tui/context/exit.tsx +60 -0
  131. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  132. package/src/cli/cmd/tui/context/keybind.tsx +105 -0
  133. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  134. package/src/cli/cmd/tui/context/local.tsx +412 -0
  135. package/src/cli/cmd/tui/context/plugin-keybinds.ts +41 -0
  136. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  137. package/src/cli/cmd/tui/context/route.tsx +52 -0
  138. package/src/cli/cmd/tui/context/sdk.tsx +115 -0
  139. package/src/cli/cmd/tui/context/sync.tsx +516 -0
  140. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  141. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  142. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  143. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  144. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  145. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  146. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  147. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  148. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  149. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  150. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  151. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  152. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  153. package/src/cli/cmd/tui/context/theme/iroder.json +245 -0
  154. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  155. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  156. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  157. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  158. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  159. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  160. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  161. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  162. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  163. package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
  164. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  165. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  166. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  167. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  168. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  169. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  170. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  171. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  172. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  173. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  174. package/src/cli/cmd/tui/context/theme.tsx +1236 -0
  175. package/src/cli/cmd/tui/context/tui-config.tsx +9 -0
  176. package/src/cli/cmd/tui/event.ts +49 -0
  177. package/src/cli/cmd/tui/feature-plugins/home/footer.tsx +93 -0
  178. package/src/cli/cmd/tui/feature-plugins/home/tips-view.tsx +155 -0
  179. package/src/cli/cmd/tui/feature-plugins/home/tips.tsx +50 -0
  180. package/src/cli/cmd/tui/feature-plugins/sidebar/context.tsx +63 -0
  181. package/src/cli/cmd/tui/feature-plugins/sidebar/files.tsx +62 -0
  182. package/src/cli/cmd/tui/feature-plugins/sidebar/footer.tsx +93 -0
  183. package/src/cli/cmd/tui/feature-plugins/sidebar/lsp.tsx +66 -0
  184. package/src/cli/cmd/tui/feature-plugins/sidebar/mcp.tsx +96 -0
  185. package/src/cli/cmd/tui/feature-plugins/sidebar/todo.tsx +48 -0
  186. package/src/cli/cmd/tui/feature-plugins/system/plugins.tsx +270 -0
  187. package/src/cli/cmd/tui/plugin/api.tsx +397 -0
  188. package/src/cli/cmd/tui/plugin/index.ts +3 -0
  189. package/src/cli/cmd/tui/plugin/internal.ts +27 -0
  190. package/src/cli/cmd/tui/plugin/runtime.ts +1031 -0
  191. package/src/cli/cmd/tui/plugin/slots.tsx +60 -0
  192. package/src/cli/cmd/tui/routes/home.tsx +84 -0
  193. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +65 -0
  194. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +110 -0
  195. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  196. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  197. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  198. package/src/cli/cmd/tui/routes/session/index.tsx +2281 -0
  199. package/src/cli/cmd/tui/routes/session/permission.tsx +691 -0
  200. package/src/cli/cmd/tui/routes/session/question.tsx +468 -0
  201. package/src/cli/cmd/tui/routes/session/sidebar.tsx +74 -0
  202. package/src/cli/cmd/tui/routes/session/subagent-footer.tsx +131 -0
  203. package/src/cli/cmd/tui/thread.ts +241 -0
  204. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  205. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +89 -0
  206. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +211 -0
  207. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  208. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +115 -0
  209. package/src/cli/cmd/tui/ui/dialog-select.tsx +417 -0
  210. package/src/cli/cmd/tui/ui/dialog.tsx +192 -0
  211. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  212. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  213. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  214. package/src/cli/cmd/tui/util/clipboard.ts +192 -0
  215. package/src/cli/cmd/tui/util/editor.ts +37 -0
  216. package/src/cli/cmd/tui/util/model.ts +23 -0
  217. package/src/cli/cmd/tui/util/provider-origin.ts +7 -0
  218. package/src/cli/cmd/tui/util/scroll.ts +23 -0
  219. package/src/cli/cmd/tui/util/selection.ts +25 -0
  220. package/src/cli/cmd/tui/util/signal.ts +7 -0
  221. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  222. package/src/cli/cmd/tui/util/transcript.ts +112 -0
  223. package/src/cli/cmd/tui/win32.ts +129 -0
  224. package/src/cli/cmd/tui/worker.ts +195 -0
  225. package/src/cli/cmd/uninstall.ts +353 -0
  226. package/src/cli/cmd/upgrade.ts +73 -0
  227. package/src/cli/cmd/web.ts +83 -0
  228. package/src/cli/commands.ts +58 -0
  229. package/src/cli/effect/prompt.ts +25 -0
  230. package/src/cli/error.ts +50 -0
  231. package/src/cli/heap.ts +59 -0
  232. package/src/cli/llm-ready.ts +22 -0
  233. package/src/cli/logo.ts +8 -0
  234. package/src/cli/network.ts +60 -0
  235. package/src/cli/ui.ts +132 -0
  236. package/src/cli/upgrade.ts +31 -0
  237. package/src/command/index.ts +195 -0
  238. package/src/command/template/initialize.txt +66 -0
  239. package/src/command/template/review.txt +101 -0
  240. package/src/config/config.ts +1659 -0
  241. package/src/config/console-state.ts +15 -0
  242. package/src/config/markdown.ts +99 -0
  243. package/src/config/paths.ts +167 -0
  244. package/src/config/tui-migrate.ts +156 -0
  245. package/src/config/tui-schema.ts +37 -0
  246. package/src/config/tui.ts +179 -0
  247. package/src/control-plane/adaptors/index.ts +23 -0
  248. package/src/control-plane/adaptors/remote-acp.ts +35 -0
  249. package/src/control-plane/adaptors/remote-http.ts +35 -0
  250. package/src/control-plane/adaptors/worktree.ts +42 -0
  251. package/src/control-plane/schema.ts +17 -0
  252. package/src/control-plane/sse.ts +66 -0
  253. package/src/control-plane/types.ts +32 -0
  254. package/src/control-plane/workspace.sql.ts +17 -0
  255. package/src/control-plane/workspace.ts +169 -0
  256. package/src/effect/cross-spawn-spawner.ts +502 -0
  257. package/src/effect/instance-ref.ts +6 -0
  258. package/src/effect/instance-registry.ts +12 -0
  259. package/src/effect/instance-state.ts +82 -0
  260. package/src/effect/oltp.ts +34 -0
  261. package/src/effect/run-service.ts +34 -0
  262. package/src/effect/runner.ts +216 -0
  263. package/src/env/index.ts +28 -0
  264. package/src/file/ignore.ts +82 -0
  265. package/src/file/index.ts +686 -0
  266. package/src/file/protected.ts +59 -0
  267. package/src/file/ripgrep.ts +376 -0
  268. package/src/file/time.ts +133 -0
  269. package/src/file/watcher.ts +171 -0
  270. package/src/filesystem/index.ts +236 -0
  271. package/src/flag/flag.ts +171 -0
  272. package/src/format/formatter.ts +413 -0
  273. package/src/format/index.ts +203 -0
  274. package/src/git/index.ts +303 -0
  275. package/src/global/index.ts +54 -0
  276. package/src/id/id.ts +85 -0
  277. package/src/ide/index.ts +74 -0
  278. package/src/index.ts +202 -0
  279. package/src/installation/index.ts +356 -0
  280. package/src/installation/meta.ts +7 -0
  281. package/src/lsp/client.ts +252 -0
  282. package/src/lsp/index.ts +558 -0
  283. package/src/lsp/language.ts +120 -0
  284. package/src/lsp/launch.ts +21 -0
  285. package/src/lsp/server.ts +1968 -0
  286. package/src/mcp/auth.ts +173 -0
  287. package/src/mcp/index.ts +921 -0
  288. package/src/mcp/oauth-callback.ts +216 -0
  289. package/src/mcp/oauth-provider.ts +186 -0
  290. package/src/node.ts +6 -0
  291. package/src/npm/index.ts +188 -0
  292. package/src/patch/index.ts +680 -0
  293. package/src/permission/arity.ts +163 -0
  294. package/src/permission/evaluate.ts +15 -0
  295. package/src/permission/index.ts +325 -0
  296. package/src/permission/schema.ts +17 -0
  297. package/src/plugin/cloudflare.ts +67 -0
  298. package/src/plugin/codex.ts +608 -0
  299. package/src/plugin/github-copilot/copilot.ts +361 -0
  300. package/src/plugin/github-copilot/models.ts +144 -0
  301. package/src/plugin/index.ts +293 -0
  302. package/src/plugin/install.ts +439 -0
  303. package/src/plugin/loader.ts +174 -0
  304. package/src/plugin/meta.ts +188 -0
  305. package/src/plugin/shared.ts +323 -0
  306. package/src/project/bootstrap.ts +31 -0
  307. package/src/project/instance.ts +175 -0
  308. package/src/project/project.sql.ts +16 -0
  309. package/src/project/project.ts +519 -0
  310. package/src/project/schema.ts +16 -0
  311. package/src/project/state.ts +70 -0
  312. package/src/project/vcs.ts +254 -0
  313. package/src/provider/auth.ts +253 -0
  314. package/src/provider/error.ts +197 -0
  315. package/src/provider/models.ts +159 -0
  316. package/src/provider/provider.ts +1748 -0
  317. package/src/provider/schema.ts +38 -0
  318. package/src/provider/sdk/copilot/README.md +5 -0
  319. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +170 -0
  320. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  321. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +19 -0
  322. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  323. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +815 -0
  324. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  325. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  326. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +83 -0
  327. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  328. package/src/provider/sdk/copilot/index.ts +2 -0
  329. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  330. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +335 -0
  331. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  332. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  333. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  334. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +214 -0
  335. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1769 -0
  336. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +173 -0
  337. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  338. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +87 -0
  339. package/src/provider/sdk/copilot/responses/tool/file-search.ts +127 -0
  340. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +114 -0
  341. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +64 -0
  342. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +103 -0
  343. package/src/provider/sdk/copilot/responses/tool/web-search.ts +102 -0
  344. package/src/provider/transform.ts +1051 -0
  345. package/src/pty/index.ts +397 -0
  346. package/src/pty/pty.bun.ts +26 -0
  347. package/src/pty/pty.node.ts +27 -0
  348. package/src/pty/pty.ts +25 -0
  349. package/src/pty/schema.ts +17 -0
  350. package/src/question/index.ts +224 -0
  351. package/src/question/schema.ts +17 -0
  352. package/src/runtime/adapters/acp/index.ts +127 -0
  353. package/src/runtime/adapters/local/index.ts +6 -0
  354. package/src/runtime/core/agent.ts +13 -0
  355. package/src/runtime/core/mcp.ts +10 -0
  356. package/src/runtime/core/ports.ts +16 -0
  357. package/src/runtime/core/service.ts +19 -0
  358. package/src/runtime/core/session-summarize.ts +52 -0
  359. package/src/runtime/core/session.ts +110 -0
  360. package/src/runtime/core/skill.ts +10 -0
  361. package/src/runtime/core/transport.ts +10 -0
  362. package/src/runtime/factory.ts +20 -0
  363. package/src/runtime/ports.ts +74 -0
  364. package/src/server/error.ts +36 -0
  365. package/src/server/event.ts +7 -0
  366. package/src/server/instance.ts +322 -0
  367. package/src/server/mdns.ts +60 -0
  368. package/src/server/middleware.ts +33 -0
  369. package/src/server/projectors.ts +28 -0
  370. package/src/server/proxy.ts +134 -0
  371. package/src/server/router.ts +161 -0
  372. package/src/server/routes/config.ts +92 -0
  373. package/src/server/routes/event.ts +83 -0
  374. package/src/server/routes/experimental.ts +379 -0
  375. package/src/server/routes/file.ts +197 -0
  376. package/src/server/routes/global.ts +312 -0
  377. package/src/server/routes/mcp.ts +226 -0
  378. package/src/server/routes/permission.ts +69 -0
  379. package/src/server/routes/project.ts +118 -0
  380. package/src/server/routes/provider.ts +171 -0
  381. package/src/server/routes/pty.ts +210 -0
  382. package/src/server/routes/question.ts +99 -0
  383. package/src/server/routes/session.ts +1011 -0
  384. package/src/server/routes/tui.ts +379 -0
  385. package/src/server/routes/workspace.ts +94 -0
  386. package/src/server/server.ts +367 -0
  387. package/src/session/compaction.ts +425 -0
  388. package/src/session/index.ts +887 -0
  389. package/src/session/instruction.ts +258 -0
  390. package/src/session/llm.ts +412 -0
  391. package/src/session/message-v2.ts +1038 -0
  392. package/src/session/message.ts +191 -0
  393. package/src/session/overflow.ts +22 -0
  394. package/src/session/processor.ts +515 -0
  395. package/src/session/projectors.ts +135 -0
  396. package/src/session/prompt/anthropic.txt +105 -0
  397. package/src/session/prompt/beast.txt +147 -0
  398. package/src/session/prompt/build-switch.txt +5 -0
  399. package/src/session/prompt/codex.txt +79 -0
  400. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  401. package/src/session/prompt/default.txt +105 -0
  402. package/src/session/prompt/gemini.txt +155 -0
  403. package/src/session/prompt/gpt.txt +107 -0
  404. package/src/session/prompt/kimi.txt +95 -0
  405. package/src/session/prompt/max-steps.txt +16 -0
  406. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  407. package/src/session/prompt/plan.txt +26 -0
  408. package/src/session/prompt/trinity.txt +97 -0
  409. package/src/session/prompt.ts +1908 -0
  410. package/src/session/retry.ts +123 -0
  411. package/src/session/revert.ts +176 -0
  412. package/src/session/schema.ts +38 -0
  413. package/src/session/session.sql.ts +103 -0
  414. package/src/session/status.ts +102 -0
  415. package/src/session/summary.ts +177 -0
  416. package/src/session/system.ts +76 -0
  417. package/src/session/todo.ts +95 -0
  418. package/src/share/share-next.ts +370 -0
  419. package/src/share/share.sql.ts +13 -0
  420. package/src/shell/shell.ts +110 -0
  421. package/src/skill/discovery.ts +116 -0
  422. package/src/skill/index.ts +289 -0
  423. package/src/snapshot/index.ts +723 -0
  424. package/src/sql.d.ts +4 -0
  425. package/src/storage/db.bun.ts +8 -0
  426. package/src/storage/db.node.ts +8 -0
  427. package/src/storage/db.ts +174 -0
  428. package/src/storage/json-migration.ts +425 -0
  429. package/src/storage/schema.sql.ts +10 -0
  430. package/src/storage/schema.ts +5 -0
  431. package/src/storage/storage.ts +353 -0
  432. package/src/sync/README.md +179 -0
  433. package/src/sync/event.sql.ts +16 -0
  434. package/src/sync/index.ts +263 -0
  435. package/src/sync/schema.ts +14 -0
  436. package/src/testing/llm-server.ts +2 -0
  437. package/src/tool/apply_patch.ts +279 -0
  438. package/src/tool/apply_patch.txt +33 -0
  439. package/src/tool/bash.ts +498 -0
  440. package/src/tool/bash.txt +117 -0
  441. package/src/tool/codesearch.ts +133 -0
  442. package/src/tool/codesearch.txt +12 -0
  443. package/src/tool/edit.ts +666 -0
  444. package/src/tool/edit.txt +10 -0
  445. package/src/tool/external-directory.ts +46 -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/plan-enter.txt +14 -0
  458. package/src/tool/plan-exit.txt +13 -0
  459. package/src/tool/plan.ts +131 -0
  460. package/src/tool/question.ts +46 -0
  461. package/src/tool/question.txt +10 -0
  462. package/src/tool/read.ts +330 -0
  463. package/src/tool/read.txt +14 -0
  464. package/src/tool/registry.ts +303 -0
  465. package/src/tool/schema.ts +17 -0
  466. package/src/tool/skill.ts +120 -0
  467. package/src/tool/task.ts +192 -0
  468. package/src/tool/task.txt +57 -0
  469. package/src/tool/todo.ts +48 -0
  470. package/src/tool/todowrite.txt +167 -0
  471. package/src/tool/tool.ts +137 -0
  472. package/src/tool/truncate.ts +144 -0
  473. package/src/tool/truncation-dir.ts +4 -0
  474. package/src/tool/webfetch.ts +210 -0
  475. package/src/tool/webfetch.txt +13 -0
  476. package/src/tool/websearch.ts +151 -0
  477. package/src/tool/websearch.txt +14 -0
  478. package/src/tool/write.ts +84 -0
  479. package/src/tool/write.txt +8 -0
  480. package/src/url/site.ts +118 -0
  481. package/src/util/abort.ts +35 -0
  482. package/src/util/archive.ts +17 -0
  483. package/src/util/color.ts +19 -0
  484. package/src/util/context.ts +25 -0
  485. package/src/util/data-url.ts +9 -0
  486. package/src/util/defer.ts +12 -0
  487. package/src/util/effect-http-client.ts +11 -0
  488. package/src/util/effect-zod.ts +98 -0
  489. package/src/util/error.ts +77 -0
  490. package/src/util/filesystem.ts +245 -0
  491. package/src/util/flock.ts +333 -0
  492. package/src/util/fn.ts +21 -0
  493. package/src/util/format.ts +20 -0
  494. package/src/util/glob.ts +34 -0
  495. package/src/util/hash.ts +7 -0
  496. package/src/util/iife.ts +3 -0
  497. package/src/util/keybind.ts +103 -0
  498. package/src/util/lazy.ts +23 -0
  499. package/src/util/locale.ts +81 -0
  500. package/src/util/lock.ts +98 -0
  501. package/src/util/log.ts +182 -0
  502. package/src/util/network.ts +9 -0
  503. package/src/util/process.ts +176 -0
  504. package/src/util/queue.ts +32 -0
  505. package/src/util/record.ts +3 -0
  506. package/src/util/rpc.ts +66 -0
  507. package/src/util/schema.ts +53 -0
  508. package/src/util/scrap.ts +10 -0
  509. package/src/util/signal.ts +12 -0
  510. package/src/util/timeout.ts +14 -0
  511. package/src/util/token.ts +7 -0
  512. package/src/util/update-schema.ts +13 -0
  513. package/src/util/which.ts +14 -0
  514. package/src/util/wildcard.ts +59 -0
  515. package/src/worktree/index.ts +612 -0
  516. package/sst-env.d.ts +10 -0
  517. package/test/AGENTS.md +81 -0
  518. package/test/account/repo.test.ts +352 -0
  519. package/test/account/service.test.ts +456 -0
  520. package/test/acp/agent-interface.test.ts +51 -0
  521. package/test/acp/event-subscription.test.ts +685 -0
  522. package/test/agent/agent.test.ts +717 -0
  523. package/test/auth/auth.test.ts +58 -0
  524. package/test/bus/bus-effect.test.ts +164 -0
  525. package/test/bus/bus-integration.test.ts +87 -0
  526. package/test/bus/bus.test.ts +219 -0
  527. package/test/cli/account.test.ts +26 -0
  528. package/test/cli/cmd/tui/prompt-part.test.ts +47 -0
  529. package/test/cli/commands.test.ts +49 -0
  530. package/test/cli/error.test.ts +18 -0
  531. package/test/cli/github-action.test.ts +198 -0
  532. package/test/cli/github-remote.test.ts +80 -0
  533. package/test/cli/import.test.ts +54 -0
  534. package/test/cli/llm-ready.test.ts +49 -0
  535. package/test/cli/plugin-auth-picker.test.ts +120 -0
  536. package/test/cli/tui/keybind-plugin.test.ts +90 -0
  537. package/test/cli/tui/plugin-add.test.ts +107 -0
  538. package/test/cli/tui/plugin-install.test.ts +89 -0
  539. package/test/cli/tui/plugin-lifecycle.test.ts +225 -0
  540. package/test/cli/tui/plugin-loader-entrypoint.test.ts +492 -0
  541. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  542. package/test/cli/tui/plugin-loader.test.ts +752 -0
  543. package/test/cli/tui/plugin-toggle.test.ts +159 -0
  544. package/test/cli/tui/slot-replace.test.tsx +47 -0
  545. package/test/cli/tui/theme-store.test.ts +51 -0
  546. package/test/cli/tui/thread.test.ts +128 -0
  547. package/test/cli/tui/transcript.test.ts +426 -0
  548. package/test/config/agent-color.test.ts +71 -0
  549. package/test/config/config.test.ts +2364 -0
  550. package/test/config/fixtures/empty-frontmatter.md +4 -0
  551. package/test/config/fixtures/frontmatter.md +28 -0
  552. package/test/config/fixtures/markdown-header.md +11 -0
  553. package/test/config/fixtures/no-frontmatter.md +1 -0
  554. package/test/config/fixtures/weird-model-id.md +13 -0
  555. package/test/config/markdown.test.ts +228 -0
  556. package/test/config/tui.test.ts +800 -0
  557. package/test/control-plane/sse.test.ts +56 -0
  558. package/test/effect/cross-spawn-spawner.test.ts +412 -0
  559. package/test/effect/instance-state.test.ts +482 -0
  560. package/test/effect/run-service.test.ts +46 -0
  561. package/test/effect/runner.test.ts +523 -0
  562. package/test/fake/provider.ts +81 -0
  563. package/test/file/fsmonitor.test.ts +62 -0
  564. package/test/file/ignore.test.ts +10 -0
  565. package/test/file/index.test.ts +946 -0
  566. package/test/file/path-traversal.test.ts +198 -0
  567. package/test/file/ripgrep.test.ts +54 -0
  568. package/test/file/time.test.ts +445 -0
  569. package/test/file/watcher.test.ts +247 -0
  570. package/test/filesystem/filesystem.test.ts +319 -0
  571. package/test/fixture/db.ts +11 -0
  572. package/test/fixture/fixture.test.ts +26 -0
  573. package/test/fixture/fixture.ts +174 -0
  574. package/test/fixture/flock-worker.ts +72 -0
  575. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  576. package/test/fixture/plug-worker.ts +93 -0
  577. package/test/fixture/plugin-meta-worker.ts +26 -0
  578. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  579. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  580. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  581. package/test/fixture/skills/index.json +6 -0
  582. package/test/fixture/tui-plugin.ts +328 -0
  583. package/test/fixture/tui-runtime.ts +27 -0
  584. package/test/format/format.test.ts +171 -0
  585. package/test/git/git.test.ts +128 -0
  586. package/test/ide/ide.test.ts +82 -0
  587. package/test/installation/installation.test.ts +151 -0
  588. package/test/keybind.test.ts +421 -0
  589. package/test/lib/effect.ts +53 -0
  590. package/test/lib/filesystem.ts +10 -0
  591. package/test/lib/llm-server.ts +795 -0
  592. package/test/lsp/client.test.ts +95 -0
  593. package/test/lsp/index.test.ts +138 -0
  594. package/test/lsp/launch.test.ts +22 -0
  595. package/test/lsp/lifecycle.test.ts +147 -0
  596. package/test/mcp/headers.test.ts +153 -0
  597. package/test/mcp/lifecycle.test.ts +750 -0
  598. package/test/mcp/oauth-auto-connect.test.ts +199 -0
  599. package/test/mcp/oauth-browser.test.ts +249 -0
  600. package/test/memory/abort-leak.test.ts +151 -0
  601. package/test/npm.test.ts +18 -0
  602. package/test/patch/patch.test.ts +348 -0
  603. package/test/permission/arity.test.ts +33 -0
  604. package/test/permission/next.test.ts +1148 -0
  605. package/test/permission-task.test.ts +323 -0
  606. package/test/plugin/auth-override.test.ts +74 -0
  607. package/test/plugin/codex.test.ts +123 -0
  608. package/test/plugin/github-copilot-models.test.ts +117 -0
  609. package/test/plugin/install-concurrency.test.ts +140 -0
  610. package/test/plugin/install.test.ts +570 -0
  611. package/test/plugin/loader-shared.test.ts +1136 -0
  612. package/test/plugin/meta.test.ts +137 -0
  613. package/test/plugin/shared.test.ts +88 -0
  614. package/test/plugin/trigger.test.ts +111 -0
  615. package/test/preload.ts +90 -0
  616. package/test/project/migrate-global.test.ts +141 -0
  617. package/test/project/project.test.ts +459 -0
  618. package/test/project/state.test.ts +115 -0
  619. package/test/project/vcs.test.ts +228 -0
  620. package/test/project/worktree-remove.test.ts +96 -0
  621. package/test/project/worktree.test.ts +173 -0
  622. package/test/provider/amazon-bedrock.test.ts +447 -0
  623. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  624. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  625. package/test/provider/gitlab-duo.test.ts +412 -0
  626. package/test/provider/provider.test.ts +2494 -0
  627. package/test/provider/transform.test.ts +2839 -0
  628. package/test/pty/pty-output-isolation.test.ts +141 -0
  629. package/test/pty/pty-session.test.ts +92 -0
  630. package/test/pty/pty-shell.test.ts +59 -0
  631. package/test/question/question.test.ts +453 -0
  632. package/test/runtime/acp-adapter.test.ts +84 -0
  633. package/test/runtime/core-service.test.ts +16 -0
  634. package/test/runtime/session-summarize.test.ts +104 -0
  635. package/test/server/global-session-list.test.ts +89 -0
  636. package/test/server/project-init-git.test.ts +121 -0
  637. package/test/server/router.test.ts +52 -0
  638. package/test/server/session-actions.test.ts +83 -0
  639. package/test/server/session-list.test.ts +98 -0
  640. package/test/server/session-messages.test.ts +159 -0
  641. package/test/server/session-select.test.ts +84 -0
  642. package/test/session/compaction.test.ts +1239 -0
  643. package/test/session/instruction.test.ts +286 -0
  644. package/test/session/llm.test.ts +1093 -0
  645. package/test/session/message-v2.test.ts +957 -0
  646. package/test/session/messages-pagination.test.ts +885 -0
  647. package/test/session/processor-effect.test.ts +741 -0
  648. package/test/session/prompt-effect.test.ts +1339 -0
  649. package/test/session/prompt.test.ts +533 -0
  650. package/test/session/retry.test.ts +251 -0
  651. package/test/session/revert-compact.test.ts +621 -0
  652. package/test/session/session.test.ts +142 -0
  653. package/test/session/snapshot-tool-race.test.ts +242 -0
  654. package/test/session/structured-output-integration.test.ts +233 -0
  655. package/test/session/structured-output.test.ts +391 -0
  656. package/test/session/system.test.ts +59 -0
  657. package/test/share/share-next.test.ts +332 -0
  658. package/test/shell/shell.test.ts +73 -0
  659. package/test/skill/discovery.test.ts +116 -0
  660. package/test/skill/skill.test.ts +428 -0
  661. package/test/snapshot/snapshot.test.ts +1397 -0
  662. package/test/storage/db.test.ts +14 -0
  663. package/test/storage/json-migration.test.ts +832 -0
  664. package/test/storage/storage.test.ts +295 -0
  665. package/test/sync/index.test.ts +191 -0
  666. package/test/tool/apply_patch.test.ts +565 -0
  667. package/test/tool/bash.test.ts +1099 -0
  668. package/test/tool/edit.test.ts +681 -0
  669. package/test/tool/external-directory.test.ts +198 -0
  670. package/test/tool/fixtures/large-image.png +0 -0
  671. package/test/tool/fixtures/models-api.json +65179 -0
  672. package/test/tool/grep.test.ts +111 -0
  673. package/test/tool/question.test.ts +126 -0
  674. package/test/tool/read.test.ts +468 -0
  675. package/test/tool/registry.test.ts +157 -0
  676. package/test/tool/skill.test.ts +170 -0
  677. package/test/tool/task.test.ts +412 -0
  678. package/test/tool/tool-define.test.ts +49 -0
  679. package/test/tool/truncation.test.ts +161 -0
  680. package/test/tool/webfetch.test.ts +96 -0
  681. package/test/tool/write.test.ts +353 -0
  682. package/test/util/data-url.test.ts +14 -0
  683. package/test/util/effect-zod.test.ts +61 -0
  684. package/test/util/error.test.ts +38 -0
  685. package/test/util/filesystem.test.ts +656 -0
  686. package/test/util/flock.test.ts +383 -0
  687. package/test/util/format.test.ts +59 -0
  688. package/test/util/glob.test.ts +164 -0
  689. package/test/util/iife.test.ts +36 -0
  690. package/test/util/lazy.test.ts +50 -0
  691. package/test/util/lock.test.ts +72 -0
  692. package/test/util/module.test.ts +59 -0
  693. package/test/util/process.test.ts +128 -0
  694. package/test/util/timeout.test.ts +21 -0
  695. package/test/util/which.test.ts +100 -0
  696. package/test/util/wildcard.test.ts +90 -0
  697. package/tsconfig.json +23 -0
@@ -0,0 +1,1031 @@
1
+ import "@opentui/solid/runtime-plugin-support"
2
+ import {
3
+ type TuiDispose,
4
+ type TuiPlugin,
5
+ type TuiPluginApi,
6
+ type TuiPluginInstallResult,
7
+ type TuiPluginModule,
8
+ type TuiPluginMeta,
9
+ type TuiPluginStatus,
10
+ type TuiSlotPlugin,
11
+ type TuiTheme,
12
+ } from "@happy-creative/plugin/tui"
13
+ import path from "path"
14
+ import { fileURLToPath } from "url"
15
+
16
+ import { Config } from "@/config/config"
17
+ import { TuiConfig } from "@/config/tui"
18
+ import { Log } from "@/util/log"
19
+ import { errorData, errorMessage } from "@/util/error"
20
+ import { isRecord } from "@/util/record"
21
+ import { Instance } from "@/project/instance"
22
+ import {
23
+ readPackageThemes,
24
+ readPluginId,
25
+ readV1Plugin,
26
+ resolvePluginId,
27
+ type PluginPackage,
28
+ type PluginSource,
29
+ } from "@/plugin/shared"
30
+ import { PluginLoader } from "@/plugin/loader"
31
+ import { PluginMeta } from "@/plugin/meta"
32
+ import { installPlugin as installModulePlugin, patchPluginConfig, readPluginManifest } from "@/plugin/install"
33
+ import { hasTheme, upsertTheme } from "../context/theme"
34
+ import { Global } from "@/global"
35
+ import { Filesystem } from "@/util/filesystem"
36
+ import { Process } from "@/util/process"
37
+ import { Flock } from "@/util/flock"
38
+ import { Flag } from "@/flag/flag"
39
+ import { INTERNAL_TUI_PLUGINS, type InternalTuiPlugin } from "./internal"
40
+ import { setupSlots, Slot as View } from "./slots"
41
+ import type { HostPluginApi, HostSlots } from "./slots"
42
+
43
+ type PluginLoad = {
44
+ options: Config.PluginOptions | undefined
45
+ spec: string
46
+ target: string
47
+ retry: boolean
48
+ source: PluginSource | "internal"
49
+ id: string
50
+ module: TuiPluginModule
51
+ origin: Config.PluginOrigin
52
+ theme_root: string
53
+ theme_files: string[]
54
+ }
55
+
56
+ type Api = HostPluginApi
57
+
58
+ type PluginScope = {
59
+ lifecycle: TuiPluginApi["lifecycle"]
60
+ track: (fn: (() => void) | undefined) => () => void
61
+ dispose: () => Promise<void>
62
+ }
63
+
64
+ type PluginEntry = {
65
+ id: string
66
+ load: PluginLoad
67
+ meta: TuiPluginMeta
68
+ themes: Record<string, PluginMeta.Theme>
69
+ plugin: TuiPlugin
70
+ enabled: boolean
71
+ scope?: PluginScope
72
+ }
73
+
74
+ type RuntimeState = {
75
+ directory: string
76
+ api: Api
77
+ slots: HostSlots
78
+ plugins: PluginEntry[]
79
+ plugins_by_id: Map<string, PluginEntry>
80
+ pending: Map<string, Config.PluginOrigin>
81
+ }
82
+
83
+ const log = Log.create({ service: "tui.plugin" })
84
+ const DISPOSE_TIMEOUT_MS = 5000
85
+ const KV_KEY = "plugin_enabled"
86
+ const EMPTY_TUI: TuiPluginModule = {
87
+ tui: async () => { },
88
+ }
89
+
90
+ function fail(message: string, data: Record<string, unknown>) {
91
+ if (!("error" in data)) {
92
+ log.error(message, data)
93
+ console.error(`[tui.plugin] ${message}`, data)
94
+ return
95
+ }
96
+
97
+ const text = `${message}: ${errorMessage(data.error)}`
98
+ const next = { ...data, error: errorData(data.error) }
99
+ log.error(text, next)
100
+ console.error(`[tui.plugin] ${text}`, next)
101
+ }
102
+
103
+ function warn(message: string, data: Record<string, unknown>) {
104
+ log.warn(message, data)
105
+ console.warn(`[tui.plugin] ${message}`, data)
106
+ }
107
+
108
+ type CleanupResult = { type: "ok" } | { type: "error"; error: unknown } | { type: "timeout" }
109
+
110
+ function runCleanup(fn: () => unknown, ms: number): Promise<CleanupResult> {
111
+ return new Promise((resolve) => {
112
+ const timer = setTimeout(() => {
113
+ resolve({ type: "timeout" })
114
+ }, ms)
115
+
116
+ Promise.resolve()
117
+ .then(fn)
118
+ .then(
119
+ () => {
120
+ resolve({ type: "ok" })
121
+ },
122
+ (error) => {
123
+ resolve({ type: "error", error })
124
+ },
125
+ )
126
+ .finally(() => {
127
+ clearTimeout(timer)
128
+ })
129
+ })
130
+ }
131
+
132
+ function isTheme(value: unknown) {
133
+ if (!isRecord(value)) return false
134
+ if (!("theme" in value)) return false
135
+ if (!isRecord(value.theme)) return false
136
+ return true
137
+ }
138
+
139
+ function resolveRoot(root: string) {
140
+ if (root.startsWith("file://")) {
141
+ const file = fileURLToPath(root)
142
+ if (root.endsWith("/")) return file
143
+ return path.dirname(file)
144
+ }
145
+ if (path.isAbsolute(root)) return root
146
+ return path.resolve(process.cwd(), root)
147
+ }
148
+
149
+ function createThemeInstaller(
150
+ meta: Config.PluginOrigin,
151
+ root: string,
152
+ spec: string,
153
+ plugin: PluginEntry,
154
+ ): TuiTheme["install"] {
155
+ return async (file) => {
156
+ const raw = file.startsWith("file://") ? fileURLToPath(file) : file
157
+ const src = path.isAbsolute(raw) ? raw : path.resolve(root, raw)
158
+ const name = path.basename(src, path.extname(src))
159
+ const source_dir = path.dirname(meta.source)
160
+ const local_dir =
161
+ path.basename(source_dir) === ".iroder"
162
+ ? path.join(source_dir, "themes")
163
+ : path.join(source_dir, ".iroder", "themes")
164
+ const dest_dir = meta.scope === "local" ? local_dir : path.join(Global.Path.config, "themes")
165
+ const dest = path.join(dest_dir, `${name}.json`)
166
+ const stat = await Filesystem.statAsync(src)
167
+ const mtime = stat ? Math.floor(typeof stat.mtimeMs === "bigint" ? Number(stat.mtimeMs) : stat.mtimeMs) : undefined
168
+ const size = stat ? (typeof stat.size === "bigint" ? Number(stat.size) : stat.size) : undefined
169
+ const info = {
170
+ src,
171
+ dest,
172
+ mtime,
173
+ size,
174
+ }
175
+
176
+ await Flock.withLock(`tui-theme:${dest}`, async () => {
177
+ const save = async () => {
178
+ plugin.themes[name] = info
179
+ await PluginMeta.setTheme(plugin.id, name, info).catch((error) => {
180
+ log.warn("failed to track tui plugin theme", {
181
+ path: spec,
182
+ id: plugin.id,
183
+ theme: src,
184
+ dest,
185
+ error,
186
+ })
187
+ })
188
+ }
189
+
190
+ const exists = hasTheme(name)
191
+ const prev = plugin.themes[name]
192
+ if (exists) {
193
+ if (plugin.meta.state !== "updated") {
194
+ if (!prev && (await Filesystem.exists(dest))) {
195
+ await save()
196
+ }
197
+ return
198
+ }
199
+ if (prev?.dest === dest && prev.mtime === mtime && prev.size === size) return
200
+ }
201
+
202
+ const text = await Filesystem.readText(src).catch((error) => {
203
+ log.warn("failed to read tui plugin theme", { path: spec, theme: src, error })
204
+ return
205
+ })
206
+ if (text === undefined) return
207
+
208
+ const fail = Symbol()
209
+ const data = await Promise.resolve(text)
210
+ .then((x) => JSON.parse(x))
211
+ .catch((error) => {
212
+ log.warn("failed to parse tui plugin theme", { path: spec, theme: src, error })
213
+ return fail
214
+ })
215
+ if (data === fail) return
216
+
217
+ if (!isTheme(data)) {
218
+ log.warn("invalid tui plugin theme", { path: spec, theme: src })
219
+ return
220
+ }
221
+
222
+ if (exists || !(await Filesystem.exists(dest))) {
223
+ await Filesystem.write(dest, text).catch((error) => {
224
+ log.warn("failed to persist tui plugin theme", { path: spec, theme: src, dest, error })
225
+ })
226
+ }
227
+
228
+ upsertTheme(name, data)
229
+ await save()
230
+ }).catch((error) => {
231
+ log.warn("failed to lock tui plugin theme install", { path: spec, theme: src, dest, error })
232
+ })
233
+ }
234
+ }
235
+
236
+ function createMeta(
237
+ source: PluginLoad["source"],
238
+ spec: string,
239
+ target: string,
240
+ meta: { state: PluginMeta.State; entry: PluginMeta.Entry } | undefined,
241
+ id?: string,
242
+ ): TuiPluginMeta {
243
+ if (meta) {
244
+ return {
245
+ state: meta.state,
246
+ ...meta.entry,
247
+ }
248
+ }
249
+
250
+ const now = Date.now()
251
+ return {
252
+ state: source === "internal" ? "same" : "first",
253
+ id: id ?? spec,
254
+ source,
255
+ spec,
256
+ target,
257
+ first_time: now,
258
+ last_time: now,
259
+ time_changed: now,
260
+ load_count: 1,
261
+ fingerprint: target,
262
+ }
263
+ }
264
+
265
+ function loadInternalPlugin(item: InternalTuiPlugin): PluginLoad {
266
+ const spec = item.id
267
+ const target = spec
268
+
269
+ return {
270
+ options: undefined,
271
+ spec,
272
+ target,
273
+ retry: false,
274
+ source: "internal",
275
+ id: item.id,
276
+ module: item,
277
+ origin: {
278
+ spec,
279
+ scope: "global",
280
+ source: target,
281
+ },
282
+ theme_root: process.cwd(),
283
+ theme_files: [],
284
+ }
285
+ }
286
+
287
+ async function readThemeFiles(spec: string, pkg?: PluginPackage) {
288
+ if (!pkg) return [] as string[]
289
+ return Promise.resolve()
290
+ .then(() => readPackageThemes(spec, pkg))
291
+ .catch((error) => {
292
+ warn("invalid tui plugin oc-themes", {
293
+ path: spec,
294
+ pkg: pkg.pkg,
295
+ error,
296
+ })
297
+ return [] as string[]
298
+ })
299
+ }
300
+
301
+ async function syncPluginThemes(plugin: PluginEntry) {
302
+ if (!plugin.load.theme_files.length) return
303
+ if (plugin.meta.state === "same") return
304
+ const install = createThemeInstaller(plugin.load.origin, plugin.load.theme_root, plugin.load.spec, plugin)
305
+ for (const file of plugin.load.theme_files) {
306
+ await install(file).catch((error) => {
307
+ warn("failed to sync tui plugin oc-themes", { path: plugin.load.spec, id: plugin.id, theme: file, error })
308
+ })
309
+ }
310
+ }
311
+
312
+ function createPluginScope(load: PluginLoad, id: string) {
313
+ const ctrl = new AbortController()
314
+ let list: { key: symbol; fn: TuiDispose }[] = []
315
+ let done = false
316
+
317
+ const onDispose = (fn: TuiDispose) => {
318
+ if (done) return () => { }
319
+ const key = Symbol()
320
+ list.push({ key, fn })
321
+ let drop = false
322
+ return () => {
323
+ if (drop) return
324
+ drop = true
325
+ list = list.filter((x) => x.key !== key)
326
+ }
327
+ }
328
+
329
+ const track = (fn: (() => void) | undefined) => {
330
+ if (!fn) return () => { }
331
+ const off = onDispose(fn)
332
+ let drop = false
333
+ return () => {
334
+ if (drop) return
335
+ drop = true
336
+ off()
337
+ fn()
338
+ }
339
+ }
340
+
341
+ const lifecycle: TuiPluginApi["lifecycle"] = {
342
+ signal: ctrl.signal,
343
+ onDispose,
344
+ }
345
+
346
+ const dispose = async () => {
347
+ if (done) return
348
+ done = true
349
+ ctrl.abort()
350
+ const queue = [...list].reverse()
351
+ list = []
352
+ const until = Date.now() + DISPOSE_TIMEOUT_MS
353
+ for (const item of queue) {
354
+ const left = until - Date.now()
355
+ if (left <= 0) {
356
+ fail("timed out cleaning up tui plugin", {
357
+ path: load.spec,
358
+ id,
359
+ timeout: DISPOSE_TIMEOUT_MS,
360
+ })
361
+ break
362
+ }
363
+
364
+ const out = await runCleanup(item.fn, left)
365
+ if (out.type === "ok") continue
366
+ if (out.type === "timeout") {
367
+ fail("timed out cleaning up tui plugin", {
368
+ path: load.spec,
369
+ id,
370
+ timeout: DISPOSE_TIMEOUT_MS,
371
+ })
372
+ break
373
+ }
374
+
375
+ if (out.type === "error") {
376
+ fail("failed to clean up tui plugin", {
377
+ path: load.spec,
378
+ id,
379
+ error: out.error,
380
+ })
381
+ }
382
+ }
383
+ }
384
+
385
+ return {
386
+ lifecycle,
387
+ track,
388
+ dispose,
389
+ }
390
+ }
391
+
392
+ function readPluginEnabledMap(value: unknown) {
393
+ if (!isRecord(value)) return {}
394
+ return Object.fromEntries(
395
+ Object.entries(value).filter((item): item is [string, boolean] => typeof item[1] === "boolean"),
396
+ )
397
+ }
398
+
399
+ function pluginEnabledState(state: RuntimeState, config: TuiConfig.Info) {
400
+ return {
401
+ ...readPluginEnabledMap(config.plugin_enabled),
402
+ ...readPluginEnabledMap(state.api.kv.get(KV_KEY, {})),
403
+ }
404
+ }
405
+
406
+ function writePluginEnabledState(api: Api, id: string, enabled: boolean) {
407
+ api.kv.set(KV_KEY, {
408
+ ...readPluginEnabledMap(api.kv.get(KV_KEY, {})),
409
+ [id]: enabled,
410
+ })
411
+ }
412
+
413
+ function listPluginStatus(state: RuntimeState): TuiPluginStatus[] {
414
+ return state.plugins.map((plugin) => ({
415
+ id: plugin.id,
416
+ source: plugin.meta.source,
417
+ spec: plugin.meta.spec,
418
+ target: plugin.meta.target,
419
+ enabled: plugin.enabled,
420
+ active: plugin.scope !== undefined,
421
+ }))
422
+ }
423
+
424
+ async function deactivatePluginEntry(state: RuntimeState, plugin: PluginEntry, persist: boolean) {
425
+ plugin.enabled = false
426
+ if (persist) writePluginEnabledState(state.api, plugin.id, false)
427
+ if (!plugin.scope) return true
428
+ const scope = plugin.scope
429
+ plugin.scope = undefined
430
+ await scope.dispose()
431
+ return true
432
+ }
433
+
434
+ async function activatePluginEntry(state: RuntimeState, plugin: PluginEntry, persist: boolean) {
435
+ plugin.enabled = true
436
+ if (persist) writePluginEnabledState(state.api, plugin.id, true)
437
+ if (plugin.scope) return true
438
+
439
+ const scope = createPluginScope(plugin.load, plugin.id)
440
+ const api = pluginApi(state, plugin, scope, plugin.id)
441
+ const ok = await Promise.resolve()
442
+ .then(async () => {
443
+ await syncPluginThemes(plugin)
444
+ await plugin.plugin(api, plugin.load.options, plugin.meta)
445
+ return true
446
+ })
447
+ .catch((error) => {
448
+ fail("failed to initialize tui plugin", {
449
+ path: plugin.load.spec,
450
+ id: plugin.id,
451
+ error,
452
+ })
453
+ return false
454
+ })
455
+
456
+ if (!ok) {
457
+ await scope.dispose()
458
+ return false
459
+ }
460
+
461
+ if (!plugin.enabled) {
462
+ await scope.dispose()
463
+ return true
464
+ }
465
+
466
+ plugin.scope = scope
467
+ return true
468
+ }
469
+
470
+ async function activatePluginById(state: RuntimeState | undefined, id: string, persist: boolean) {
471
+ if (!state) return false
472
+ const plugin = state.plugins_by_id.get(id)
473
+ if (!plugin) return false
474
+ return activatePluginEntry(state, plugin, persist)
475
+ }
476
+
477
+ async function deactivatePluginById(state: RuntimeState | undefined, id: string, persist: boolean) {
478
+ if (!state) return false
479
+ const plugin = state.plugins_by_id.get(id)
480
+ if (!plugin) return false
481
+ return deactivatePluginEntry(state, plugin, persist)
482
+ }
483
+
484
+ function pluginApi(runtime: RuntimeState, plugin: PluginEntry, scope: PluginScope, base: string): TuiPluginApi {
485
+ const api = runtime.api
486
+ const host = runtime.slots
487
+ const load = plugin.load
488
+ const command: TuiPluginApi["command"] = {
489
+ register(cb) {
490
+ return scope.track(api.command.register(cb))
491
+ },
492
+ trigger(value) {
493
+ api.command.trigger(value)
494
+ },
495
+ show() {
496
+ api.command.show()
497
+ },
498
+ }
499
+
500
+ const route: TuiPluginApi["route"] = {
501
+ register(list) {
502
+ return scope.track(api.route.register(list))
503
+ },
504
+ navigate(name, params) {
505
+ api.route.navigate(name, params)
506
+ },
507
+ get current() {
508
+ return api.route.current
509
+ },
510
+ }
511
+
512
+ const theme: TuiPluginApi["theme"] = Object.assign(Object.create(api.theme), {
513
+ install: createThemeInstaller(load.origin, load.theme_root, load.spec, plugin),
514
+ })
515
+
516
+ const event: TuiPluginApi["event"] = {
517
+ on(type, handler) {
518
+ return scope.track(api.event.on(type, handler))
519
+ },
520
+ }
521
+
522
+ let count = 0
523
+
524
+ const slots: TuiPluginApi["slots"] = {
525
+ register(plugin: TuiSlotPlugin) {
526
+ const id = count ? `${base}:${count}` : base
527
+ count += 1
528
+ scope.track(host.register({ ...plugin, id }))
529
+ return id
530
+ },
531
+ }
532
+
533
+ return {
534
+ app: api.app,
535
+ command,
536
+ route,
537
+ ui: api.ui,
538
+ keybind: api.keybind,
539
+ tuiConfig: api.tuiConfig,
540
+ kv: api.kv,
541
+ state: api.state,
542
+ theme,
543
+ get client() {
544
+ return api.client
545
+ },
546
+ event,
547
+ renderer: api.renderer,
548
+ slots,
549
+ plugins: {
550
+ list() {
551
+ return listPluginStatus(runtime)
552
+ },
553
+ activate(id) {
554
+ return activatePluginById(runtime, id, true)
555
+ },
556
+ deactivate(id) {
557
+ return deactivatePluginById(runtime, id, true)
558
+ },
559
+ add(spec) {
560
+ return addPluginBySpec(runtime, spec)
561
+ },
562
+ install(spec, options) {
563
+ return installPluginBySpec(runtime, spec, options?.global)
564
+ },
565
+ },
566
+ lifecycle: scope.lifecycle,
567
+ }
568
+ }
569
+
570
+ function addPluginEntry(state: RuntimeState, plugin: PluginEntry) {
571
+ if (state.plugins_by_id.has(plugin.id)) {
572
+ fail("duplicate tui plugin id", {
573
+ id: plugin.id,
574
+ path: plugin.load.spec,
575
+ })
576
+ return false
577
+ }
578
+
579
+ state.plugins_by_id.set(plugin.id, plugin)
580
+ state.plugins.push(plugin)
581
+ return true
582
+ }
583
+
584
+ function applyInitialPluginEnabledState(state: RuntimeState, config: TuiConfig.Info) {
585
+ const map = pluginEnabledState(state, config)
586
+ for (const plugin of state.plugins) {
587
+ const enabled = map[plugin.id]
588
+ if (enabled === undefined) continue
589
+ plugin.enabled = enabled
590
+ }
591
+ }
592
+
593
+ async function resolveExternalPlugins(list: Config.PluginOrigin[], wait: () => Promise<void>) {
594
+ return PluginLoader.loadExternal({
595
+ items: list,
596
+ kind: "tui",
597
+ wait: async () => {
598
+ await wait().catch((error) => {
599
+ log.warn("failed waiting for tui plugin dependencies", { error })
600
+ })
601
+ },
602
+ finish: async (loaded, origin, retry) => {
603
+ const mod = await Promise.resolve()
604
+ .then(() => readV1Plugin(loaded.mod as Record<string, unknown>, loaded.spec, "tui") as TuiPluginModule)
605
+ .catch((error) => {
606
+ fail("failed to load tui plugin", {
607
+ path: loaded.spec,
608
+ target: loaded.entry,
609
+ retry,
610
+ error,
611
+ })
612
+ return
613
+ })
614
+ if (!mod) return
615
+
616
+ const id = await resolvePluginId(
617
+ loaded.source,
618
+ loaded.spec,
619
+ loaded.target,
620
+ readPluginId(mod.id, loaded.spec),
621
+ loaded.pkg,
622
+ ).catch((error) => {
623
+ fail("failed to load tui plugin", { path: loaded.spec, target: loaded.target, retry, error })
624
+ return
625
+ })
626
+ if (!id) return
627
+
628
+ const theme_files = await readThemeFiles(loaded.spec, loaded.pkg)
629
+
630
+ return {
631
+ options: loaded.options,
632
+ spec: loaded.spec,
633
+ target: loaded.target,
634
+ retry,
635
+ source: loaded.source,
636
+ id,
637
+ module: mod,
638
+ origin,
639
+ theme_root: loaded.pkg?.dir ?? resolveRoot(loaded.target),
640
+ theme_files,
641
+ }
642
+ },
643
+ missing: async (loaded, origin, retry) => {
644
+ const theme_files = await readThemeFiles(loaded.spec, loaded.pkg)
645
+ if (!theme_files.length) return
646
+
647
+ const name =
648
+ typeof loaded.pkg?.json.name === "string" && loaded.pkg.json.name.trim().length > 0
649
+ ? loaded.pkg.json.name.trim()
650
+ : undefined
651
+ const id = await resolvePluginId(loaded.source, loaded.spec, loaded.target, name, loaded.pkg).catch((error) => {
652
+ fail("failed to load tui plugin", { path: loaded.spec, target: loaded.target, retry, error })
653
+ return
654
+ })
655
+ if (!id) return
656
+
657
+ return {
658
+ options: loaded.options,
659
+ spec: loaded.spec,
660
+ target: loaded.target,
661
+ retry,
662
+ source: loaded.source,
663
+ id,
664
+ module: EMPTY_TUI,
665
+ origin,
666
+ theme_root: loaded.pkg?.dir ?? resolveRoot(loaded.target),
667
+ theme_files,
668
+ }
669
+ },
670
+ report: {
671
+ start(candidate, retry) {
672
+ log.info("loading tui plugin", { path: candidate.plan.spec, retry })
673
+ },
674
+ missing(candidate, retry, message) {
675
+ warn("tui plugin has no entrypoint", { path: candidate.plan.spec, retry, message })
676
+ },
677
+ error(candidate, retry, stage, error, resolved) {
678
+ const spec = candidate.plan.spec
679
+ if (stage === "install") {
680
+ fail("failed to resolve tui plugin", { path: spec, retry, error })
681
+ return
682
+ }
683
+ if (stage === "compatibility") {
684
+ fail("tui plugin incompatible", { path: spec, retry, error })
685
+ return
686
+ }
687
+ if (stage === "entry") {
688
+ fail("failed to resolve tui plugin entry", { path: spec, retry, error })
689
+ return
690
+ }
691
+ fail("failed to load tui plugin", { path: spec, target: resolved?.entry, retry, error })
692
+ },
693
+ },
694
+ })
695
+ }
696
+
697
+ async function addExternalPluginEntries(state: RuntimeState, ready: PluginLoad[]) {
698
+ if (!ready.length) return { plugins: [] as PluginEntry[], ok: true }
699
+
700
+ const meta = await PluginMeta.touchMany(
701
+ ready.map((item) => ({
702
+ spec: item.spec,
703
+ target: item.target,
704
+ id: item.id,
705
+ })),
706
+ ).catch((error) => {
707
+ log.warn("failed to track tui plugins", { error })
708
+ return undefined
709
+ })
710
+
711
+ const plugins: PluginEntry[] = []
712
+ let ok = true
713
+ for (let i = 0; i < ready.length; i++) {
714
+ const entry = ready[i]
715
+ if (!entry) continue
716
+ const hit = meta?.[i]
717
+ if (hit && hit.state !== "same") {
718
+ log.info("tui plugin metadata updated", {
719
+ path: entry.spec,
720
+ retry: entry.retry,
721
+ state: hit.state,
722
+ source: hit.entry.source,
723
+ version: hit.entry.version,
724
+ modified: hit.entry.modified,
725
+ })
726
+ }
727
+
728
+ const info = createMeta(entry.source, entry.spec, entry.target, hit, entry.id)
729
+ const themes = hit?.entry.themes ? { ...hit.entry.themes } : {}
730
+ const plugin: PluginEntry = {
731
+ id: entry.id,
732
+ load: entry,
733
+ meta: info,
734
+ themes,
735
+ plugin: entry.module.tui,
736
+ enabled: true,
737
+ }
738
+ if (!addPluginEntry(state, plugin)) {
739
+ ok = false
740
+ continue
741
+ }
742
+ plugins.push(plugin)
743
+ }
744
+
745
+ return { plugins, ok }
746
+ }
747
+
748
+ function defaultPluginOrigin(state: RuntimeState, spec: string): Config.PluginOrigin {
749
+ return {
750
+ spec,
751
+ scope: "local",
752
+ source: state.api.state.path.config || path.join(state.directory, ".iroder", "tui.json"),
753
+ }
754
+ }
755
+
756
+ function installCause(err: unknown) {
757
+ if (!err || typeof err !== "object") return
758
+ if (!("cause" in err)) return
759
+ return (err as { cause?: unknown }).cause
760
+ }
761
+
762
+ function installDetail(err: unknown) {
763
+ const hit = installCause(err) ?? err
764
+ if (!(hit instanceof Process.RunFailedError)) {
765
+ return {
766
+ message: errorMessage(hit),
767
+ missing: false,
768
+ }
769
+ }
770
+
771
+ const lines = hit.stderr
772
+ .toString()
773
+ .split(/\r?\n/)
774
+ .map((line) => line.trim())
775
+ .filter(Boolean)
776
+ const errs = lines.filter((line) => line.startsWith("error:")).map((line) => line.replace(/^error:\s*/, ""))
777
+ return {
778
+ message: errs[0] ?? lines.at(-1) ?? errorMessage(hit),
779
+ missing: lines.some((line) => line.includes("No version matching")),
780
+ }
781
+ }
782
+
783
+ async function addPluginBySpec(state: RuntimeState | undefined, raw: string) {
784
+ if (!state) return false
785
+ const spec = raw.trim()
786
+ if (!spec) return false
787
+
788
+ const cfg = state.pending.get(spec) ?? defaultPluginOrigin(state, spec)
789
+ const next = Config.pluginSpecifier(cfg.spec)
790
+ if (state.plugins.some((plugin) => plugin.load.spec === next)) {
791
+ state.pending.delete(spec)
792
+ return true
793
+ }
794
+
795
+ const ready = await Instance.provide({
796
+ directory: state.directory,
797
+ fn: () => resolveExternalPlugins([cfg], () => TuiConfig.waitForDependencies()),
798
+ }).catch((error) => {
799
+ fail("failed to add tui plugin", { path: next, error })
800
+ return [] as PluginLoad[]
801
+ })
802
+ if (!ready.length) {
803
+ return false
804
+ }
805
+
806
+ const first = ready[0]
807
+ if (!first) {
808
+ fail("failed to add tui plugin", { path: next })
809
+ return false
810
+ }
811
+ if (state.plugins_by_id.has(first.id)) {
812
+ state.pending.delete(spec)
813
+ return true
814
+ }
815
+
816
+ const out = await addExternalPluginEntries(state, [first])
817
+ let ok = out.ok && out.plugins.length > 0
818
+ for (const plugin of out.plugins) {
819
+ const active = await activatePluginEntry(state, plugin, false)
820
+ if (!active) ok = false
821
+ }
822
+
823
+ if (ok) state.pending.delete(spec)
824
+ if (!ok) {
825
+ fail("failed to add tui plugin", { path: next })
826
+ }
827
+ return ok
828
+ }
829
+
830
+ async function installPluginBySpec(
831
+ state: RuntimeState | undefined,
832
+ raw: string,
833
+ global = false,
834
+ ): Promise<TuiPluginInstallResult> {
835
+ if (!state) {
836
+ return {
837
+ ok: false,
838
+ message: "Plugin runtime is not ready.",
839
+ }
840
+ }
841
+
842
+ const spec = raw.trim()
843
+ if (!spec) {
844
+ return {
845
+ ok: false,
846
+ message: "Plugin package name is required",
847
+ }
848
+ }
849
+
850
+ const dir = state.api.state.path
851
+ if (!dir.directory) {
852
+ return {
853
+ ok: false,
854
+ message: "Paths are still syncing. Try again in a moment.",
855
+ }
856
+ }
857
+
858
+ const install = await installModulePlugin(spec)
859
+ if (!install.ok) {
860
+ const out = installDetail(install.error)
861
+ return {
862
+ ok: false,
863
+ message: out.message,
864
+ missing: out.missing,
865
+ }
866
+ }
867
+
868
+ const manifest = await readPluginManifest(install.target)
869
+ if (!manifest.ok) {
870
+ if (manifest.code === "manifest_no_targets") {
871
+ return {
872
+ ok: false,
873
+ message: `"${spec}" does not expose plugin entrypoints or oc-themes in package.json`,
874
+ }
875
+ }
876
+
877
+ return {
878
+ ok: false,
879
+ message: `Installed "${spec}" but failed to read ${manifest.file}`,
880
+ }
881
+ }
882
+
883
+ const patch = await patchPluginConfig({
884
+ spec,
885
+ targets: manifest.targets,
886
+ global,
887
+ vcs: dir.worktree && dir.worktree !== "/" ? "git" : undefined,
888
+ worktree: dir.worktree,
889
+ directory: dir.directory,
890
+ })
891
+ if (!patch.ok) {
892
+ if (patch.code === "invalid_json") {
893
+ return {
894
+ ok: false,
895
+ message: `Invalid JSON in ${patch.file} (${patch.parse} at line ${patch.line}, column ${patch.col})`,
896
+ }
897
+ }
898
+
899
+ return {
900
+ ok: false,
901
+ message: errorMessage(patch.error),
902
+ }
903
+ }
904
+
905
+ const tui = manifest.targets.find((item) => item.kind === "tui")
906
+ if (tui) {
907
+ const file = patch.items.find((item) => item.kind === "tui")?.file
908
+ const next = tui.opts ? ([spec, tui.opts] as Config.PluginSpec) : spec
909
+ state.pending.set(spec, {
910
+ spec: next,
911
+ scope: global ? "global" : "local",
912
+ source: (file ?? dir.config) || path.join(patch.dir, "tui.json"),
913
+ })
914
+ }
915
+
916
+ return {
917
+ ok: true,
918
+ dir: patch.dir,
919
+ tui: Boolean(tui),
920
+ }
921
+ }
922
+
923
+ export namespace TuiPluginRuntime {
924
+ let dir = ""
925
+ let loaded: Promise<void> | undefined
926
+ let runtime: RuntimeState | undefined
927
+ export const Slot = View
928
+
929
+ export async function init(api: HostPluginApi) {
930
+ const cwd = process.cwd()
931
+ if (loaded) {
932
+ if (dir !== cwd) {
933
+ throw new Error(`TuiPluginRuntime.init() called with a different working directory. expected=${dir} got=${cwd}`)
934
+ }
935
+ return loaded
936
+ }
937
+
938
+ dir = cwd
939
+ loaded = load(api)
940
+ return loaded
941
+ }
942
+
943
+ export function list() {
944
+ if (!runtime) return []
945
+ return listPluginStatus(runtime)
946
+ }
947
+
948
+ export async function activatePlugin(id: string) {
949
+ return activatePluginById(runtime, id, true)
950
+ }
951
+
952
+ export async function deactivatePlugin(id: string) {
953
+ return deactivatePluginById(runtime, id, true)
954
+ }
955
+
956
+ export async function addPlugin(spec: string) {
957
+ return addPluginBySpec(runtime, spec)
958
+ }
959
+
960
+ export async function installPlugin(spec: string, options?: { global?: boolean }) {
961
+ return installPluginBySpec(runtime, spec, options?.global)
962
+ }
963
+
964
+ export async function dispose() {
965
+ const task = loaded
966
+ loaded = undefined
967
+ dir = ""
968
+ if (task) await task
969
+ const state = runtime
970
+ runtime = undefined
971
+ if (!state) return
972
+ const queue = [...state.plugins].reverse()
973
+ for (const plugin of queue) {
974
+ await deactivatePluginEntry(state, plugin, false)
975
+ }
976
+ }
977
+
978
+ async function load(api: Api) {
979
+ const cwd = process.cwd()
980
+ const slots = setupSlots(api)
981
+ const next: RuntimeState = {
982
+ directory: cwd,
983
+ api,
984
+ slots,
985
+ plugins: [],
986
+ plugins_by_id: new Map(),
987
+ pending: new Map(),
988
+ }
989
+ runtime = next
990
+
991
+ await Instance.provide({
992
+ directory: cwd,
993
+ fn: async () => {
994
+ const config = await TuiConfig.get()
995
+ const records = Flag.IRODER_PURE ? [] : (config.plugin_origins ?? [])
996
+ if (Flag.IRODER_PURE && config.plugin_origins?.length) {
997
+ log.info("skipping external tui plugins in pure mode", { count: config.plugin_origins.length })
998
+ }
999
+
1000
+ for (const item of INTERNAL_TUI_PLUGINS) {
1001
+ log.info("loading internal tui plugin", { id: item.id })
1002
+ const entry = loadInternalPlugin(item)
1003
+ const meta = createMeta(entry.source, entry.spec, entry.target, undefined, entry.id)
1004
+ addPluginEntry(next, {
1005
+ id: entry.id,
1006
+ load: entry,
1007
+ meta,
1008
+ themes: {},
1009
+ plugin: entry.module.tui,
1010
+ enabled: true,
1011
+ })
1012
+ }
1013
+
1014
+ const ready = await resolveExternalPlugins(records, () => TuiConfig.waitForDependencies())
1015
+ await addExternalPluginEntries(next, ready)
1016
+
1017
+ applyInitialPluginEnabledState(next, config)
1018
+ for (const plugin of next.plugins) {
1019
+ if (!plugin.enabled) continue
1020
+ // Keep plugin execution sequential for deterministic side effects:
1021
+ // command registration order affects keybind/command precedence,
1022
+ // route registration is last-wins when ids collide,
1023
+ // and hook chains rely on stable plugin ordering.
1024
+ await activatePluginEntry(next, plugin, false)
1025
+ }
1026
+ },
1027
+ }).catch((error) => {
1028
+ fail("failed to load tui plugins", { directory: cwd, error })
1029
+ })
1030
+ }
1031
+ }