@miaws/miaw 1.18.4 → 1.18.5

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