@farhanic017/octocode 3.5.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 (716) hide show
  1. package/.swarm_lessons.json +85 -0
  2. package/AGENTS.md +151 -0
  3. package/BUN_SHELL_MIGRATION_PLAN.md +136 -0
  4. package/Dockerfile +18 -0
  5. package/README.md +15 -0
  6. package/bin/octocode +31 -0
  7. package/bunfig.toml +7 -0
  8. package/git +0 -0
  9. package/package.json +151 -0
  10. package/parsers-config.ts +1 -0
  11. package/script/bench-search.ts +115 -0
  12. package/script/bench-test-suite.ts +52 -0
  13. package/script/build.ts +243 -0
  14. package/script/generate.ts +14 -0
  15. package/script/httpapi-exercise.ts +1 -0
  16. package/script/octo-shim.mjs +34 -0
  17. package/script/postinstall.mjs +198 -0
  18. package/script/profile-test-files.ts +42 -0
  19. package/script/publish.ts +211 -0
  20. package/script/run-workspace-server +106 -0
  21. package/script/schema.ts +77 -0
  22. package/script/time.ts +6 -0
  23. package/script/trace-imports.ts +154 -0
  24. package/specs/effect/error-boundaries-plan.md +235 -0
  25. package/specs/effect/errors.md +207 -0
  26. package/specs/effect/facades.md +218 -0
  27. package/specs/effect/guide.md +247 -0
  28. package/specs/effect/instance-context.md +13 -0
  29. package/specs/effect/loose-ends.md +30 -0
  30. package/specs/effect/migration.md +62 -0
  31. package/specs/effect/routes.md +61 -0
  32. package/specs/effect/schema.md +88 -0
  33. package/specs/effect/server-package.md +58 -0
  34. package/specs/effect/todo.md +241 -0
  35. package/specs/effect/tools.md +88 -0
  36. package/specs/openapi-translation-cleanup.md +204 -0
  37. package/specs/tui-plugins.md +544 -0
  38. package/specs/v2/api.ts +67 -0
  39. package/specs/v2/message-shape.md +136 -0
  40. package/specs/v2/notifications.md +13 -0
  41. package/specs/v2/tui-command-shim.md +67 -0
  42. package/src/account/account.ts +459 -0
  43. package/src/account/repo.ts +170 -0
  44. package/src/account/schema.ts +99 -0
  45. package/src/account/url.ts +8 -0
  46. package/src/acp/agent.ts +95 -0
  47. package/src/acp/config-option.ts +203 -0
  48. package/src/acp/content.ts +250 -0
  49. package/src/acp/directory.ts +210 -0
  50. package/src/acp/error.ts +90 -0
  51. package/src/acp/event.ts +345 -0
  52. package/src/acp/permission.ts +145 -0
  53. package/src/acp/profile.ts +42 -0
  54. package/src/acp/service.ts +1062 -0
  55. package/src/acp/tool.ts +321 -0
  56. package/src/acp/usage.ts +239 -0
  57. package/src/agent/agent.ts +451 -0
  58. package/src/agent/generate.txt +75 -0
  59. package/src/agent/prompt/compaction.txt +30 -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 +50 -0
  63. package/src/agent/subagent-permissions.ts +35 -0
  64. package/src/audio.d.ts +14 -0
  65. package/src/auth/index.ts +96 -0
  66. package/src/background/job.ts +36 -0
  67. package/src/bus/global.ts +22 -0
  68. package/src/cli/bootstrap.ts +11 -0
  69. package/src/cli/cmd/account.ts +264 -0
  70. package/src/cli/cmd/acp.ts +76 -0
  71. package/src/cli/cmd/agent.ts +259 -0
  72. package/src/cli/cmd/attach.ts +97 -0
  73. package/src/cli/cmd/cmd.ts +7 -0
  74. package/src/cli/cmd/db.ts +62 -0
  75. package/src/cli/cmd/debug/agent.handler.ts +188 -0
  76. package/src/cli/cmd/debug/agent.ts +27 -0
  77. package/src/cli/cmd/debug/config.ts +14 -0
  78. package/src/cli/cmd/debug/file.ts +87 -0
  79. package/src/cli/cmd/debug/index.ts +87 -0
  80. package/src/cli/cmd/debug/lsp.ts +51 -0
  81. package/src/cli/cmd/debug/ripgrep.ts +99 -0
  82. package/src/cli/cmd/debug/scrap.ts +18 -0
  83. package/src/cli/cmd/debug/skill.ts +15 -0
  84. package/src/cli/cmd/debug/snapshot.ts +50 -0
  85. package/src/cli/cmd/debug/startup.ts +11 -0
  86. package/src/cli/cmd/debug/v2.ts +46 -0
  87. package/src/cli/cmd/export.ts +292 -0
  88. package/src/cli/cmd/generate.ts +54 -0
  89. package/src/cli/cmd/github.handler.ts +1594 -0
  90. package/src/cli/cmd/github.shared.ts +30 -0
  91. package/src/cli/cmd/github.ts +42 -0
  92. package/src/cli/cmd/import.ts +224 -0
  93. package/src/cli/cmd/mcp.ts +859 -0
  94. package/src/cli/cmd/models.ts +66 -0
  95. package/src/cli/cmd/plug.ts +230 -0
  96. package/src/cli/cmd/pr.ts +115 -0
  97. package/src/cli/cmd/prompt-display.ts +1 -0
  98. package/src/cli/cmd/providers.ts +550 -0
  99. package/src/cli/cmd/run/demo.ts +1274 -0
  100. package/src/cli/cmd/run/entry.body.ts +205 -0
  101. package/src/cli/cmd/run/footer.command.tsx +1088 -0
  102. package/src/cli/cmd/run/footer.menu.tsx +353 -0
  103. package/src/cli/cmd/run/footer.permission.tsx +482 -0
  104. package/src/cli/cmd/run/footer.prompt.tsx +1351 -0
  105. package/src/cli/cmd/run/footer.question.tsx +580 -0
  106. package/src/cli/cmd/run/footer.subagent.tsx +180 -0
  107. package/src/cli/cmd/run/footer.ts +1153 -0
  108. package/src/cli/cmd/run/footer.view.tsx +992 -0
  109. package/src/cli/cmd/run/footer.width.ts +27 -0
  110. package/src/cli/cmd/run/otel.ts +117 -0
  111. package/src/cli/cmd/run/permission.shared.ts +256 -0
  112. package/src/cli/cmd/run/prompt.editor.ts +157 -0
  113. package/src/cli/cmd/run/prompt.shared.ts +153 -0
  114. package/src/cli/cmd/run/question.shared.ts +340 -0
  115. package/src/cli/cmd/run/runtime.boot.ts +202 -0
  116. package/src/cli/cmd/run/runtime.lifecycle.ts +431 -0
  117. package/src/cli/cmd/run/runtime.queue.ts +349 -0
  118. package/src/cli/cmd/run/runtime.shared.ts +17 -0
  119. package/src/cli/cmd/run/runtime.stdin.ts +37 -0
  120. package/src/cli/cmd/run/runtime.ts +905 -0
  121. package/src/cli/cmd/run/scrollback.shared.ts +92 -0
  122. package/src/cli/cmd/run/scrollback.surface.ts +448 -0
  123. package/src/cli/cmd/run/scrollback.writer.tsx +353 -0
  124. package/src/cli/cmd/run/splash.ts +284 -0
  125. package/src/cli/cmd/run/stream.transport.ts +1465 -0
  126. package/src/cli/cmd/run/stream.ts +175 -0
  127. package/src/cli/cmd/run/subagent-data.ts +876 -0
  128. package/src/cli/cmd/run/theme.ts +690 -0
  129. package/src/cli/cmd/run/tool.ts +1489 -0
  130. package/src/cli/cmd/run/trace.ts +94 -0
  131. package/src/cli/cmd/run/turn-summary.ts +47 -0
  132. package/src/cli/cmd/run/types.ts +350 -0
  133. package/src/cli/cmd/run/variant.shared.ts +215 -0
  134. package/src/cli/cmd/run.ts +897 -0
  135. package/src/cli/cmd/serve.ts +24 -0
  136. package/src/cli/cmd/stats.ts +393 -0
  137. package/src/cli/cmd/tui.ts +264 -0
  138. package/src/cli/cmd/uninstall.ts +353 -0
  139. package/src/cli/cmd/upgrade.ts +74 -0
  140. package/src/cli/cmd/web.ts +84 -0
  141. package/src/cli/effect/prompt.ts +37 -0
  142. package/src/cli/effect-cmd.ts +96 -0
  143. package/src/cli/error.ts +118 -0
  144. package/src/cli/heap.ts +66 -0
  145. package/src/cli/logo.ts +1 -0
  146. package/src/cli/network.ts +64 -0
  147. package/src/cli/tui/layer.ts +7 -0
  148. package/src/cli/tui/photon_rs_bg-bq08arze.wasm +0 -0
  149. package/src/cli/tui/tree-sitter-3jzf13jk.wasm +0 -0
  150. package/src/cli/tui/tree-sitter-bash-hq5s6fxb.wasm +0 -0
  151. package/src/cli/tui/tree-sitter-powershell-ryb2ffqs.wasm +0 -0
  152. package/src/cli/tui/worker.ts +99 -0
  153. package/src/cli/ui.ts +144 -0
  154. package/src/cli/upgrade.ts +53 -0
  155. package/src/command/index.ts +189 -0
  156. package/src/command/template/initialize.txt +66 -0
  157. package/src/command/template/review.txt +101 -0
  158. package/src/command/terminal-ai-commands.ts +567 -0
  159. package/src/config/agent.ts +68 -0
  160. package/src/config/command.ts +45 -0
  161. package/src/config/config.ts +686 -0
  162. package/src/config/entry-name.ts +19 -0
  163. package/src/config/managed.ts +77 -0
  164. package/src/config/markdown.ts +36 -0
  165. package/src/config/parse.ts +79 -0
  166. package/src/config/paths.ts +47 -0
  167. package/src/config/plugin.ts +79 -0
  168. package/src/config/reference.ts +48 -0
  169. package/src/config/tui-cwd.ts +5 -0
  170. package/src/config/tui-host-attention.ts +21 -0
  171. package/src/config/tui-migrate.ts +156 -0
  172. package/src/config/tui.ts +294 -0
  173. package/src/config/variable.ts +91 -0
  174. package/src/control-plane/adapters/index.ts +41 -0
  175. package/src/control-plane/adapters/worktree.ts +96 -0
  176. package/src/control-plane/dev/README.md +19 -0
  177. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  178. package/src/control-plane/types.ts +59 -0
  179. package/src/control-plane/util.ts +39 -0
  180. package/src/control-plane/workspace-adapter-runtime.ts +51 -0
  181. package/src/control-plane/workspace-context.ts +26 -0
  182. package/src/control-plane/workspace.ts +1075 -0
  183. package/src/desktop/agent.ts +147 -0
  184. package/src/desktop/index.ts +247 -0
  185. package/src/desktop/platform.ts +97 -0
  186. package/src/desktop/stream.ts +64 -0
  187. package/src/desktop/vision.ts +52 -0
  188. package/src/desktop/websocket.ts +120 -0
  189. package/src/effect/app-runtime.ts +133 -0
  190. package/src/effect/bootstrap-runtime.ts +23 -0
  191. package/src/effect/bridge.ts +84 -0
  192. package/src/effect/config-service.ts +67 -0
  193. package/src/effect/instance-ref.ts +11 -0
  194. package/src/effect/instance-registry.ts +12 -0
  195. package/src/effect/instance-state.ts +72 -0
  196. package/src/effect/promise.ts +17 -0
  197. package/src/effect/run-service.ts +47 -0
  198. package/src/effect/runner.ts +217 -0
  199. package/src/effect/runtime-flags.ts +76 -0
  200. package/src/env/index.ts +40 -0
  201. package/src/event-v2-bridge.ts +76 -0
  202. package/src/format/formatter.ts +404 -0
  203. package/src/format/index.ts +212 -0
  204. package/src/git/index.ts +347 -0
  205. package/src/id/id.ts +80 -0
  206. package/src/ide/index.ts +70 -0
  207. package/src/image/image.ts +177 -0
  208. package/src/index.ts +208 -0
  209. package/src/installation/index.ts +350 -0
  210. package/src/lsp/client.ts +686 -0
  211. package/src/lsp/diagnostic.ts +29 -0
  212. package/src/lsp/language.ts +121 -0
  213. package/src/lsp/launch.ts +21 -0
  214. package/src/lsp/lsp.ts +517 -0
  215. package/src/lsp/server.ts +2116 -0
  216. package/src/markdown.d.ts +4 -0
  217. package/src/mcp/auth.ts +171 -0
  218. package/src/mcp/builtin.ts +92 -0
  219. package/src/mcp/index.ts +1000 -0
  220. package/src/mcp/oauth-callback.ts +232 -0
  221. package/src/mcp/oauth-provider.ts +217 -0
  222. package/src/node.ts +5 -0
  223. package/src/patch/index.ts +689 -0
  224. package/src/permission/arity.ts +163 -0
  225. package/src/permission/evaluate.ts +1 -0
  226. package/src/permission/index.ts +230 -0
  227. package/src/plugin/azure.ts +26 -0
  228. package/src/plugin/cloudflare.ts +76 -0
  229. package/src/plugin/digitalocean.ts +391 -0
  230. package/src/plugin/github-copilot/copilot.ts +417 -0
  231. package/src/plugin/github-copilot/models.ts +246 -0
  232. package/src/plugin/index.ts +320 -0
  233. package/src/plugin/install.ts +439 -0
  234. package/src/plugin/loader.ts +237 -0
  235. package/src/plugin/meta.ts +188 -0
  236. package/src/plugin/openai/README.md +31 -0
  237. package/src/plugin/openai/codex.ts +647 -0
  238. package/src/plugin/openai/ws-pool.ts +290 -0
  239. package/src/plugin/openai/ws.ts +381 -0
  240. package/src/plugin/shared.ts +323 -0
  241. package/src/plugin/tui/internal.ts +12 -0
  242. package/src/plugin/tui/runtime.ts +1174 -0
  243. package/src/plugin/xai.ts +742 -0
  244. package/src/project/bootstrap-service.ts +9 -0
  245. package/src/project/bootstrap.ts +80 -0
  246. package/src/project/instance-context.ts +24 -0
  247. package/src/project/instance-layer.ts +11 -0
  248. package/src/project/instance-runtime.ts +16 -0
  249. package/src/project/instance-store.ts +207 -0
  250. package/src/project/project.ts +520 -0
  251. package/src/project/vcs.ts +435 -0
  252. package/src/provider/auth.ts +230 -0
  253. package/src/provider/error.ts +188 -0
  254. package/src/provider/model-status.ts +8 -0
  255. package/src/provider/provider.ts +2037 -0
  256. package/src/provider/transform.ts +1367 -0
  257. package/src/pty-preparation.ts +30 -0
  258. package/src/question/index.ts +229 -0
  259. package/src/question/schema.ts +10 -0
  260. package/src/reference/reference.ts +239 -0
  261. package/src/reference/repository-cache.ts +320 -0
  262. package/src/server/auth.ts +48 -0
  263. package/src/server/cors.ts +34 -0
  264. package/src/server/event.ts +13 -0
  265. package/src/server/global-lifecycle.ts +37 -0
  266. package/src/server/init-projectors.ts +3 -0
  267. package/src/server/mdns.ts +60 -0
  268. package/src/server/projectors.ts +1 -0
  269. package/src/server/proxy-util.ts +48 -0
  270. package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
  271. package/src/server/routes/instance/httpapi/api.ts +80 -0
  272. package/src/server/routes/instance/httpapi/errors.ts +193 -0
  273. package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
  274. package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
  275. package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
  276. package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
  277. package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
  278. package/src/server/routes/instance/httpapi/groups/file.ts +172 -0
  279. package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
  280. package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
  281. package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
  282. package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
  283. package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
  284. package/src/server/routes/instance/httpapi/groups/project-copy.ts +88 -0
  285. package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
  286. package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
  287. package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
  288. package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
  289. package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
  290. package/src/server/routes/instance/httpapi/groups/reference.ts +60 -0
  291. package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
  292. package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
  293. package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
  294. package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
  295. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
  296. package/src/server/routes/instance/httpapi/handlers/control.ts +37 -0
  297. package/src/server/routes/instance/httpapi/handlers/event.ts +102 -0
  298. package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
  299. package/src/server/routes/instance/httpapi/handlers/file.ts +128 -0
  300. package/src/server/routes/instance/httpapi/handlers/global.ts +157 -0
  301. package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
  302. package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
  303. package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
  304. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +157 -0
  305. package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
  306. package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
  307. package/src/server/routes/instance/httpapi/handlers/pty.ts +258 -0
  308. package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
  309. package/src/server/routes/instance/httpapi/handlers/reference.ts +27 -0
  310. package/src/server/routes/instance/httpapi/handlers/sync.ts +95 -0
  311. package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
  312. package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
  313. package/src/server/routes/instance/httpapi/lifecycle.ts +57 -0
  314. package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
  315. package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
  316. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
  317. package/src/server/routes/instance/httpapi/middleware/error.ts +36 -0
  318. package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
  319. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
  320. package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
  321. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +42 -0
  322. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
  323. package/src/server/routes/instance/httpapi/public.ts +535 -0
  324. package/src/server/routes/instance/httpapi/server.ts +281 -0
  325. package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
  326. package/src/server/server.ts +218 -0
  327. package/src/server/shared/fence.ts +68 -0
  328. package/src/server/shared/pty-ticket.ts +15 -0
  329. package/src/server/shared/public-ui.ts +12 -0
  330. package/src/server/shared/tui-control.ts +28 -0
  331. package/src/server/shared/ui.ts +108 -0
  332. package/src/server/shared/workspace-routing.ts +38 -0
  333. package/src/server/tui-event.ts +53 -0
  334. package/src/share/share-next.ts +379 -0
  335. package/src/shell/shell.ts +215 -0
  336. package/src/skill/discovery.ts +115 -0
  337. package/src/skill/index.ts +382 -0
  338. package/src/snapshot/index.ts +762 -0
  339. package/src/sql.d.ts +4 -0
  340. package/src/storage/schema.ts +5 -0
  341. package/src/storage/storage.ts +329 -0
  342. package/src/swarm/evolution.ts +304 -0
  343. package/src/swarm/index.ts +7 -0
  344. package/src/swarm/learning.ts +736 -0
  345. package/src/swarm/vision.ts +131 -0
  346. package/src/sync/README.md +179 -0
  347. package/src/sync/schema.ts +11 -0
  348. package/src/temporary.ts +33 -0
  349. package/src/tool/apply_patch.ts +313 -0
  350. package/src/tool/apply_patch.txt +33 -0
  351. package/src/tool/browser_click.ts +51 -0
  352. package/src/tool/browser_drag.ts +76 -0
  353. package/src/tool/browser_evaluate.ts +51 -0
  354. package/src/tool/browser_hover.ts +51 -0
  355. package/src/tool/browser_navigate.ts +45 -0
  356. package/src/tool/browser_screenshot.ts +66 -0
  357. package/src/tool/browser_select.ts +52 -0
  358. package/src/tool/browser_type.ts +52 -0
  359. package/src/tool/browser_wait.ts +60 -0
  360. package/src/tool/desktop_clipboard.ts +82 -0
  361. package/src/tool/desktop_control.ts +160 -0
  362. package/src/tool/desktop_record.ts +93 -0
  363. package/src/tool/desktop_replay.ts +99 -0
  364. package/src/tool/desktop_screen_record.ts +143 -0
  365. package/src/tool/desktop_window.ts +142 -0
  366. package/src/tool/desktop_workflow.ts +81 -0
  367. package/src/tool/edit.ts +737 -0
  368. package/src/tool/edit.txt +10 -0
  369. package/src/tool/external-directory.ts +49 -0
  370. package/src/tool/glob.ts +84 -0
  371. package/src/tool/glob.txt +6 -0
  372. package/src/tool/grep.ts +140 -0
  373. package/src/tool/grep.txt +8 -0
  374. package/src/tool/invalid.ts +21 -0
  375. package/src/tool/json-schema.ts +164 -0
  376. package/src/tool/lsp.ts +113 -0
  377. package/src/tool/lsp.txt +24 -0
  378. package/src/tool/mcp-websearch.ts +96 -0
  379. package/src/tool/open_app.ts +55 -0
  380. package/src/tool/open_terminal.ts +69 -0
  381. package/src/tool/plan-enter.txt +14 -0
  382. package/src/tool/plan-exit.txt +13 -0
  383. package/src/tool/plan.ts +79 -0
  384. package/src/tool/question.ts +44 -0
  385. package/src/tool/question.txt +10 -0
  386. package/src/tool/read.ts +392 -0
  387. package/src/tool/read.txt +14 -0
  388. package/src/tool/registry.ts +527 -0
  389. package/src/tool/schema.ts +14 -0
  390. package/src/tool/screenshot.ts +72 -0
  391. package/src/tool/shell/id.ts +19 -0
  392. package/src/tool/shell/prompt.ts +307 -0
  393. package/src/tool/shell/shell.txt +21 -0
  394. package/src/tool/shell.ts +669 -0
  395. package/src/tool/skill.ts +72 -0
  396. package/src/tool/skill.txt +5 -0
  397. package/src/tool/task.ts +339 -0
  398. package/src/tool/task.txt +19 -0
  399. package/src/tool/todo.ts +57 -0
  400. package/src/tool/todowrite.txt +44 -0
  401. package/src/tool/tool.ts +180 -0
  402. package/src/tool/truncate.ts +160 -0
  403. package/src/tool/truncation-dir.ts +4 -0
  404. package/src/tool/visual_qa.ts +77 -0
  405. package/src/tool/webfetch.ts +192 -0
  406. package/src/tool/webfetch.txt +13 -0
  407. package/src/tool/websearch.ts +143 -0
  408. package/src/tool/websearch.txt +14 -0
  409. package/src/tool/write.ts +104 -0
  410. package/src/tool/write.txt +8 -0
  411. package/src/util/archive.ts +17 -0
  412. package/src/util/bom.ts +27 -0
  413. package/src/util/data-url.ts +9 -0
  414. package/src/util/defer.ts +10 -0
  415. package/src/util/effect-http-client.ts +11 -0
  416. package/src/util/error.ts +1 -0
  417. package/src/util/filesystem.ts +251 -0
  418. package/src/util/iife.ts +3 -0
  419. package/src/util/lazy.ts +20 -0
  420. package/src/util/local-context.ts +25 -0
  421. package/src/util/locale.ts +2 -0
  422. package/src/util/media.ts +26 -0
  423. package/src/util/process.ts +177 -0
  424. package/src/util/proxy-env.ts +72 -0
  425. package/src/util/queue.ts +32 -0
  426. package/src/util/record.ts +1 -0
  427. package/src/util/repository.ts +232 -0
  428. package/src/util/rpc.ts +66 -0
  429. package/src/util/signal.ts +12 -0
  430. package/src/util/sys-monitor.ts +223 -0
  431. package/src/util/timeout.ts +13 -0
  432. package/src/util/wildcard.ts +59 -0
  433. package/src/worktree/index.ts +645 -0
  434. package/sst-env.d.ts +10 -0
  435. package/test/AGENTS.md +204 -0
  436. package/test/EFFECT_TEST_MIGRATION.md +169 -0
  437. package/test/account/repo.test.ts +353 -0
  438. package/test/account/service.test.ts +453 -0
  439. package/test/acp/config-option.test.ts +229 -0
  440. package/test/acp/content.test.ts +201 -0
  441. package/test/acp/directory.test.ts +186 -0
  442. package/test/acp/error.test.ts +67 -0
  443. package/test/acp/event.test.ts +743 -0
  444. package/test/acp/permission.test.ts +273 -0
  445. package/test/acp/tool.test.ts +210 -0
  446. package/test/acp/usage.test.ts +315 -0
  447. package/test/agent/agent.test.ts +711 -0
  448. package/test/agent/plan-mode-subagent-bypass.test.ts +213 -0
  449. package/test/agent/plugin-agent-regression.test.ts +62 -0
  450. package/test/auth/auth.test.ts +77 -0
  451. package/test/background/job.test.ts +243 -0
  452. package/test/cli/account.test.ts +30 -0
  453. package/test/cli/acp/acp-test-client.ts +97 -0
  454. package/test/cli/acp/config-options.test.ts +103 -0
  455. package/test/cli/acp/helpers.ts +96 -0
  456. package/test/cli/acp/initialize-auth.test.ts +61 -0
  457. package/test/cli/acp/lifecycle.test.ts +118 -0
  458. package/test/cli/acp/prompt-content.test.ts +97 -0
  459. package/test/cli/acp/skills.test.ts +38 -0
  460. package/test/cli/cmd/tui/attention.test.ts +484 -0
  461. package/test/cli/effect-cmd-instance-als.test.ts +39 -0
  462. package/test/cli/error.test.ts +95 -0
  463. package/test/cli/github-action.test.ts +199 -0
  464. package/test/cli/github-remote.test.ts +90 -0
  465. package/test/cli/help/__snapshots__/help-snapshots.test.ts.snap +631 -0
  466. package/test/cli/help/help-snapshots.test.ts +137 -0
  467. package/test/cli/import.test.ts +54 -0
  468. package/test/cli/mcp-add.test.ts +74 -0
  469. package/test/cli/plugin-auth-picker.test.ts +120 -0
  470. package/test/cli/run/entry.body.test.ts +536 -0
  471. package/test/cli/run/footer.menu.test.ts +43 -0
  472. package/test/cli/run/footer.view.test.tsx +1336 -0
  473. package/test/cli/run/footer.width.test.ts +35 -0
  474. package/test/cli/run/permission.shared.test.ts +144 -0
  475. package/test/cli/run/prompt.editor.test.ts +101 -0
  476. package/test/cli/run/prompt.shared.test.ts +101 -0
  477. package/test/cli/run/question.shared.test.ts +115 -0
  478. package/test/cli/run/run-process.test.ts +84 -0
  479. package/test/cli/run/runtime.boot.test.ts +283 -0
  480. package/test/cli/run/runtime.queue.test.ts +481 -0
  481. package/test/cli/run/runtime.stdin.test.ts +71 -0
  482. package/test/cli/run/runtime.test.ts +238 -0
  483. package/test/cli/run/scrollback.surface.test.ts +1065 -0
  484. package/test/cli/run/stream.test.ts +56 -0
  485. package/test/cli/run/stream.transport.test.ts +2363 -0
  486. package/test/cli/run/subagent-data.test.ts +547 -0
  487. package/test/cli/run/theme.test.ts +177 -0
  488. package/test/cli/run/variant.shared.test.ts +217 -0
  489. package/test/cli/serve/serve-process.test.ts +61 -0
  490. package/test/cli/smokes/read-only.test.ts +115 -0
  491. package/test/cli/tui/attach.test.ts +11 -0
  492. package/test/cli/tui/editor-context-zed.test.ts +379 -0
  493. package/test/cli/tui/editor-context.test.tsx +297 -0
  494. package/test/cli/tui/plugin-add.test.ts +110 -0
  495. package/test/cli/tui/plugin-install.test.ts +87 -0
  496. package/test/cli/tui/plugin-lifecycle.test.ts +224 -0
  497. package/test/cli/tui/plugin-loader-entrypoint.test.ts +485 -0
  498. package/test/cli/tui/plugin-loader-pure.test.ts +72 -0
  499. package/test/cli/tui/plugin-loader.test.ts +1332 -0
  500. package/test/cli/tui/plugin-toggle.test.ts +264 -0
  501. package/test/cli/tui/thread.test.ts +36 -0
  502. package/test/cli.test.ts +7 -0
  503. package/test/command/acp-slash-detect.test.ts +202 -0
  504. package/test/command/demo-slash.test.ts +213 -0
  505. package/test/command/slash-head.test.ts +232 -0
  506. package/test/command/slash-parsing.test.ts +193 -0
  507. package/test/command/terminal-ai-commands.test.ts +536 -0
  508. package/test/config/agent-color.test.ts +47 -0
  509. package/test/config/config.test.ts +1994 -0
  510. package/test/config/entry-name.test.ts +57 -0
  511. package/test/config/fixtures/empty-frontmatter.md +4 -0
  512. package/test/config/fixtures/frontmatter.md +28 -0
  513. package/test/config/fixtures/markdown-header.md +11 -0
  514. package/test/config/fixtures/no-frontmatter.md +1 -0
  515. package/test/config/fixtures/weird-model-id.md +13 -0
  516. package/test/config/lsp.test.ts +69 -0
  517. package/test/config/markdown.test.ts +228 -0
  518. package/test/config/plugin.test.ts +0 -0
  519. package/test/config/tui.test.ts +886 -0
  520. package/test/control-plane/adapters.test.ts +71 -0
  521. package/test/control-plane/workspace.test.ts +1704 -0
  522. package/test/effect/app-runtime-logger.test.ts +105 -0
  523. package/test/effect/config-service.test.ts +65 -0
  524. package/test/effect/instance-state.test.ts +391 -0
  525. package/test/effect/run-service.test.ts +89 -0
  526. package/test/effect/runner.test.ts +514 -0
  527. package/test/effect/runtime-flags.test.ts +373 -0
  528. package/test/fake/account.ts +9 -0
  529. package/test/fake/auth.ts +8 -0
  530. package/test/fake/npm.ts +8 -0
  531. package/test/fake/provider.ts +82 -0
  532. package/test/fake/skill.ts +8 -0
  533. package/test/filesystem/filesystem.test.ts +319 -0
  534. package/test/fixture/agent-plugin.constants.ts +6 -0
  535. package/test/fixture/agent-plugin.ts +12 -0
  536. package/test/fixture/config.ts +23 -0
  537. package/test/fixture/db.ts +11 -0
  538. package/test/fixture/fixture.test.ts +26 -0
  539. package/test/fixture/fixture.ts +224 -0
  540. package/test/fixture/flag.ts +20 -0
  541. package/test/fixture/flock-worker.ts +72 -0
  542. package/test/fixture/lsp/fake-lsp-server.js +249 -0
  543. package/test/fixture/plug-worker.ts +96 -0
  544. package/test/fixture/plugin-meta-worker.ts +19 -0
  545. package/test/fixture/plugin.ts +10 -0
  546. package/test/fixture/skills/agents-sdk/SKILL.md +152 -0
  547. package/test/fixture/skills/agents-sdk/references/callable.md +92 -0
  548. package/test/fixture/skills/cloudflare/SKILL.md +211 -0
  549. package/test/fixture/skills/index.json +6 -0
  550. package/test/fixture/tui-environment.tsx +32 -0
  551. package/test/fixture/tui-plugin.ts +357 -0
  552. package/test/fixture/tui-runtime.ts +56 -0
  553. package/test/fixture/tui-sdk.ts +82 -0
  554. package/test/fixture/workspace.ts +30 -0
  555. package/test/format/format.test.ts +228 -0
  556. package/test/git/git.test.ts +178 -0
  557. package/test/ide/ide.test.ts +82 -0
  558. package/test/image/fixtures/picture-5mb-base64.png +0 -0
  559. package/test/image/image.test.ts +123 -0
  560. package/test/installation/installation.test.ts +231 -0
  561. package/test/lib/cli-process.ts +459 -0
  562. package/test/lib/effect.ts +177 -0
  563. package/test/lib/filesystem.ts +10 -0
  564. package/test/lib/llm-server.ts +771 -0
  565. package/test/lib/snapshot.ts +73 -0
  566. package/test/lib/test-provider.ts +37 -0
  567. package/test/lib/websocket.ts +46 -0
  568. package/test/lsp/client.test.ts +493 -0
  569. package/test/lsp/index.test.ts +232 -0
  570. package/test/lsp/jdtls-root.test.ts +459 -0
  571. package/test/lsp/launch.test.ts +22 -0
  572. package/test/lsp/lifecycle.test.ts +160 -0
  573. package/test/mcp/auth.test.ts +78 -0
  574. package/test/mcp/builtin.test.ts +95 -0
  575. package/test/mcp/headers.test.ts +126 -0
  576. package/test/mcp/lifecycle.test.ts +999 -0
  577. package/test/mcp/oauth-auto-connect.test.ts +274 -0
  578. package/test/mcp/oauth-browser.test.ts +232 -0
  579. package/test/mcp/oauth-callback.test.ts +34 -0
  580. package/test/mcp/oauth-provider.test.ts +61 -0
  581. package/test/patch/patch.test.ts +383 -0
  582. package/test/permission/arity.test.ts +33 -0
  583. package/test/permission/next.test.ts +1176 -0
  584. package/test/permission-task.test.ts +318 -0
  585. package/test/plugin/auth-override.test.ts +105 -0
  586. package/test/plugin/cloudflare.test.ts +68 -0
  587. package/test/plugin/codex.test.ts +247 -0
  588. package/test/plugin/github-copilot-models.test.ts +332 -0
  589. package/test/plugin/install-concurrency.test.ts +140 -0
  590. package/test/plugin/install.test.ts +573 -0
  591. package/test/plugin/loader-shared.test.ts +1303 -0
  592. package/test/plugin/meta.test.ts +137 -0
  593. package/test/plugin/openai-rollout.test.ts +17 -0
  594. package/test/plugin/openai-ws.test.ts +877 -0
  595. package/test/plugin/shared.test.ts +88 -0
  596. package/test/plugin/trigger.test.ts +120 -0
  597. package/test/plugin/workspace-adapter.test.ts +137 -0
  598. package/test/plugin/xai.test.ts +634 -0
  599. package/test/preload.ts +99 -0
  600. package/test/project/instance-bootstrap.test.ts +110 -0
  601. package/test/project/instance.test.ts +245 -0
  602. package/test/project/migrate-global.test.ts +170 -0
  603. package/test/project/project-directory.test.ts +169 -0
  604. package/test/project/project.test.ts +818 -0
  605. package/test/project/vcs.test.ts +336 -0
  606. package/test/project/worktree-remove.test.ts +126 -0
  607. package/test/project/worktree.test.ts +320 -0
  608. package/test/provider/amazon-bedrock.test.ts +360 -0
  609. package/test/provider/cf-ai-gateway-e2e.test.ts +132 -0
  610. package/test/provider/digitalocean.test.ts +123 -0
  611. package/test/provider/gitlab-duo.test.ts +412 -0
  612. package/test/provider/header-timeout.test.ts +233 -0
  613. package/test/provider/model-status.test.ts +61 -0
  614. package/test/provider/provider.test.ts +1795 -0
  615. package/test/provider/transform.test.ts +3937 -0
  616. package/test/pty/pty-shell.test.ts +102 -0
  617. package/test/question/question.test.ts +465 -0
  618. package/test/reference/reference.test.ts +311 -0
  619. package/test/server/AGENTS.md +15 -0
  620. package/test/server/auth.test.ts +59 -0
  621. package/test/server/global-bus.ts +31 -0
  622. package/test/server/httpapi-authorization.test.ts +174 -0
  623. package/test/server/httpapi-compression.test.ts +154 -0
  624. package/test/server/httpapi-config.test.ts +113 -0
  625. package/test/server/httpapi-control-plane.test.ts +63 -0
  626. package/test/server/httpapi-cors-vary.test.ts +66 -0
  627. package/test/server/httpapi-cors.test.ts +122 -0
  628. package/test/server/httpapi-error-middleware.test.ts +96 -0
  629. package/test/server/httpapi-event.test.ts +97 -0
  630. package/test/server/httpapi-exercise/assertions.ts +64 -0
  631. package/test/server/httpapi-exercise/backend.ts +144 -0
  632. package/test/server/httpapi-exercise/dsl.ts +210 -0
  633. package/test/server/httpapi-exercise/environment.ts +40 -0
  634. package/test/server/httpapi-exercise/index.ts +1538 -0
  635. package/test/server/httpapi-exercise/report.ts +66 -0
  636. package/test/server/httpapi-exercise/routing.ts +96 -0
  637. package/test/server/httpapi-exercise/runner.ts +267 -0
  638. package/test/server/httpapi-exercise/runtime.ts +52 -0
  639. package/test/server/httpapi-exercise/types.ts +123 -0
  640. package/test/server/httpapi-experimental.test.ts +300 -0
  641. package/test/server/httpapi-file.test.ts +76 -0
  642. package/test/server/httpapi-global.test.ts +66 -0
  643. package/test/server/httpapi-instance-context.test.ts +347 -0
  644. package/test/server/httpapi-instance-route-auth.test.ts +84 -0
  645. package/test/server/httpapi-instance.test.ts +265 -0
  646. package/test/server/httpapi-layer.ts +33 -0
  647. package/test/server/httpapi-listen.test.ts +415 -0
  648. package/test/server/httpapi-mcp-oauth.test.ts +73 -0
  649. package/test/server/httpapi-mcp.test.ts +234 -0
  650. package/test/server/httpapi-mdns.test.ts +82 -0
  651. package/test/server/httpapi-promptasync-context.test.ts +222 -0
  652. package/test/server/httpapi-provider.test.ts +403 -0
  653. package/test/server/httpapi-pty.test.ts +275 -0
  654. package/test/server/httpapi-public-openapi.test.ts +297 -0
  655. package/test/server/httpapi-query-schema-drift.test.ts +330 -0
  656. package/test/server/httpapi-reference.test.ts +56 -0
  657. package/test/server/httpapi-schema-error-body.test.ts +165 -0
  658. package/test/server/httpapi-sdk.test.ts +909 -0
  659. package/test/server/httpapi-sync.test.ts +154 -0
  660. package/test/server/httpapi-ui.test.ts +456 -0
  661. package/test/server/httpapi-v2-location.test.ts +85 -0
  662. package/test/server/httpapi-workspace-routing.test.ts +554 -0
  663. package/test/server/httpapi-workspace.test.ts +515 -0
  664. package/test/server/project-copy.test.ts +101 -0
  665. package/test/server/project-init-git.test.ts +117 -0
  666. package/test/server/proxy-util.test.ts +113 -0
  667. package/test/server/sdk-error-shape.test.ts +84 -0
  668. package/test/server/sdk-v1-smoke.test.ts +60 -0
  669. package/test/server/workspace-proxy.test.ts +181 -0
  670. package/test/server/workspace-routing.test.ts +94 -0
  671. package/test/server/worktree-endpoint-repro.test.ts +307 -0
  672. package/test/share/share-next.test.ts +344 -0
  673. package/test/shell/shell.test.ts +99 -0
  674. package/test/skill/discovery.test.ts +139 -0
  675. package/test/skill/skill.test.ts +571 -0
  676. package/test/snapshot/snapshot.test.ts +1121 -0
  677. package/test/storage/storage.test.ts +296 -0
  678. package/test/storage/workspace-time-migration.test.ts +50 -0
  679. package/test/tool/__snapshots__/parameters.test.ts.snap +484 -0
  680. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  681. package/test/tool/apply_patch.test.ts +533 -0
  682. package/test/tool/browser.integration.test.ts +141 -0
  683. package/test/tool/desktop.integration.test.ts +129 -0
  684. package/test/tool/desktop.test.ts +85 -0
  685. package/test/tool/edit.test.ts +578 -0
  686. package/test/tool/external-directory.test.ts +167 -0
  687. package/test/tool/fixtures/large-image.png +0 -0
  688. package/test/tool/fixtures/models-api.json +117299 -0
  689. package/test/tool/glob.test.ts +188 -0
  690. package/test/tool/grep.test.ts +266 -0
  691. package/test/tool/lsp.test.ts +181 -0
  692. package/test/tool/parameters.test.ts +293 -0
  693. package/test/tool/question.test.ts +138 -0
  694. package/test/tool/read.test.ts +659 -0
  695. package/test/tool/registry.test.ts +539 -0
  696. package/test/tool/shell.test.ts +1256 -0
  697. package/test/tool/skill.test.ts +135 -0
  698. package/test/tool/task.test.ts +901 -0
  699. package/test/tool/tool-define.test.ts +153 -0
  700. package/test/tool/truncation.test.ts +266 -0
  701. package/test/tool/webfetch.test.ts +113 -0
  702. package/test/tool/websearch.test.ts +99 -0
  703. package/test/tool/write.test.ts +276 -0
  704. package/test/util/data-url.test.ts +14 -0
  705. package/test/util/error.test.ts +16 -0
  706. package/test/util/filesystem.test.ts +656 -0
  707. package/test/util/glob.test.ts +164 -0
  708. package/test/util/iife.test.ts +36 -0
  709. package/test/util/lazy.test.ts +50 -0
  710. package/test/util/log.test.ts +77 -0
  711. package/test/util/module.test.ts +59 -0
  712. package/test/util/process.test.ts +128 -0
  713. package/test/util/repository.test.ts +93 -0
  714. package/test/util/timeout.test.ts +21 -0
  715. package/test/util/wildcard.test.ts +90 -0
  716. package/tsconfig.json +16 -0
