@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,230 @@
1
+ import { describe, expect } from "bun:test"
2
+ import { Effect, Layer, Stream } from "effect"
3
+ import { HttpClient, HttpClientRequest, HttpClientResponse } from "effect/unstable/http"
4
+ import { ChildProcess, ChildProcessSpawner } from "effect/unstable/process"
5
+ import { Installation } from "../../src/installation"
6
+ import { InstallationChannel } from "@miaw/core/installation/version"
7
+ import { AppProcess } from "@miaw/core/process"
8
+ import { testEffect } from "../lib/effect"
9
+
10
+ const encoder = new TextEncoder()
11
+
12
+ function mockHttpClient(handler: (request: HttpClientRequest.HttpClientRequest) => Response) {
13
+ const client = HttpClient.make((request) => Effect.succeed(HttpClientResponse.fromWeb(request, handler(request))))
14
+ return Layer.succeed(HttpClient.HttpClient, client)
15
+ }
16
+
17
+ function mockSpawner(
18
+ handler: (cmd: string, args: readonly string[]) => string | { code: number; stdout?: string; stderr?: string } = () =>
19
+ "",
20
+ ) {
21
+ const spawner = ChildProcessSpawner.make((command) => {
22
+ const std = ChildProcess.isStandardCommand(command) ? command : undefined
23
+ const result = handler(std?.command ?? "", std?.args ?? [])
24
+ const output = typeof result === "string" ? { code: 0, stdout: result, stderr: "" } : result
25
+ return Effect.succeed(
26
+ ChildProcessSpawner.makeHandle({
27
+ pid: ChildProcessSpawner.ProcessId(0),
28
+ exitCode: Effect.succeed(ChildProcessSpawner.ExitCode(output.code)),
29
+ isRunning: Effect.succeed(false),
30
+ kill: () => Effect.void,
31
+ stdin: { [Symbol.for("effect/Sink/TypeId")]: Symbol.for("effect/Sink/TypeId") } as any,
32
+ stdout: output.stdout ? Stream.make(encoder.encode(output.stdout)) : Stream.empty,
33
+ stderr: output.stderr ? Stream.make(encoder.encode(output.stderr)) : Stream.empty,
34
+ all: Stream.empty,
35
+ getInputFd: () => ({ [Symbol.for("effect/Sink/TypeId")]: Symbol.for("effect/Sink/TypeId") }) as any,
36
+ getOutputFd: () => Stream.empty,
37
+ unref: Effect.succeed(Effect.void),
38
+ }),
39
+ )
40
+ })
41
+ return Layer.succeed(ChildProcessSpawner.ChildProcessSpawner, spawner)
42
+ }
43
+
44
+ function jsonResponse(body: unknown) {
45
+ return new Response(JSON.stringify(body), {
46
+ status: 200,
47
+ headers: { "content-type": "application/json" },
48
+ })
49
+ }
50
+
51
+ function testLayer(
52
+ httpHandler: (request: HttpClientRequest.HttpClientRequest) => Response,
53
+ spawnHandler?: (cmd: string, args: readonly string[]) => string | { code: number; stdout?: string; stderr?: string },
54
+ ) {
55
+ const appProcess = AppProcess.layer.pipe(Layer.provide(mockSpawner(spawnHandler)))
56
+ return Installation.layer.pipe(Layer.provide(mockHttpClient(httpHandler)), Layer.provide(appProcess))
57
+ }
58
+
59
+ describe("installation", () => {
60
+ describe("latest", () => {
61
+ testEffect(testLayer(() => jsonResponse({ tag_name: "v1.2.3" }))).effect(
62
+ "reads release version from GitHub releases",
63
+ () =>
64
+ Effect.gen(function* () {
65
+ const result = yield* Installation.use.latest("unknown")
66
+ expect(result).toBe("1.2.3")
67
+ }),
68
+ )
69
+
70
+ testEffect(testLayer(() => jsonResponse({ tag_name: "v4.0.0-beta.1" }))).effect(
71
+ "strips v prefix from GitHub release tag",
72
+ () =>
73
+ Effect.gen(function* () {
74
+ const result = yield* Installation.use.latest("curl")
75
+ expect(result).toBe("4.0.0-beta.1")
76
+ }),
77
+ )
78
+
79
+ const npmCalls: string[] = []
80
+ testEffect(
81
+ testLayer((request) => {
82
+ npmCalls.push(request.url)
83
+ return jsonResponse({ version: "1.5.0" })
84
+ }),
85
+ ).effect("reads npm versions via registry", () =>
86
+ Effect.gen(function* () {
87
+ const result = yield* Installation.use.latest("npm")
88
+ expect(result).toBe("1.5.0")
89
+ expect(npmCalls).toContain(`https://registry.npmjs.org/miaw/${InstallationChannel}`)
90
+ }),
91
+ )
92
+
93
+ const bunCalls: string[] = []
94
+ testEffect(
95
+ testLayer((request) => {
96
+ bunCalls.push(request.url)
97
+ return jsonResponse({ version: "1.6.0" })
98
+ }),
99
+ ).effect("reads bun versions via registry", () =>
100
+ Effect.gen(function* () {
101
+ const result = yield* Installation.use.latest("bun")
102
+ expect(result).toBe("1.6.0")
103
+ expect(bunCalls).toContain(`https://registry.npmjs.org/miaw/${InstallationChannel}`)
104
+ }),
105
+ )
106
+
107
+ const pnpmCalls: string[] = []
108
+ testEffect(
109
+ testLayer((request) => {
110
+ pnpmCalls.push(request.url)
111
+ return jsonResponse({ version: "1.7.0" })
112
+ }),
113
+ ).effect("reads pnpm versions via registry", () =>
114
+ Effect.gen(function* () {
115
+ const result = yield* Installation.use.latest("pnpm")
116
+ expect(result).toBe("1.7.0")
117
+ expect(pnpmCalls).toContain(`https://registry.npmjs.org/miaw/${InstallationChannel}`)
118
+ }),
119
+ )
120
+
121
+ testEffect(testLayer(() => jsonResponse({ version: "2.3.4" }))).effect("reads scoop manifest versions", () =>
122
+ Effect.gen(function* () {
123
+ const result = yield* Installation.use.latest("scoop")
124
+ expect(result).toBe("2.3.4")
125
+ }),
126
+ )
127
+
128
+ testEffect(testLayer(() => jsonResponse({ d: { results: [{ Version: "3.4.5" }] } }))).effect(
129
+ "reads chocolatey feed versions",
130
+ () =>
131
+ Effect.gen(function* () {
132
+ const result = yield* Installation.use.latest("choco")
133
+ expect(result).toBe("3.4.5")
134
+ }),
135
+ )
136
+
137
+ testEffect(
138
+ testLayer(
139
+ () => jsonResponse({ versions: { stable: "2.0.0" } }),
140
+ (cmd, args) => {
141
+ // getBrewFormula: return core formula (no tap)
142
+ if (cmd === "brew" && args.includes("--formula") && args.includes("anomalyco/tap/miaw")) return ""
143
+ if (cmd === "brew" && args.includes("--formula") && args.includes("miaw")) return "miaw"
144
+ return ""
145
+ },
146
+ ),
147
+ ).effect("reads brew formulae API versions", () =>
148
+ Effect.gen(function* () {
149
+ const result = yield* Installation.use.latest("brew")
150
+ expect(result).toBe("2.0.0")
151
+ }),
152
+ )
153
+
154
+ const brewInfoJson = JSON.stringify({
155
+ formulae: [{ versions: { stable: "2.1.0" } }],
156
+ })
157
+ testEffect(
158
+ testLayer(
159
+ () => jsonResponse({}), // HTTP not used for tap formula
160
+ (cmd, args) => {
161
+ if (cmd === "brew" && args.includes("anomalyco/tap/miaw") && args.includes("--formula")) return "miaw"
162
+ if (cmd === "brew" && args.includes("--json=v2")) return brewInfoJson
163
+ return ""
164
+ },
165
+ ),
166
+ ).effect("reads brew tap info JSON via CLI", () =>
167
+ Effect.gen(function* () {
168
+ const result = yield* Installation.use.latest("brew")
169
+ expect(result).toBe("2.1.0")
170
+ }),
171
+ )
172
+ })
173
+
174
+ describe("upgrade", () => {
175
+ testEffect(
176
+ testLayer(
177
+ () => jsonResponse({}),
178
+ (cmd) => {
179
+ if (cmd === "npm") return { code: 1, stderr: "token=secret command output" }
180
+ return ""
181
+ },
182
+ ),
183
+ ).effect("returns sanitized typed errors for failed package upgrades", () =>
184
+ Effect.gen(function* () {
185
+ const error = yield* Effect.flip(Installation.use.upgrade("npm", "9.9.9"))
186
+ expect(error).toBeInstanceOf(Installation.UpgradeFailedError)
187
+ expect(error.stderr).toBe("Upgrade failed for npm (exit code 1).")
188
+ expect(error.message).toBe(error.stderr)
189
+ expect(error.stderr).not.toContain("secret")
190
+ expect(error.stderr).not.toContain("command output")
191
+ }),
192
+ )
193
+
194
+ testEffect(
195
+ testLayer(
196
+ () => new Response("install script with token=secret", { status: 200 }),
197
+ (cmd, args) => {
198
+ if (cmd === "bash" && args[0] === "--version") return "GNU bash"
199
+ if (cmd === "bash" || cmd === "sh") return { code: 1, stderr: "script output with token=secret" }
200
+ return ""
201
+ },
202
+ ),
203
+ ).effect("returns sanitized typed errors when the curl install script fails", () =>
204
+ Effect.gen(function* () {
205
+ const error = yield* Effect.flip(Installation.use.upgrade("curl", "9.9.9"))
206
+ expect(error).toBeInstanceOf(Installation.UpgradeFailedError)
207
+ expect(error.stderr).toBe("Upgrade failed for curl (exit code 1).")
208
+ expect(error.message).toBe(error.stderr)
209
+ expect(error.stderr).not.toContain("secret")
210
+ expect(error.stderr).not.toContain("script output")
211
+ }),
212
+ )
213
+
214
+ testEffect(
215
+ testLayer(
216
+ () => new Response("install script", { status: 200 }),
217
+ (cmd, args) => {
218
+ if (cmd === "bash" && args[0] === "--version") return { code: 1, stderr: "missing" }
219
+ if (cmd === "bash") return { code: 1, stderr: "should not execute installer with bash" }
220
+ if (cmd === "sh") return "ok"
221
+ return ""
222
+ },
223
+ ),
224
+ ).effect("falls back to sh when bash is unavailable during curl upgrade", () =>
225
+ Effect.gen(function* () {
226
+ yield* Installation.use.upgrade("curl", "9.9.9")
227
+ }),
228
+ )
229
+ })
230
+ })
@@ -0,0 +1,459 @@
1
+ // Subprocess test harness for the miaw CLI. Spawns the real binary against
2
+ // a TestLLMServer running in-process at a random port, with full env isolation.
3
+ //
4
+ // This is the missing test tier: in-process tests can't catch bugs that span
5
+ // argv parsing → server boot → SDK call → event consumption → exit code (like
6
+ // the original /event race or #27371's invalid-model hang).
7
+ //
8
+ // Configuration flows through miaw's built-in test affordances:
9
+ // - MIAW_CONFIG_CONTENT : provider config inline, no files to find
10
+ // - MIAW_TEST_HOME : pins os.homedir() → tmpdir
11
+ // - MIAW_DISABLE_PROJECT_CONFIG : skip walking up for miaw.json
12
+ // - MIAW_PURE : skip external plugin discovery + install
13
+ // - MIAW_DISABLE_AUTOUPDATE / AUTOCOMPACT / MODELS_FETCH : no background work
14
+ // Plus HOME / XDG_* pointing at the tmpdir for belt-and-suspenders isolation.
15
+ //
16
+ // Today only `miaw.run` is fully wired. The shape supports adding more
17
+ // builders (`miaw.serve(opts)`, `miaw.acp(opts)`, `miaw.auth(...)`)
18
+ // without changing the fixture. Long-lived commands like `serve` will need a
19
+ // different return shape — see the TODO at the bottom of MiawCli.
20
+ import { test, type TestOptions } from "bun:test"
21
+ import { FSUtil } from "@miaw/core/fs-util"
22
+ import { AppProcess } from "@miaw/core/process"
23
+ import { Deferred, Duration, Effect, Layer, Queue, Scope, Stream } from "effect"
24
+ import { FetchHttpClient, HttpClient } from "effect/unstable/http"
25
+ import { ChildProcess } from "effect/unstable/process"
26
+ import path from "node:path"
27
+ import { TestLLMServer } from "./llm-server"
28
+ import { testProviderConfig } from "./test-provider"
29
+ import { it } from "./effect"
30
+
31
+ const miawRoot = path.resolve(import.meta.dir, "../../")
32
+ const cliEntry = path.join(miawRoot, "src/index.ts")
33
+
34
+ export const testModelID = "test/test-model"
35
+
36
+ // Wrap a Bun subprocess pipe (or any ReadableStream<Uint8Array>) as a Stream.
37
+ // Centralizes the `evaluate` + `onError` boilerplate and tags errors with the
38
+ // stream name so a stderr/stdout failure is greppable in logs.
39
+ function fromBunStream(name: string, get: () => ReadableStream<Uint8Array>) {
40
+ return Stream.fromReadableStream({
41
+ evaluate: get,
42
+ onError: (cause) => new Error(`${name} stream error: ${String(cause)}`),
43
+ })
44
+ }
45
+
46
+ // Long-lived processes (serve, acp) all want the same stderr drain: read every
47
+ // chunk, push to a tail buffer, swallow stream errors (the child closing the
48
+ // pipe is normal). `log: true` surfaces a real protocol error to logs so a
49
+ // regression doesn't silently disappear.
50
+ function forkStderrDrain(stream: ReadableStream<Uint8Array>, into: string[]) {
51
+ return Effect.forkScoped(
52
+ fromBunStream("stderr", () => stream).pipe(
53
+ Stream.decodeText(),
54
+ Stream.runForEach((chunk) => Effect.sync(() => into.push(chunk))),
55
+ Effect.ignore({ log: true }),
56
+ ),
57
+ )
58
+ }
59
+
60
+ function isolatedEnv(home: string, configJson: string): Record<string, string> {
61
+ return {
62
+ MIAW_TEST_HOME: home,
63
+ HOME: home,
64
+ XDG_CONFIG_HOME: path.join(home, ".config"),
65
+ XDG_DATA_HOME: path.join(home, ".local/share"),
66
+ XDG_STATE_HOME: path.join(home, ".local/state"),
67
+ XDG_CACHE_HOME: path.join(home, ".cache"),
68
+ MIAW_CONFIG_CONTENT: configJson,
69
+ MIAW_DISABLE_PROJECT_CONFIG: "1",
70
+ MIAW_PURE: "1",
71
+ MIAW_DISABLE_AUTOUPDATE: "1",
72
+ MIAW_DISABLE_AUTOCOMPACT: "1",
73
+ MIAW_DISABLE_MODELS_FETCH: "1",
74
+ MIAW_AUTH_CONTENT: "{}",
75
+ }
76
+ }
77
+
78
+ export type RunResult = {
79
+ readonly exitCode: number
80
+ readonly stdout: string
81
+ readonly stderr: string
82
+ readonly durationMs: number
83
+ }
84
+
85
+ export type SpawnOpts = { readonly timeoutMs?: number; readonly env?: Record<string, string> }
86
+
87
+ // Typed equivalent of constructing argv for `miaw run`. New flags should
88
+ // land here so tests stay grep-able and refactor-safe.
89
+ export type RunOpts = SpawnOpts & {
90
+ readonly model?: string
91
+ readonly agent?: string
92
+ readonly format?: "default" | "json"
93
+ readonly command?: string
94
+ readonly printLogs?: boolean
95
+ readonly extraArgs?: string[]
96
+ }
97
+
98
+ // `miaw serve` is a long-lived process — it never exits on its own.
99
+ // `serve(opts)` therefore returns a handle inside the caller's Scope: the
100
+ // subprocess is killed when the scope closes (test end), and the URL the
101
+ // server actually bound to (port 0 means OS-assigned) is parsed off stdout.
102
+ export type ServeOpts = SpawnOpts & {
103
+ readonly port?: number
104
+ readonly hostname?: string
105
+ readonly extraArgs?: string[]
106
+ // How long to wait for the "listening on http://..." line before failing.
107
+ // Default 15s — startup is dominated by bun's transpile + plugin init, not
108
+ // the actual listen() call.
109
+ readonly readyTimeoutMs?: number
110
+ }
111
+
112
+ export type ServeHandle = {
113
+ // Full URL the server is bound to, e.g. "http://127.0.0.1:54321". Use this
114
+ // as the base for HTTP requests in tests — never assume the port.
115
+ readonly url: string
116
+ readonly hostname: string
117
+ readonly port: number
118
+ // Sends SIGTERM. The scope finalizer also calls this, so tests rarely need
119
+ // to invoke it directly — useful for tests that assert exit behavior.
120
+ readonly kill: () => void
121
+ // Resolves with the exit code once the process exits. Bun returns a number.
122
+ readonly exited: Promise<number>
123
+ }
124
+
125
+ // `miaw acp` speaks newline-delimited JSON-RPC over stdin/stdout. It is
126
+ // long-lived and exits cleanly when stdin is closed. The handle exposes the
127
+ // duplex stream as send/receive rather than raw pipes so tests don't have to
128
+ // reimplement framing on every call site.
129
+ export type AcpOpts = SpawnOpts & {
130
+ readonly cwd?: string
131
+ readonly extraArgs?: string[]
132
+ }
133
+
134
+ export type AcpHandle = {
135
+ // Writes a single JSON-RPC message to the child's stdin as one ndjson line.
136
+ readonly send: (msg: object) => Effect.Effect<void>
137
+ // Resolves with the next parsed JSON-RPC line from the child's stdout.
138
+ // Lines are buffered in a queue so multiple receives in a row won't drop
139
+ // anything. Pair with `Effect.timeout` if a test wants a deadline.
140
+ readonly receive: Effect.Effect<unknown>
141
+ // Closes stdin. ACP exits cleanly on stdin EOF; the scope finalizer also
142
+ // calls this, so tests only need it when asserting exit behavior.
143
+ readonly close: () => void
144
+ readonly exited: Promise<number>
145
+ }
146
+
147
+ export type MiawCli = {
148
+ // High-level: run a single prompt against the test model. Short-lived.
149
+ readonly run: (message: string, opts?: RunOpts) => Effect.Effect<RunResult>
150
+ // Spawn `miaw serve` and wait until it's listening. Long-lived: the
151
+ // returned handle is killed when the caller's Scope closes. Fails if the
152
+ // listening line doesn't appear within `readyTimeoutMs`.
153
+ readonly serve: (opts?: ServeOpts) => Effect.Effect<ServeHandle, Error, Scope.Scope>
154
+ // Spawn `miaw acp` and return a duplex JSON-RPC handle. Long-lived:
155
+ // the subprocess exits on stdin close, which the scope finalizer triggers.
156
+ readonly acp: (opts?: AcpOpts) => Effect.Effect<AcpHandle, Error, Scope.Scope>
157
+ // Escape hatch: any CLI invocation with full control over argv. Used to test
158
+ // commands that don't yet have a typed builder.
159
+ readonly spawn: (args: string[], opts?: SpawnOpts) => Effect.Effect<RunResult>
160
+ // Convenience assertion. Dumps captured stderr/stdout on mismatch so CI
161
+ // failures are debuggable without re-running locally.
162
+ readonly expectExit: (result: RunResult, expected: number, label?: string) => void
163
+ // Parse `--format json` stdout into one event object per non-empty line.
164
+ // The CLI writes `JSON.stringify({ type, sessionID, ... }) + EOL` for each
165
+ // event (see src/cli/cmd/run.ts `emit`). Throws on a malformed line so
166
+ // tests fail loudly rather than silently skipping data.
167
+ readonly parseJsonEvents: (stdout: string) => Array<Record<string, unknown>>
168
+ }
169
+
170
+ export type CliFixture = {
171
+ readonly llm: TestLLMServer["Service"]
172
+ readonly home: string
173
+ readonly miaw: MiawCli
174
+ }
175
+
176
+ // Provisions a TestLLMServer + tmpdir + spawn helper and invokes fn. Cleans
177
+ // up the tmpdir on scope exit. TestLLMServer.layer is provided internally so
178
+ // the caller doesn't need to wire it up — the fixture's lifetime is tied to
179
+ // the surrounding Scope.
180
+ export function withCliFixture<A, E>(
181
+ fn: (input: CliFixture) => Effect.Effect<A, E, Scope.Scope | HttpClient.HttpClient>,
182
+ ): Effect.Effect<A, E | unknown, Scope.Scope> {
183
+ return Effect.gen(function* () {
184
+ const llm = yield* TestLLMServer
185
+ const fs = yield* FSUtil.Service
186
+ const appProc = yield* AppProcess.Service
187
+
188
+ // FileSystem.makeTempDirectoryScoped handles both creation and scope-tied
189
+ // cleanup — replaces the old mkdir + addFinalizer pair.
190
+ const home = yield* fs.makeTempDirectoryScoped({ prefix: "oc-cli-" })
191
+
192
+ const configJson = JSON.stringify(testProviderConfig(llm.url))
193
+ const env = isolatedEnv(home, configJson)
194
+
195
+ const spawn = Effect.fn("miaw.spawn")(function* (args: string[], opts?: SpawnOpts) {
196
+ const start = Date.now()
197
+ const timeoutMs = opts?.timeoutMs ?? 30_000
198
+ // stdin: "ignore" so the child doesn't see a piped stdin and block
199
+ // on `Bun.stdin.text()` (see src/cli/cmd/run.ts — non-TTY stdin is
200
+ // consumed as the prompt). The old Process.run wrapper defaulted to
201
+ // ignore; ChildProcess.make defaults to pipe, so we set it explicitly.
202
+ const command = ChildProcess.make("bun", ["run", "--conditions=browser", cliEntry, ...args], {
203
+ cwd: home,
204
+ env: { ...env, ...opts?.env },
205
+ extendEnv: true,
206
+ stdin: "ignore",
207
+ })
208
+ // Pass timeout to appProc.run rather than wrapping with
209
+ // Effect.timeoutOrElse externally: AppProcess.run is itself scoped, so
210
+ // its built-in timeout triggers the acquireRelease kill finalizer
211
+ // inside cross-spawn-spawner *before* surfacing the AppProcessError —
212
+ // guaranteeing the child is dead by the time the test continues.
213
+ // External timeoutOrElse interrupts the run fiber but races the
214
+ // scope close, which can leak the child past the test boundary.
215
+ //
216
+ // Catch AppProcessError (timeout OR spawn failure) and synthesize a
217
+ // non-zero result so the test sees it via the usual `expectExit`
218
+ // path rather than as an unhandled Effect failure.
219
+ const result = yield* appProc.run(command, { timeout: Duration.millis(timeoutMs) }).pipe(
220
+ Effect.catchTag("AppProcessError", (err) =>
221
+ Effect.succeed({
222
+ command: err.command,
223
+ exitCode: err.exitCode ?? -1,
224
+ stdout: Buffer.alloc(0),
225
+ stderr: Buffer.from((err.stderr ?? String(err.cause ?? err.message)) + "\n"),
226
+ stdoutTruncated: false,
227
+ stderrTruncated: false,
228
+ } satisfies AppProcess.RunResult),
229
+ ),
230
+ )
231
+ return {
232
+ exitCode: result.exitCode,
233
+ stdout: result.stdout.toString(),
234
+ stderr: result.stderr.toString(),
235
+ durationMs: Date.now() - start,
236
+ }
237
+ })
238
+
239
+ const run = (message: string, opts?: RunOpts): Effect.Effect<RunResult> => {
240
+ const argv: string[] = ["run"]
241
+ if (opts?.printLogs) argv.push("--print-logs")
242
+ argv.push("--model", opts?.model ?? testModelID)
243
+ if (opts?.agent) argv.push("--agent", opts.agent)
244
+ if (opts?.format) argv.push("--format", opts.format)
245
+ if (opts?.command) argv.push("--command", opts.command)
246
+ if (opts?.extraArgs) argv.push(...opts.extraArgs)
247
+ argv.push(message)
248
+ return spawn(argv, opts)
249
+ }
250
+
251
+ const serve = Effect.fn("miaw.serve")(function* (opts?: ServeOpts) {
252
+ const argv = ["serve"]
253
+ // Default port 0 — let the OS pick a free port, parse the actual one
254
+ // off stdout. Hard-coded ports flake under parallel tests.
255
+ argv.push("--port", String(opts?.port ?? 0))
256
+ if (opts?.hostname) argv.push("--hostname", opts.hostname)
257
+ if (opts?.extraArgs) argv.push(...opts.extraArgs)
258
+
259
+ // Acquire the subprocess; release sends SIGTERM and awaits exit on
260
+ // scope close. Wrapped in Effect.ignore so a flaky kill doesn't surface
261
+ // as a finalizer error during test teardown.
262
+ const proc = yield* Effect.acquireRelease(
263
+ Effect.sync(() =>
264
+ Bun.spawn(["bun", "run", "--conditions=browser", cliEntry, ...argv], {
265
+ cwd: home,
266
+ env: { ...process.env, ...env, ...opts?.env },
267
+ stdout: "pipe",
268
+ stderr: "pipe",
269
+ }),
270
+ ),
271
+ (p) =>
272
+ Effect.promise(() => {
273
+ p.kill()
274
+ return p.exited
275
+ }).pipe(Effect.ignore),
276
+ )
277
+
278
+ // Tail buffer so timeout failures can include stderr context. The fork
279
+ // also keeps the OS pipe buffer from filling and wedging the child.
280
+ const stderrChunks: string[] = []
281
+ yield* forkStderrDrain(proc.stderr, stderrChunks)
282
+
283
+ // Watch stdout line-by-line for the listening sentinel. Format
284
+ // (see src/cli/cmd/serve.ts):
285
+ // "miaw server listening on http://<host>:<port>"
286
+ const readyRe = /listening on (http:\/\/([^\s:]+):(\d+))/
287
+ const readyDeferred = yield* Deferred.make<{ url: string; hostname: string; port: number }>()
288
+ yield* Effect.forkScoped(
289
+ fromBunStream("stdout", () => proc.stdout).pipe(
290
+ Stream.decodeText(),
291
+ Stream.splitLines,
292
+ Stream.runForEach((line) => {
293
+ const m = line.match(readyRe)
294
+ return m ? Deferred.succeed(readyDeferred, { url: m[1], hostname: m[2], port: Number(m[3]) }) : Effect.void
295
+ }),
296
+ Effect.ignore({ log: true }),
297
+ ),
298
+ )
299
+
300
+ const readyTimeoutMs = opts?.readyTimeoutMs ?? 15_000
301
+ const match = yield* Deferred.await(readyDeferred).pipe(
302
+ Effect.timeoutOrElse({
303
+ duration: Duration.millis(readyTimeoutMs),
304
+ orElse: () =>
305
+ Effect.fail(
306
+ new Error(
307
+ `miaw serve did not become ready within ${readyTimeoutMs}ms\n` +
308
+ `stderr (last 2000):\n${stderrChunks.join("").slice(-2000)}`,
309
+ ),
310
+ ),
311
+ }),
312
+ )
313
+
314
+ return {
315
+ url: match.url,
316
+ hostname: match.hostname,
317
+ port: match.port,
318
+ kill: () => {
319
+ proc.kill()
320
+ },
321
+ exited: proc.exited as Promise<number>,
322
+ } satisfies ServeHandle
323
+ })
324
+
325
+ const acp = Effect.fn("miaw.acp")(function* (opts?: AcpOpts) {
326
+ const argv = ["acp"]
327
+ if (opts?.cwd) argv.push("--cwd", opts.cwd)
328
+ if (opts?.extraArgs) argv.push(...opts.extraArgs)
329
+
330
+ // Acquire the subprocess. Release ends stdin (clean shutdown — ACP exits
331
+ // on stdin EOF) and falls back to SIGTERM if it doesn't exit promptly.
332
+ // Either way we await proc.exited so the test scope doesn't leak.
333
+ const proc = yield* Effect.acquireRelease(
334
+ Effect.sync(() =>
335
+ Bun.spawn(["bun", "run", "--conditions=browser", cliEntry, ...argv], {
336
+ cwd: opts?.cwd ?? home,
337
+ env: { ...process.env, ...env, ...opts?.env },
338
+ stdin: "pipe",
339
+ stdout: "pipe",
340
+ stderr: "pipe",
341
+ }),
342
+ ),
343
+ (p) =>
344
+ // Graceful shutdown: close stdin (ACP exits on EOF), give it a
345
+ // window to exit, then SIGTERM. The Effect.timeoutOrElse expresses
346
+ // exactly that race without raw setTimeout or Promise.race.
347
+ Effect.gen(function* () {
348
+ yield* Effect.sync(() => p.stdin.end())
349
+ yield* Effect.promise(() => p.exited).pipe(
350
+ Effect.timeoutOrElse({
351
+ duration: Duration.seconds(2),
352
+ orElse: () =>
353
+ Effect.sync(() => {
354
+ p.kill()
355
+ }),
356
+ }),
357
+ )
358
+ yield* Effect.promise(() => p.exited)
359
+ }).pipe(Effect.ignore),
360
+ )
361
+
362
+ const stderrChunks: string[] = []
363
+ yield* forkStderrDrain(proc.stderr, stderrChunks)
364
+
365
+ // Each ndjson line becomes one queue entry. JSON.parse failures are
366
+ // surfaced as the raw string so a malformed protocol message doesn't
367
+ // silently wedge the test in `receive`.
368
+ const responses = yield* Queue.unbounded<unknown>()
369
+ yield* Effect.forkScoped(
370
+ fromBunStream("stdout", () => proc.stdout).pipe(
371
+ Stream.decodeText(),
372
+ Stream.splitLines,
373
+ Stream.runForEach((line) => {
374
+ if (line.length === 0) return Effect.void
375
+ let parsed: unknown
376
+ try {
377
+ parsed = JSON.parse(line)
378
+ } catch {
379
+ parsed = { _rawLine: line }
380
+ }
381
+ return Queue.offer(responses, parsed)
382
+ }),
383
+ Effect.ignore({ log: true }),
384
+ ),
385
+ )
386
+
387
+ return {
388
+ // `proc.stdin.write` returns `number | Promise<number>`. The promise
389
+ // form is the backpressure signal — if we don't await it, rapid
390
+ // successive sends can interleave under pipe-buffer-full conditions
391
+ // and corrupt the ndjson framing.
392
+ send: (msg: object) =>
393
+ Effect.promise(async () => {
394
+ const ret = proc.stdin.write(JSON.stringify(msg) + "\n")
395
+ if (typeof ret !== "number") await ret
396
+ }),
397
+ receive: Queue.take(responses),
398
+ // proc.stdin.end() is idempotent in Bun; no try/catch needed.
399
+ close: () => proc.stdin.end(),
400
+ exited: proc.exited as Promise<number>,
401
+ } satisfies AcpHandle
402
+ })
403
+
404
+ const miaw: MiawCli = { run, serve, acp, spawn, expectExit, parseJsonEvents }
405
+
406
+ return yield* fn({ llm, home, miaw })
407
+ // FetchHttpClient is provided so test bodies can `yield* HttpClient.HttpClient`
408
+ // and hit endpoints on `miaw.serve()` without rolling their own fetch.
409
+ }).pipe(
410
+ Effect.provide(
411
+ Layer.mergeAll(TestLLMServer.layer, FetchHttpClient.layer, FSUtil.defaultLayer, AppProcess.defaultLayer),
412
+ ),
413
+ )
414
+ }
415
+
416
+ function parseJsonEvents(stdout: string): Array<Record<string, unknown>> {
417
+ return stdout
418
+ .split("\n")
419
+ .map((line) => line.trim())
420
+ .filter((line) => line.length > 0)
421
+ .map((line) => JSON.parse(line) as Record<string, unknown>)
422
+ }
423
+
424
+ // Convenience for the common assertion pattern. Dumps stderr/stdout when
425
+ // the exit code doesn't match — saves debugging time on CI failures.
426
+ function expectExit(result: RunResult, expected: number, label = "miaw") {
427
+ if (result.exitCode === expected) return
428
+ const tail = (s: string, n: number) => (s.length > n ? "..." + s.slice(-n) : s)
429
+ // eslint-disable-next-line no-console
430
+ console.error(`[${label}] expected exit ${expected}, got ${result.exitCode} after ${result.durationMs}ms`)
431
+ // eslint-disable-next-line no-console
432
+ console.error(`[${label}] stderr (last 2000):\n${tail(result.stderr, 2000)}`)
433
+ // eslint-disable-next-line no-console
434
+ console.error(`[${label}] stdout (last 500):\n${tail(result.stdout, 500)}`)
435
+ throw new Error(`${label}: expected exit ${expected}, got ${result.exitCode}`)
436
+ }
437
+
438
+ // `cliIt.live(name, fixture => effect)` is the same as
439
+ // `it.live(name, () => withCliFixture(fixture))` — one fewer nesting level at
440
+ // every call site. Use this for any test that needs the miaw CLI fixture.
441
+ //
442
+ // Subprocess tests must run against the real clock — a TestClock-paused
443
+ // environment can't drive a child process. If you need `.only` or `.skip`, fall
444
+ // back to `it.live` + `withCliFixture` directly.
445
+ // Body's R is `Scope.Scope | never` so tests can yield* scope-requiring
446
+ // resources (e.g. `miaw.serve`) without an extra `Effect.scoped` wrapper —
447
+ // `withCliFixture`'s outer scope is the natural lifetime.
448
+ export const cliIt = {
449
+ live: <A, E>(
450
+ name: string,
451
+ body: (input: CliFixture) => Effect.Effect<A, E, Scope.Scope | HttpClient.HttpClient>,
452
+ opts?: number | TestOptions,
453
+ ) => it.live(name, () => withCliFixture(body), opts),
454
+ concurrent: <A, E>(
455
+ name: string,
456
+ body: (input: CliFixture) => Effect.Effect<A, E, Scope.Scope | HttpClient.HttpClient>,
457
+ opts?: number | TestOptions,
458
+ ) => test.concurrent(name, () => Effect.runPromise(Effect.scoped(withCliFixture(body))), opts),
459
+ }