@@ -0,0 +1,1174 @@
1
+ import { runtimeModules as keymapRuntimeModules } from "@opentui/keymap/runtime-modules"
2
+ import { ensureRuntimePluginSupport } from "@opentui/solid/runtime-plugin-support/configure"
3
+ import {
4
+ type TuiDispose,
5
+ type TuiPlugin,
6
+ type TuiPluginApi,
7
+ type TuiPluginInstallResult,
8
+ type TuiPluginModule,
9
+ type TuiPluginMeta,
10
+ type TuiPluginStatus,
11
+ type TuiSlotPlugin,
12
+ type TuiTheme,
13
+ } from "@octocode-ai/plugin/tui"
14
+ import path from "path"
15
+ import { fileURLToPath } from "url"
16
+ import { TuiConfig } from "@/config/tui"
17
+ import * as Log from "@octocode-ai/core/util/log"
18
+ import { errorData, errorMessage } from "@octocode-ai/tui/util/error"
19
+ import { isRecord } from "@octocode-ai/tui/util/record"
20
+ import { resolveHostAttentionSoundPaths } from "@/config/tui-host-attention"
21
+ import {
22
+ readPackageThemes,
23
+ readPluginId,
24
+ readV1Plugin,
25
+ resolvePluginId,
26
+ type PluginPackage,
27
+ type PluginSource,
28
+ } from "@/plugin/shared"
29
+ import { PluginLoader } from "@/plugin/loader"
30
+ import { PluginMeta } from "@/plugin/meta"
31
+ import { installPlugin as installModulePlugin, patchPluginConfig, readPluginManifest } from "@/plugin/install"
32
+ import { hasTheme, upsertTheme } from "@octocode-ai/tui/context/theme"
33
+ import { Global } from "@octocode-ai/core/global"
34
+ import { Filesystem } from "@/util/filesystem"
35
+ import { Process } from "@/util/process"
36
+ import { Flock } from "@octocode-ai/core/util/flock"
37
+ import { Flag } from "@octocode-ai/core/flag/flag"
38
+ import { internalTuiPlugins, type InternalTuiPlugin } from "./internal"
39
+ import type { HostPluginApi, HostSlots } from "@octocode-ai/tui/plugin/slots"
40
+ import { ConfigPlugin } from "@/config/plugin"
41
+ import { ConfigPluginV1 } from "@octocode-ai/core/v1/config/plugin"
42
+ import { createCommandShim } from "@octocode-ai/tui/plugin/command-shim"
43
+ import { RuntimeFlags } from "@/effect/runtime-flags"
44
+ import { Effect } from "effect"
45
+ import { createPluginRuntime, type PluginRuntime, type TuiPluginHost } from "@octocode-ai/tui/plugin/runtime"
46
+
47
+ ensureRuntimePluginSupport({ additional: keymapRuntimeModules })
48
+
49
+ type PluginLoad = {
50
+ options: ConfigPluginV1.Options | undefined
51
+ spec: string
52
+ target: string
53
+ retry: boolean
54
+ source: PluginSource | "internal"
55
+ id: string
56
+ module: TuiPluginModule
57
+ origin: ConfigPlugin.Origin
58
+ plugin_root: string
59
+ theme_files: string[]
60
+ }
61
+
62
+ type Api = HostPluginApi
63
+
64
+ type PluginScope = {
65
+ lifecycle: TuiPluginApi["lifecycle"]
66
+ track: (fn: (() => void) | undefined) => () => void
67
+ dispose: () => Promise<void>
68
+ }
69
+
70
+ type PluginEntry = {
71
+ id: string
72
+ load: PluginLoad
73
+ meta: TuiPluginMeta
74
+ themes: Record<string, PluginMeta.Theme>
75
+ plugin: TuiPlugin
76
+ enabled: boolean
77
+ scope?: PluginScope
78
+ }
79
+
80
+ const ScopedKeymapMethods = new Set<PropertyKey>([
81
+ "acquireResource",
82
+ "registerLayer",
83
+ "registerLayerFields",
84
+ "prependLayerBindingsTransformer",
85
+ "appendLayerBindingsTransformer",
86
+ "prependBindingTransformer",
87
+ "appendBindingTransformer",
88
+ "prependBindingParser",
89
+ "appendBindingParser",
90
+ "registerToken",
91
+ "registerSequencePattern",
92
+ "prependBindingExpander",
93
+ "appendBindingExpander",
94
+ "registerBindingFields",
95
+ "registerCommandFields",
96
+ "prependCommandTransformer",
97
+ "appendCommandTransformer",
98
+ "prependCommandResolver",
99
+ "appendCommandResolver",
100
+ "prependLayerAnalyzer",
101
+ "appendLayerAnalyzer",
102
+ "intercept",
103
+ "on",
104
+ "prependEventMatchResolver",
105
+ "appendEventMatchResolver",
106
+ "prependDisambiguationResolver",
107
+ "appendDisambiguationResolver",
108
+ ])
109
+
110
+ type RuntimeState = {
111
+ directory: string
112
+ api: Api
113
+ view: PluginRuntime
114
+ dispose?: () => void
115
+ slots: HostSlots
116
+ plugins: PluginEntry[]
117
+ plugins_by_id: Map<string, PluginEntry>
118
+ pending: Map<string, ConfigPlugin.Origin>
119
+ dispose_timeout_ms: number
120
+ }
121
+
122
+ const log = Log.create({ service: "tui.plugin" })
123
+ const DISPOSE_TIMEOUT_MS = 5000
124
+ const KV_KEY = "plugin_enabled"
125
+ const EMPTY_TUI: TuiPluginModule = {
126
+ tui: async () => {},
127
+ }
128
+
129
+ function fail(message: string, data: Record<string, unknown>) {
130
+ if (!("error" in data)) {
131
+ log.error(message, data)
132
+ console.error(`[tui.plugin] ${message}`, data)
133
+ return
134
+ }
135
+
136
+ const text = `${message}: ${errorMessage(data.error)}`
137
+ const next = { ...data, error: errorData(data.error) }
138
+ log.error(text, next)
139
+ console.error(`[tui.plugin] ${text}`, next)
140
+ }
141
+
142
+ function warn(message: string, data: Record<string, unknown>) {
143
+ log.warn(message, data)
144
+ console.warn(`[tui.plugin] ${message}`, data)
145
+ }
146
+
147
+ function createScopedKeymap(keymap: TuiPluginApi["keymap"], scope: PluginScope): TuiPluginApi["keymap"] {
148
+ const cache = new Map<PropertyKey, unknown>()
149
+ return new Proxy(keymap, {
150
+ get(target, prop) {
151
+ const value = Reflect.get(target, prop, target)
152
+ if (typeof value !== "function") return value
153
+ if (cache.has(prop)) return cache.get(prop)
154
+ const fn = ScopedKeymapMethods.has(prop)
155
+ ? (...args: unknown[]) => {
156
+ const dispose = (value as (...args: unknown[]) => unknown).apply(target, args)
157
+ return scope.track(typeof dispose === "function" ? (dispose as () => void) : undefined)
158
+ }
159
+ : (...args: unknown[]) => (value as (...args: unknown[]) => unknown).apply(target, args)
160
+ cache.set(prop, fn)
161
+ return fn
162
+ },
163
+ })
164
+ }
165
+
166
+ function createScopedAttention(
167
+ attention: TuiPluginApi["attention"],
168
+ scope: PluginScope,
169
+ root: string,
170
+ ): TuiPluginApi["attention"] {
171
+ return {
172
+ notify(input) {
173
+ return attention.notify(input)
174
+ },
175
+ soundboard: {
176
+ registerPack(pack) {
177
+ return scope.track(
178
+ attention.soundboard.registerPack({
179
+ ...pack,
180
+ sounds: resolveHostAttentionSoundPaths(root, pack.sounds, { trim: true }),
181
+ }),
182
+ )
183
+ },
184
+ activate(id, options) {
185
+ return attention.soundboard.activate(id, options)
186
+ },
187
+ current() {
188
+ return attention.soundboard.current()
189
+ },
190
+ list() {
191
+ return attention.soundboard.list()
192
+ },
193
+ },
194
+ }
195
+ }
196
+
197
+ function createScopedMode(mode: TuiPluginApi["mode"], scope: PluginScope): TuiPluginApi["mode"] {
198
+ return {
199
+ current() {
200
+ return mode.current()
201
+ },
202
+ push(value) {
203
+ return scope.track(mode.push(value))
204
+ },
205
+ }
206
+ }
207
+
208
+ type CleanupResult = { type: "ok" } | { type: "error"; error: unknown } | { type: "timeout" }
209
+
210
+ function runCleanup(fn: () => unknown, ms: number): Promise<CleanupResult> {
211
+ return new Promise((resolve) => {
212
+ const timer = setTimeout(() => {
213
+ resolve({ type: "timeout" })
214
+ }, ms)
215
+
216
+ Promise.resolve()
217
+ .then(fn)
218
+ .then(
219
+ () => {
220
+ resolve({ type: "ok" })
221
+ },
222
+ (error) => {
223
+ resolve({ type: "error", error })
224
+ },
225
+ )
226
+ .finally(() => {
227
+ clearTimeout(timer)
228
+ })
229
+ })
230
+ }
231
+
232
+ function isTheme(value: unknown) {
233
+ if (!isRecord(value)) return false
234
+ if (!("theme" in value)) return false
235
+ if (!isRecord(value.theme)) return false
236
+ return true
237
+ }
238
+
239
+ function resolveRoot(root: string) {
240
+ if (root.startsWith("file://")) {
241
+ const file = fileURLToPath(root)
242
+ if (root.endsWith("/")) return file
243
+ return path.dirname(file)
244
+ }
245
+ if (path.isAbsolute(root)) return root
246
+ return path.resolve(process.cwd(), root)
247
+ }
248
+
249
+ function createThemeInstaller(
250
+ meta: ConfigPlugin.Origin,
251
+ root: string,
252
+ spec: string,
253
+ plugin: PluginEntry,
254
+ ): TuiTheme["install"] {
255
+ return async (file) => {
256
+ const src = Filesystem.resolveFilePath(root, file)
257
+ const name = path.basename(src, path.extname(src))
258
+ const source_dir = path.dirname(meta.source)
259
+ const local_dir =
260
+ path.basename(source_dir) === ".octo" || path.basename(source_dir) === ".octocode"
261
+ ? path.join(source_dir, "themes")
262
+ : path.join(source_dir, ".octo", "themes")
263
+ const dest_dir = meta.scope === "local" ? local_dir : path.join(Global.Path.config, "themes")
264
+ const dest = path.join(dest_dir, `${name}.json`)
265
+ const stat = await Filesystem.statAsync(src)
266
+ const mtime = stat ? Math.floor(typeof stat.mtimeMs === "bigint" ? Number(stat.mtimeMs) : stat.mtimeMs) : undefined
267
+ const size = stat ? (typeof stat.size === "bigint" ? Number(stat.size) : stat.size) : undefined
268
+ const info = {
269
+ src,
270
+ dest,
271
+ mtime,
272
+ size,
273
+ }
274
+
275
+ await Flock.withLock(`tui-theme:${dest}`, async () => {
276
+ const save = async () => {
277
+ plugin.themes[name] = info
278
+ await PluginMeta.setTheme(plugin.id, name, info).catch((error) => {
279
+ log.warn("failed to track tui plugin theme", {
280
+ path: spec,
281
+ id: plugin.id,
282
+ theme: src,
283
+ dest,
284
+ error,
285
+ })
286
+ })
287
+ }
288
+
289
+ const exists = hasTheme(name)
290
+ const prev = plugin.themes[name]
291
+ if (exists) {
292
+ if (plugin.meta.state !== "updated") {
293
+ if (!prev && (await Filesystem.exists(dest))) {
294
+ await save()
295
+ }
296
+ return
297
+ }
298
+ if (prev?.dest === dest && prev.mtime === mtime && prev.size === size) return
299
+ }
300
+
301
+ const text = await Filesystem.readText(src).catch((error) => {
302
+ log.warn("failed to read tui plugin theme", { path: spec, theme: src, error })
303
+ return
304
+ })
305
+ if (text === undefined) return
306
+
307
+ const fail = Symbol()
308
+ const data = await Promise.resolve(text)
309
+ .then((x) => JSON.parse(x))
310
+ .catch((error) => {
311
+ log.warn("failed to parse tui plugin theme", { path: spec, theme: src, error })
312
+ return fail
313
+ })
314
+ if (data === fail) return
315
+
316
+ if (!isTheme(data)) {
317
+ log.warn("invalid tui plugin theme", { path: spec, theme: src })
318
+ return
319
+ }
320
+
321
+ if (exists || !(await Filesystem.exists(dest))) {
322
+ await Filesystem.write(dest, text).catch((error) => {
323
+ log.warn("failed to persist tui plugin theme", { path: spec, theme: src, dest, error })
324
+ })
325
+ }
326
+
327
+ upsertTheme(name, data)
328
+ await save()
329
+ }).catch((error) => {
330
+ log.warn("failed to lock tui plugin theme install", { path: spec, theme: src, dest, error })
331
+ })
332
+ }
333
+ }
334
+
335
+ function createMeta(
336
+ source: PluginLoad["source"],
337
+ spec: string,
338
+ target: string,
339
+ meta: { state: PluginMeta.State; entry: PluginMeta.Entry } | undefined,
340
+ id?: string,
341
+ ): TuiPluginMeta {
342
+ if (meta) {
343
+ return {
344
+ state: meta.state,
345
+ ...meta.entry,
346
+ }
347
+ }
348
+
349
+ const now = Date.now()
350
+ return {
351
+ state: source === "internal" ? "same" : "first",
352
+ id: id ?? spec,
353
+ source,
354
+ spec,
355
+ target,
356
+ first_time: now,
357
+ last_time: now,
358
+ time_changed: now,
359
+ load_count: 1,
360
+ fingerprint: target,
361
+ }
362
+ }
363
+
364
+ function loadInternalPlugin(item: InternalTuiPlugin): PluginLoad {
365
+ const spec = item.id
366
+ const target = spec
367
+
368
+ return {
369
+ options: undefined,
370
+ spec,
371
+ target,
372
+ retry: false,
373
+ source: "internal",
374
+ id: item.id,
375
+ module: item,
376
+ origin: {
377
+ spec,
378
+ scope: "global",
379
+ source: target,
380
+ },
381
+ plugin_root: process.cwd(),
382
+ theme_files: [],
383
+ }
384
+ }
385
+
386
+ async function readThemeFiles(spec: string, pkg?: PluginPackage) {
387
+ if (!pkg) return [] as string[]
388
+ return Promise.resolve()
389
+ .then(() => readPackageThemes(spec, pkg))
390
+ .catch((error) => {
391
+ warn("invalid tui plugin oc-themes", {
392
+ path: spec,
393
+ pkg: pkg.pkg,
394
+ error,
395
+ })
396
+ return [] as string[]
397
+ })
398
+ }
399
+
400
+ async function syncPluginThemes(plugin: PluginEntry) {
401
+ if (!plugin.load.theme_files.length) return
402
+ if (plugin.meta.state === "same") return
403
+ const install = createThemeInstaller(plugin.load.origin, plugin.load.plugin_root, plugin.load.spec, plugin)
404
+ for (const file of plugin.load.theme_files) {
405
+ await install(file).catch((error) => {
406
+ warn("failed to sync tui plugin oc-themes", { path: plugin.load.spec, id: plugin.id, theme: file, error })
407
+ })
408
+ }
409
+ }
410
+
411
+ function createPluginScope(load: PluginLoad, id: string, disposeTimeoutMs: number) {
412
+ const ctrl = new AbortController()
413
+ let list: { key: symbol; fn: TuiDispose }[] = []
414
+ let done = false
415
+
416
+ const onDispose = (fn: TuiDispose) => {
417
+ if (done) return () => {}
418
+ const key = Symbol()
419
+ list.push({ key, fn })
420
+ let drop = false
421
+ return () => {
422
+ if (drop) return
423
+ drop = true
424
+ list = list.filter((x) => x.key !== key)
425
+ }
426
+ }
427
+
428
+ const track = (fn: (() => void) | undefined) => {
429
+ if (!fn) return () => {}
430
+ let drop = false
431
+ let off = () => {}
432
+ const wrapped = () => {
433
+ if (drop) return
434
+ drop = true
435
+ off()
436
+ fn()
437
+ }
438
+ off = onDispose(wrapped)
439
+ return wrapped
440
+ }
441
+
442
+ const lifecycle: TuiPluginApi["lifecycle"] = {
443
+ signal: ctrl.signal,
444
+ onDispose,
445
+ }
446
+
447
+ const dispose = async () => {
448
+ if (done) return
449
+ done = true
450
+ ctrl.abort()
451
+ const queue = [...list].reverse()
452
+ list = []
453
+ const until = Date.now() + disposeTimeoutMs
454
+ for (const item of queue) {
455
+ const left = until - Date.now()
456
+ if (left <= 0) {
457
+ fail("timed out cleaning up tui plugin", {
458
+ path: load.spec,
459
+ id,
460
+ timeout: disposeTimeoutMs,
461
+ })
462
+ break
463
+ }
464
+
465
+ const out = await runCleanup(item.fn, left)
466
+ if (out.type === "ok") continue
467
+ if (out.type === "timeout") {
468
+ fail("timed out cleaning up tui plugin", {
469
+ path: load.spec,
470
+ id,
471
+ timeout: disposeTimeoutMs,
472
+ })
473
+ break
474
+ }
475
+
476
+ if (out.type === "error") {
477
+ fail("failed to clean up tui plugin", {
478
+ path: load.spec,
479
+ id,
480
+ error: out.error,
481
+ })
482
+ }
483
+ }
484
+ }
485
+
486
+ return {
487
+ lifecycle,
488
+ track,
489
+ dispose,
490
+ }
491
+ }
492
+
493
+ function readPluginEnabledMap(value: unknown) {
494
+ if (!isRecord(value)) return {}
495
+ return Object.fromEntries(
496
+ Object.entries(value).filter((item): item is [string, boolean] => typeof item[1] === "boolean"),
497
+ )
498
+ }
499
+
500
+ function pluginEnabledState(state: RuntimeState, config: TuiConfig.Resolved) {
501
+ return {
502
+ ...readPluginEnabledMap(config.plugin_enabled),
503
+ ...readPluginEnabledMap(state.api.kv.get(KV_KEY, {})),
504
+ }
505
+ }
506
+
507
+ function writePluginEnabledState(api: Api, id: string, enabled: boolean) {
508
+ api.kv.set(KV_KEY, {
509
+ ...readPluginEnabledMap(api.kv.get(KV_KEY, {})),
510
+ [id]: enabled,
511
+ })
512
+ }
513
+
514
+ function listPluginStatus(state: RuntimeState): TuiPluginStatus[] {
515
+ return state.plugins.map((plugin) => ({
516
+ id: plugin.id,
517
+ source: plugin.meta.source,
518
+ spec: plugin.meta.spec,
519
+ target: plugin.meta.target,
520
+ enabled: plugin.enabled,
521
+ active: plugin.scope !== undefined,
522
+ }))
523
+ }
524
+
525
+ async function deactivatePluginEntry(state: RuntimeState, plugin: PluginEntry, persist: boolean) {
526
+ plugin.enabled = false
527
+ if (persist) writePluginEnabledState(state.api, plugin.id, false)
528
+ if (!plugin.scope) {
529
+ state.view.update({ status: listPluginStatus(state) })
530
+ return true
531
+ }
532
+ const scope = plugin.scope
533
+ plugin.scope = undefined
534
+ await scope.dispose()
535
+ state.view.update({ status: listPluginStatus(state) })
536
+ return true
537
+ }
538
+
539
+ async function activatePluginEntry(state: RuntimeState, plugin: PluginEntry, persist: boolean) {
540
+ plugin.enabled = true
541
+ if (persist) writePluginEnabledState(state.api, plugin.id, true)
542
+ if (plugin.scope) {
543
+ state.view.update({ status: listPluginStatus(state) })
544
+ return true
545
+ }
546
+
547
+ const scope = createPluginScope(plugin.load, plugin.id, state.dispose_timeout_ms)
548
+ const api = pluginApi(state, plugin, scope, plugin.id)
549
+ const ok = await Promise.resolve()
550
+ .then(async () => {
551
+ await syncPluginThemes(plugin)
552
+ await plugin.plugin(api, plugin.load.options, plugin.meta)
553
+ return true
554
+ })
555
+ .catch((error) => {
556
+ fail("failed to initialize tui plugin", {
557
+ path: plugin.load.spec,
558
+ id: plugin.id,
559
+ error,
560
+ })
561
+ return false
562
+ })
563
+
564
+ if (!ok) {
565
+ await scope.dispose()
566
+ state.view.update({ status: listPluginStatus(state) })
567
+ return false
568
+ }
569
+
570
+ if (!plugin.enabled) {
571
+ await scope.dispose()
572
+ state.view.update({ status: listPluginStatus(state) })
573
+ return true
574
+ }
575
+
576
+ plugin.scope = scope
577
+ state.view.update({ status: listPluginStatus(state) })
578
+ return true
579
+ }
580
+
581
+ async function activatePluginById(state: RuntimeState | undefined, id: string, persist: boolean) {
582
+ if (!state) return false
583
+ const plugin = state.plugins_by_id.get(id)
584
+ if (!plugin) return false
585
+ return activatePluginEntry(state, plugin, persist)
586
+ }
587
+
588
+ async function deactivatePluginById(state: RuntimeState | undefined, id: string, persist: boolean) {
589
+ if (!state) return false
590
+ const plugin = state.plugins_by_id.get(id)
591
+ if (!plugin) return false
592
+ return deactivatePluginEntry(state, plugin, persist)
593
+ }
594
+
595
+ function pluginApi(runtime: RuntimeState, plugin: PluginEntry, scope: PluginScope, base: string): TuiPluginApi {
596
+ const api = runtime.api
597
+ const host = runtime.slots
598
+ const load = plugin.load
599
+
600
+ const route: TuiPluginApi["route"] = {
601
+ register(list) {
602
+ return scope.track(api.route.register(list))
603
+ },
604
+ navigate(name, params) {
605
+ api.route.navigate(name, params)
606
+ },
607
+ get current() {
608
+ return api.route.current
609
+ },
610
+ }
611
+
612
+ const theme: TuiPluginApi["theme"] = Object.assign(Object.create(api.theme), {
613
+ install: createThemeInstaller(load.origin, load.plugin_root, load.spec, plugin),
614
+ })
615
+
616
+ const event: TuiPluginApi["event"] = {
617
+ on(type, handler) {
618
+ return scope.track(api.event.on(type, handler))
619
+ },
620
+ }
621
+
622
+ const keymap = createScopedKeymap(api.keymap, scope)
623
+
624
+ let count = 0
625
+
626
+ const slots: TuiPluginApi["slots"] = {
627
+ register(plugin: TuiSlotPlugin) {
628
+ const id = count ? `${base}:${count}` : base
629
+ count += 1
630
+ scope.track(host.register({ ...plugin, id }))
631
+ return id
632
+ },
633
+ }
634
+
635
+ return {
636
+ app: api.app,
637
+ attention: createScopedAttention(api.attention, scope, load.plugin_root),
638
+ // Keep deprecated `api.command` working for v1 plugins; remove in v2.
639
+ command: createCommandShim(keymap, api.ui.dialog, api.tuiConfig.keybinds),
640
+ keys: api.keys,
641
+ keymap,
642
+ mode: createScopedMode(api.mode, scope),
643
+ route,
644
+ ui: api.ui,
645
+ tuiConfig: api.tuiConfig,
646
+ kv: api.kv,
647
+ state: api.state,
648
+ theme,
649
+ get client() {
650
+ return api.client
651
+ },
652
+ event,
653
+ renderer: api.renderer,
654
+ slots,
655
+ plugins: {
656
+ list() {
657
+ return listPluginStatus(runtime)
658
+ },
659
+ activate(id) {
660
+ return activatePluginById(runtime, id, true)
661
+ },
662
+ deactivate(id) {
663
+ return deactivatePluginById(runtime, id, true)
664
+ },
665
+ add(spec) {
666
+ return addPluginBySpec(runtime, spec)
667
+ },
668
+ install(spec, options) {
669
+ return installPluginBySpec(runtime, spec, options?.global)
670
+ },
671
+ },
672
+ lifecycle: scope.lifecycle,
673
+ }
674
+ }
675
+
676
+ function addPluginEntry(state: RuntimeState, plugin: PluginEntry) {
677
+ if (state.plugins_by_id.has(plugin.id)) {
678
+ fail("duplicate tui plugin id", {
679
+ id: plugin.id,
680
+ path: plugin.load.spec,
681
+ })
682
+ return false
683
+ }
684
+
685
+ state.plugins_by_id.set(plugin.id, plugin)
686
+ state.plugins.push(plugin)
687
+ return true
688
+ }
689
+
690
+ function applyInitialPluginEnabledState(state: RuntimeState, config: TuiConfig.Resolved) {
691
+ const map = pluginEnabledState(state, config)
692
+ for (const plugin of state.plugins) {
693
+ const enabled = map[plugin.id]
694
+ if (enabled === undefined) continue
695
+ plugin.enabled = enabled
696
+ }
697
+ }
698
+
699
+ async function resolveExternalPlugins(list: ConfigPlugin.Origin[], wait: () => Promise<void>) {
700
+ return PluginLoader.loadExternal({
701
+ items: list,
702
+ kind: "tui",
703
+ wait: async () => {
704
+ await wait().catch((error) => {
705
+ log.warn("failed waiting for tui plugin dependencies", { error })
706
+ })
707
+ },
708
+ finish: async (loaded, origin, retry) => {
709
+ const mod = await Promise.resolve()
710
+ .then(() => readV1Plugin(loaded.mod as Record<string, unknown>, loaded.spec, "tui") as TuiPluginModule)
711
+ .catch((error) => {
712
+ fail("failed to load tui plugin", {
713
+ path: loaded.spec,
714
+ target: loaded.entry,
715
+ retry,
716
+ error,
717
+ })
718
+ return
719
+ })
720
+ if (!mod) return
721
+
722
+ const id = await resolvePluginId(
723
+ loaded.source,
724
+ loaded.spec,
725
+ loaded.target,
726
+ readPluginId(mod.id, loaded.spec),
727
+ loaded.pkg,
728
+ ).catch((error) => {
729
+ fail("failed to load tui plugin", { path: loaded.spec, target: loaded.target, retry, error })
730
+ return
731
+ })
732
+ if (!id) return
733
+
734
+ const theme_files = await readThemeFiles(loaded.spec, loaded.pkg)
735
+
736
+ return {
737
+ options: loaded.options,
738
+ spec: loaded.spec,
739
+ target: loaded.target,
740
+ retry,
741
+ source: loaded.source,
742
+ id,
743
+ module: mod,
744
+ origin,
745
+ plugin_root: loaded.pkg?.dir ?? resolveRoot(loaded.target),
746
+ theme_files,
747
+ }
748
+ },
749
+ missing: async (loaded, origin, retry) => {
750
+ const theme_files = await readThemeFiles(loaded.spec, loaded.pkg)
751
+ if (!theme_files.length) return
752
+
753
+ const name =
754
+ typeof loaded.pkg?.json.name === "string" && loaded.pkg.json.name.trim().length > 0
755
+ ? loaded.pkg.json.name.trim()
756
+ : undefined
757
+ const id = await resolvePluginId(loaded.source, loaded.spec, loaded.target, name, loaded.pkg).catch((error) => {
758
+ fail("failed to load tui plugin", { path: loaded.spec, target: loaded.target, retry, error })
759
+ return
760
+ })
761
+ if (!id) return
762
+
763
+ return {
764
+ options: loaded.options,
765
+ spec: loaded.spec,
766
+ target: loaded.target,
767
+ retry,
768
+ source: loaded.source,
769
+ id,
770
+ module: EMPTY_TUI,
771
+ origin,
772
+ plugin_root: loaded.pkg?.dir ?? resolveRoot(loaded.target),
773
+ theme_files,
774
+ }
775
+ },
776
+ report: {
777
+ start(candidate, retry) {
778
+ log.info("loading tui plugin", { path: candidate.plan.spec, retry })
779
+ },
780
+ missing(candidate, retry, message) {
781
+ warn("tui plugin has no entrypoint", { path: candidate.plan.spec, retry, message })
782
+ },
783
+ error(candidate, retry, stage, error, resolved) {
784
+ const spec = candidate.plan.spec
785
+ if (stage === "install") {
786
+ fail("failed to resolve tui plugin", { path: spec, retry, error })
787
+ return
788
+ }
789
+ if (stage === "compatibility") {
790
+ fail("tui plugin incompatible", { path: spec, retry, error })
791
+ return
792
+ }
793
+ if (stage === "entry") {
794
+ fail("failed to resolve tui plugin entry", { path: spec, retry, error })
795
+ return
796
+ }
797
+ fail("failed to load tui plugin", { path: spec, target: resolved?.entry, retry, error })
798
+ },
799
+ },
800
+ })
801
+ }
802
+
803
+ async function addExternalPluginEntries(state: RuntimeState, ready: PluginLoad[]) {
804
+ if (!ready.length) return { plugins: [] as PluginEntry[], ok: true }
805
+
806
+ const meta = await PluginMeta.touchMany(
807
+ ready.map((item) => ({
808
+ spec: item.spec,
809
+ target: item.target,
810
+ id: item.id,
811
+ })),
812
+ ).catch((error) => {
813
+ log.warn("failed to track tui plugins", { error })
814
+ return undefined
815
+ })
816
+
817
+ const plugins: PluginEntry[] = []
818
+ let ok = true
819
+ for (let i = 0; i < ready.length; i++) {
820
+ const entry = ready[i]
821
+ if (!entry) continue
822
+ const hit = meta?.[i]
823
+ if (hit && hit.state !== "same") {
824
+ log.info("tui plugin metadata updated", {
825
+ path: entry.spec,
826
+ retry: entry.retry,
827
+ state: hit.state,
828
+ source: hit.entry.source,
829
+ version: hit.entry.version,
830
+ modified: hit.entry.modified,
831
+ })
832
+ }
833
+
834
+ const info = createMeta(entry.source, entry.spec, entry.target, hit, entry.id)
835
+ const themes = hit?.entry.themes ? { ...hit.entry.themes } : {}
836
+ const plugin: PluginEntry = {
837
+ id: entry.id,
838
+ load: entry,
839
+ meta: info,
840
+ themes,
841
+ plugin: entry.module.tui,
842
+ enabled: true,
843
+ }
844
+ if (!addPluginEntry(state, plugin)) {
845
+ ok = false
846
+ continue
847
+ }
848
+ plugins.push(plugin)
849
+ }
850
+
851
+ return { plugins, ok }
852
+ }
853
+
854
+ function defaultPluginOrigin(state: RuntimeState, spec: string): ConfigPlugin.Origin {
855
+ return {
856
+ spec,
857
+ scope: "local",
858
+ source: state.api.state.path.config || path.join(state.directory, ".octo", "tui.json"),
859
+ }
860
+ }
861
+
862
+ function installCause(err: unknown) {
863
+ if (!err || typeof err !== "object") return
864
+ if (!("cause" in err)) return
865
+ return (err as { cause?: unknown }).cause
866
+ }
867
+
868
+ function installDetail(err: unknown) {
869
+ const hit = installCause(err) ?? err
870
+ if (!(hit instanceof Process.RunFailedError)) {
871
+ return {
872
+ message: errorMessage(hit),
873
+ missing: false,
874
+ }
875
+ }
876
+
877
+ const lines = hit.stderr
878
+ .toString()
879
+ .split(/\r?\n/)
880
+ .map((line) => line.trim())
881
+ .filter(Boolean)
882
+ const errs = lines.filter((line) => line.startsWith("error:")).map((line) => line.replace(/^error:\s*/, ""))
883
+ return {
884
+ message: errs[0] ?? lines.at(-1) ?? errorMessage(hit),
885
+ missing: lines.some((line) => line.includes("No version matching")),
886
+ }
887
+ }
888
+
889
+ async function addPluginBySpec(state: RuntimeState | undefined, raw: string) {
890
+ if (!state) return false
891
+ const spec = raw.trim()
892
+ if (!spec) return false
893
+
894
+ const cfg = state.pending.get(spec) ?? defaultPluginOrigin(state, spec)
895
+ const next = ConfigPlugin.pluginSpecifier(cfg.spec)
896
+ if (state.plugins.some((plugin) => plugin.load.spec === next)) {
897
+ state.pending.delete(spec)
898
+ return true
899
+ }
900
+ const ready = await resolveExternalPlugins([cfg], () => TuiConfig.waitForDependencies()).catch((error) => {
901
+ fail("failed to add tui plugin", { path: next, error })
902
+ return [] as PluginLoad[]
903
+ })
904
+ if (!ready.length) {
905
+ return false
906
+ }
907
+
908
+ const first = ready[0]
909
+ if (!first) {
910
+ fail("failed to add tui plugin", { path: next })
911
+ return false
912
+ }
913
+ if (state.plugins_by_id.has(first.id)) {
914
+ state.pending.delete(spec)
915
+ return true
916
+ }
917
+
918
+ const out = await addExternalPluginEntries(state, [first])
919
+ let ok = out.ok && out.plugins.length > 0
920
+ for (const plugin of out.plugins) {
921
+ const active = await activatePluginEntry(state, plugin, false)
922
+ if (!active) ok = false
923
+ }
924
+
925
+ if (ok) state.pending.delete(spec)
926
+ if (!ok) {
927
+ fail("failed to add tui plugin", { path: next })
928
+ }
929
+ return ok
930
+ }
931
+
932
+ async function installPluginBySpec(
933
+ state: RuntimeState | undefined,
934
+ raw: string,
935
+ global = false,
936
+ ): Promise<TuiPluginInstallResult> {
937
+ if (!state) {
938
+ return {
939
+ ok: false,
940
+ message: "Plugin runtime is not ready.",
941
+ }
942
+ }
943
+
944
+ const spec = raw.trim()
945
+ if (!spec) {
946
+ return {
947
+ ok: false,
948
+ message: "Plugin package name is required",
949
+ }
950
+ }
951
+
952
+ const dir = state.api.state.path
953
+ if (!dir.directory) {
954
+ return {
955
+ ok: false,
956
+ message: "Paths are still syncing. Try again in a moment.",
957
+ }
958
+ }
959
+
960
+ const install = await installModulePlugin(spec)
961
+ if (!install.ok) {
962
+ const out = installDetail(install.error)
963
+ return {
964
+ ok: false,
965
+ message: out.message,
966
+ missing: out.missing,
967
+ }
968
+ }
969
+
970
+ const manifest = await readPluginManifest(install.target)
971
+ if (!manifest.ok) {
972
+ if (manifest.code === "manifest_no_targets") {
973
+ return {
974
+ ok: false,
975
+ message: `"${spec}" does not expose plugin entrypoints or oc-themes in package.json`,
976
+ }
977
+ }
978
+
979
+ return {
980
+ ok: false,
981
+ message: `Installed "${spec}" but failed to read ${manifest.file}`,
982
+ }
983
+ }
984
+
985
+ const patch = await patchPluginConfig({
986
+ spec,
987
+ targets: manifest.targets,
988
+ global,
989
+ vcs: dir.worktree && dir.worktree !== "/" ? "git" : undefined,
990
+ worktree: dir.worktree,
991
+ directory: dir.directory,
992
+ })
993
+ if (!patch.ok) {
994
+ if (patch.code === "invalid_json") {
995
+ return {
996
+ ok: false,
997
+ message: `Invalid JSON in ${patch.file} (${patch.parse} at line ${patch.line}, column ${patch.col})`,
998
+ }
999
+ }
1000
+
1001
+ return {
1002
+ ok: false,
1003
+ message: errorMessage(patch.error),
1004
+ }
1005
+ }
1006
+
1007
+ const tui = manifest.targets.find((item) => item.kind === "tui")
1008
+ if (tui) {
1009
+ const file = patch.items.find((item) => item.kind === "tui")?.file
1010
+ const next = tui.opts ? ([spec, tui.opts] as ConfigPluginV1.Spec) : spec
1011
+ state.pending.set(spec, {
1012
+ spec: next,
1013
+ scope: global ? "global" : "local",
1014
+ source: (file ?? dir.config) || path.join(patch.dir, "tui.json"),
1015
+ })
1016
+ }
1017
+
1018
+ return {
1019
+ ok: true,
1020
+ dir: patch.dir,
1021
+ tui: Boolean(tui),
1022
+ }
1023
+ }
1024
+
1025
+ let dir = ""
1026
+ let loaded: Promise<void> | undefined
1027
+ let runtime: RuntimeState | undefined
1028
+
1029
+ export async function init(input: {
1030
+ api: HostPluginApi
1031
+ config: TuiConfig.Resolved & TuiConfig.HostMetadata
1032
+ runtime?: PluginRuntime
1033
+ dispose?: () => void
1034
+ disposeTimeoutMs?: number
1035
+ }) {
1036
+ const cwd = process.cwd()
1037
+ if (loaded) {
1038
+ if (dir !== cwd) {
1039
+ throw new Error(`TuiPluginRuntime.init() called with a different working directory. expected=${dir} got=${cwd}`)
1040
+ }
1041
+ return loaded
1042
+ }
1043
+
1044
+ dir = cwd
1045
+ loaded = load({ ...input, runtime: input.runtime ?? createPluginRuntime() })
1046
+ return loaded
1047
+ }
1048
+
1049
+ export function list() {
1050
+ if (!runtime) return []
1051
+ return listPluginStatus(runtime)
1052
+ }
1053
+
1054
+ export async function activatePlugin(id: string) {
1055
+ return activatePluginById(runtime, id, true)
1056
+ }
1057
+
1058
+ export async function deactivatePlugin(id: string) {
1059
+ return deactivatePluginById(runtime, id, true)
1060
+ }
1061
+
1062
+ export async function addPlugin(spec: string) {
1063
+ return addPluginBySpec(runtime, spec)
1064
+ }
1065
+
1066
+ export async function installPlugin(spec: string, options?: { global?: boolean }) {
1067
+ return installPluginBySpec(runtime, spec, options?.global)
1068
+ }
1069
+
1070
+ export async function dispose() {
1071
+ const task = loaded
1072
+ loaded = undefined
1073
+ dir = ""
1074
+ if (task) await task.catch((error) => fail("failed to finish loading tui plugins during disposal", { error }))
1075
+ const state = runtime
1076
+ runtime = undefined
1077
+ if (!state) return
1078
+ const queue = [...state.plugins].reverse()
1079
+ for (const plugin of queue) {
1080
+ await deactivatePluginEntry(state, plugin, false).catch((error) =>
1081
+ fail("failed to dispose tui plugin", { id: plugin.id, error }),
1082
+ )
1083
+ }
1084
+ try {
1085
+ state.dispose?.()
1086
+ } finally {
1087
+ state.slots.dispose()
1088
+ state.view.clear()
1089
+ }
1090
+ }
1091
+
1092
+ async function load(input: {
1093
+ api: Api
1094
+ config: TuiConfig.Resolved & TuiConfig.HostMetadata
1095
+ runtime: PluginRuntime
1096
+ dispose?: () => void
1097
+ disposeTimeoutMs?: number
1098
+ }) {
1099
+ const { api, config } = input
1100
+ const cwd = process.cwd()
1101
+ const slots = input.runtime.setupSlots(api)
1102
+ const next: RuntimeState = {
1103
+ directory: cwd,
1104
+ api,
1105
+ view: input.runtime,
1106
+ dispose: input.dispose,
1107
+ slots,
1108
+ plugins: [],
1109
+ plugins_by_id: new Map(),
1110
+ pending: new Map(),
1111
+ dispose_timeout_ms: input.disposeTimeoutMs ?? DISPOSE_TIMEOUT_MS,
1112
+ }
1113
+ runtime = next
1114
+ next.view.update({
1115
+ commands: {
1116
+ activate: activatePlugin,
1117
+ deactivate: deactivatePlugin,
1118
+ add: addPlugin,
1119
+ install: installPlugin,
1120
+ },
1121
+ status: listPluginStatus(next),
1122
+ })
1123
+ try {
1124
+ const flags = await Effect.runPromise(
1125
+ Effect.gen(function* () {
1126
+ return yield* RuntimeFlags.Service
1127
+ }).pipe(Effect.provide(RuntimeFlags.defaultLayer)),
1128
+ )
1129
+ const pluginOrigins = config.plugin_origins ?? (await TuiConfig.pluginOrigins())
1130
+ const records = Flag.OCTOCODE_PURE ? [] : pluginOrigins
1131
+ if (Flag.OCTOCODE_PURE && pluginOrigins.length) {
1132
+ log.info("skipping external tui plugins in pure mode", { count: pluginOrigins.length })
1133
+ }
1134
+
1135
+ for (const item of internalTuiPlugins(flags)) {
1136
+ log.info("loading internal tui plugin", { id: item.id })
1137
+ const entry = loadInternalPlugin(item)
1138
+ const meta = createMeta(entry.source, entry.spec, entry.target, undefined, entry.id)
1139
+ addPluginEntry(next, {
1140
+ id: entry.id,
1141
+ load: entry,
1142
+ meta,
1143
+ themes: {},
1144
+ plugin: entry.module.tui,
1145
+ enabled: item.enabled ?? true,
1146
+ })
1147
+ }
1148
+
1149
+ const ready = await resolveExternalPlugins(records, () => TuiConfig.waitForDependencies())
1150
+ await addExternalPluginEntries(next, ready)
1151
+
1152
+ applyInitialPluginEnabledState(next, config)
1153
+ for (const plugin of next.plugins) {
1154
+ if (!plugin.enabled) continue
1155
+ // Keep plugin execution sequential for deterministic side effects:
1156
+ // command registration order affects keybind/command precedence,
1157
+ // route registration is last-wins when ids collide,
1158
+ // and hook chains rely on stable plugin ordering.
1159
+ await activatePluginEntry(next, plugin, false)
1160
+ }
1161
+ next.view.update({ status: listPluginStatus(next) })
1162
+ } catch (error) {
1163
+ fail("failed to load tui plugins", { directory: cwd, error })
1164
+ }
1165
+ }
1166
+
1167
+ export function createLegacyTuiPluginHost(): TuiPluginHost {
1168
+ return {
1169
+ start: init,
1170
+ dispose,
1171
+ }
1172
+ }
1173
+
1174
+ export * as TuiPluginRuntime from "./runtime"