rird 2.1.231 → 2.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (381) hide show
  1. package/AGENTS.md +86 -0
  2. package/COMPLETED_TEST_SUITE.txt +280 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +397 -6
  5. package/RIRD_ERROR_HANDLING_SUMMARY.md +307 -0
  6. package/TESTING.md +512 -0
  7. package/TEST_IMPLEMENTATION_REPORT.md +463 -0
  8. package/TEST_SUITE.md +307 -0
  9. package/TEST_SUMMARY.txt +380 -0
  10. package/bin/rird-perf.js +37 -0
  11. package/bin/rird.js +43 -8
  12. package/bunfig.toml +4 -0
  13. package/create-wrapper.ps1 +51 -0
  14. package/docs/ARCHITECTURE.md +768 -0
  15. package/docs/CLI_REFERENCE.md +681 -0
  16. package/docs/DOCUMENTATION_MANIFEST.md +392 -0
  17. package/docs/INDEX.md +295 -0
  18. package/docs/PRODUCTION_SETUP.md +633 -0
  19. package/docs/TROUBLESHOOTING.md +914 -0
  20. package/facebook_ads_library.png +0 -0
  21. package/nul +0 -0
  22. package/nul`nif +0 -0
  23. package/package.json +104 -15
  24. package/parsers-config.ts +239 -0
  25. package/rird-1.0.199.tgz +0 -0
  26. package/rird-1.0.205.tgz +0 -0
  27. package/script/build-windows.ts +56 -0
  28. package/script/build.ts +165 -0
  29. package/{postinstall.mjs → script/postinstall.mjs} +47 -68
  30. package/script/publish-registries.ts +187 -0
  31. package/script/publish.ts +85 -0
  32. package/script/schema.ts +47 -0
  33. package/src/acp/README.md +164 -0
  34. package/src/acp/agent.ts +1063 -0
  35. package/src/acp/session.ts +101 -0
  36. package/src/acp/types.ts +22 -0
  37. package/src/agent/agent.ts +367 -0
  38. package/src/agent/generate.txt +75 -0
  39. package/src/agent/prompt/compaction.txt +12 -0
  40. package/src/agent/prompt/explore.txt +18 -0
  41. package/src/agent/prompt/summary.txt +10 -0
  42. package/src/agent/prompt/title.txt +36 -0
  43. package/src/auth/index.ts +70 -0
  44. package/src/bun/index.ts +114 -0
  45. package/src/bus/bus-event.ts +43 -0
  46. package/src/bus/global.ts +10 -0
  47. package/src/bus/index.ts +105 -0
  48. package/src/cli/bootstrap.ts +17 -0
  49. package/src/cli/cmd/acp.ts +104 -0
  50. package/src/cli/cmd/activate.ts +50 -0
  51. package/src/cli/cmd/agent.ts +256 -0
  52. package/src/cli/cmd/auth.ts +412 -0
  53. package/src/cli/cmd/cmd.ts +7 -0
  54. package/src/cli/cmd/debug/config.ts +15 -0
  55. package/src/cli/cmd/debug/file.ts +91 -0
  56. package/src/cli/cmd/debug/index.ts +43 -0
  57. package/src/cli/cmd/debug/lsp.ts +48 -0
  58. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  59. package/src/cli/cmd/debug/scrap.ts +15 -0
  60. package/src/cli/cmd/debug/skill.ts +15 -0
  61. package/src/cli/cmd/debug/snapshot.ts +48 -0
  62. package/src/cli/cmd/export.ts +88 -0
  63. package/src/cli/cmd/generate.ts +38 -0
  64. package/src/cli/cmd/github.ts +1400 -0
  65. package/src/cli/cmd/import.ts +98 -0
  66. package/src/cli/cmd/mcp.ts +654 -0
  67. package/src/cli/cmd/models.ts +68 -0
  68. package/src/cli/cmd/pr.ts +112 -0
  69. package/src/cli/cmd/run.ts +434 -0
  70. package/src/cli/cmd/serve.ts +31 -0
  71. package/src/cli/cmd/session.ts +106 -0
  72. package/src/cli/cmd/stats.ts +298 -0
  73. package/src/cli/cmd/tui/app.tsx +694 -0
  74. package/src/cli/cmd/tui/attach.ts +30 -0
  75. package/src/cli/cmd/tui/component/border.tsx +21 -0
  76. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  77. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  78. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  79. package/src/cli/cmd/tui/component/dialog-model.tsx +236 -0
  80. package/src/cli/cmd/tui/component/dialog-provider.tsx +240 -0
  81. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  82. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  83. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  84. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  85. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  86. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  87. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  88. package/src/cli/cmd/tui/component/logo.tsx +48 -0
  89. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +574 -0
  90. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  91. package/src/cli/cmd/tui/component/prompt/index.tsx +1087 -0
  92. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  93. package/src/cli/cmd/tui/component/tips.ts +27 -0
  94. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  95. package/src/cli/cmd/tui/context/args.tsx +14 -0
  96. package/src/cli/cmd/tui/context/directory.ts +13 -0
  97. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  98. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  99. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  100. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  101. package/src/cli/cmd/tui/context/local.tsx +345 -0
  102. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  103. package/src/cli/cmd/tui/context/route.tsx +46 -0
  104. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  105. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  106. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  107. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  108. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  109. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  110. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  111. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  112. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  114. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  115. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  116. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  117. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  118. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  119. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  120. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  121. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  122. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  123. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  124. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  125. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  126. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  127. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  128. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  129. package/src/cli/cmd/tui/context/theme/rird.json +245 -0
  130. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  131. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  132. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  133. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  134. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  135. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  136. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  137. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  138. package/src/cli/cmd/tui/event.ts +40 -0
  139. package/src/cli/cmd/tui/hooks/use-safe-terminal-dimensions.ts +12 -0
  140. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  141. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  142. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  143. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  144. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  145. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  146. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  147. package/src/cli/cmd/tui/routes/session/index.tsx +1876 -0
  148. package/src/cli/cmd/tui/routes/session/sidebar.tsx +320 -0
  149. package/src/cli/cmd/tui/spawn.ts +60 -0
  150. package/src/cli/cmd/tui/thread.ts +142 -0
  151. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  152. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  153. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  154. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  155. package/src/cli/cmd/tui/ui/dialog-select.tsx +333 -0
  156. package/src/cli/cmd/tui/ui/dialog.tsx +171 -0
  157. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  158. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  159. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  160. package/src/cli/cmd/tui/util/editor.ts +32 -0
  161. package/src/cli/cmd/tui/util/terminal.ts +146 -0
  162. package/src/cli/cmd/tui/worker.ts +63 -0
  163. package/src/cli/cmd/uninstall.ts +344 -0
  164. package/src/cli/cmd/upgrade.ts +127 -0
  165. package/src/cli/cmd/web.ts +84 -0
  166. package/src/cli/error.ts +69 -0
  167. package/src/cli/ui.ts +101 -0
  168. package/src/cli/upgrade.ts +28 -0
  169. package/src/command/index.ts +80 -0
  170. package/src/command/template/initialize.txt +10 -0
  171. package/src/command/template/review.txt +97 -0
  172. package/src/config/config.ts +994 -0
  173. package/src/config/markdown.ts +41 -0
  174. package/src/env/index.ts +26 -0
  175. package/src/file/ignore.ts +83 -0
  176. package/src/file/index.ts +328 -0
  177. package/src/file/ripgrep.ts +393 -0
  178. package/src/file/time.ts +64 -0
  179. package/src/file/watcher.ts +103 -0
  180. package/src/flag/flag.ts +84 -0
  181. package/src/format/formatter.ts +315 -0
  182. package/src/format/index.ts +137 -0
  183. package/src/global/index.ts +101 -0
  184. package/src/id/id.ts +73 -0
  185. package/src/ide/index.ts +76 -0
  186. package/src/index.ts +297 -0
  187. package/src/index.ts.backup +271 -0
  188. package/src/installation/index.ts +258 -0
  189. package/src/lib/IMPLEMENTATION_NOTES.md +345 -0
  190. package/src/lib/error-handler.ts +225 -0
  191. package/src/lib/error-testing-guide.md +258 -0
  192. package/src/lib/errors.ts +285 -0
  193. package/src/lib/performance.ts +70 -0
  194. package/src/lib/telemetry.ts +282 -0
  195. package/src/lsp/client.ts +229 -0
  196. package/src/lsp/index.ts +485 -0
  197. package/src/lsp/language.ts +116 -0
  198. package/src/lsp/server.ts +1895 -0
  199. package/src/mcp/auth.ts +135 -0
  200. package/src/mcp/index.ts +1117 -0
  201. package/src/mcp/intent-analyzer.ts +376 -0
  202. package/src/mcp/oauth-callback.ts +200 -0
  203. package/src/mcp/oauth-provider.ts +154 -0
  204. package/src/patch/index.ts +632 -0
  205. package/src/permission/index.ts +199 -0
  206. package/src/plugin/index.ts +91 -0
  207. package/src/project/bootstrap.ts +33 -0
  208. package/src/project/instance.ts +78 -0
  209. package/src/project/project.ts +236 -0
  210. package/src/project/state.ts +65 -0
  211. package/src/project/vcs.ts +76 -0
  212. package/src/provider/auth.ts +143 -0
  213. package/src/provider/models-macro.ts +55 -0
  214. package/src/provider/models.ts +161 -0
  215. package/src/provider/provider.ts +1109 -0
  216. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  217. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  218. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  219. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  220. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  221. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  222. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  223. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  224. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  225. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  226. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  227. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  228. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  229. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  230. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  231. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  232. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  233. package/src/provider/transform.ts +455 -0
  234. package/src/pty/index.ts +231 -0
  235. package/src/security/guardrails.test.ts +341 -0
  236. package/src/security/guardrails.ts +570 -0
  237. package/src/security/index.ts +19 -0
  238. package/src/server/error.ts +36 -0
  239. package/src/server/project.ts +79 -0
  240. package/src/server/server.ts +2641 -0
  241. package/src/server/tui.ts +71 -0
  242. package/src/session/compaction.ts +228 -0
  243. package/src/session/index.ts +464 -0
  244. package/src/session/llm.ts +201 -0
  245. package/src/session/message-v2.ts +695 -0
  246. package/src/session/message.ts +189 -0
  247. package/src/session/processor.ts +409 -0
  248. package/src/session/prompt/act-switch.txt +5 -0
  249. package/src/session/prompt/anthropic-20250930.txt +166 -0
  250. package/src/session/prompt/anthropic.txt +63 -0
  251. package/src/session/prompt/anthropic_spoof.txt +1 -0
  252. package/src/session/prompt/beast.txt +76 -0
  253. package/src/session/prompt/codex.txt +304 -0
  254. package/src/session/prompt/copilot-gpt-5.txt +137 -0
  255. package/src/session/prompt/gemini.txt +62 -0
  256. package/src/session/prompt/max-steps.txt +16 -0
  257. package/src/session/prompt/plan-reminder-anthropic.txt +35 -0
  258. package/src/session/prompt/plan.txt +24 -0
  259. package/src/session/prompt/polaris.txt +88 -0
  260. package/src/session/prompt/qwen.txt +59 -0
  261. package/src/session/prompt.ts +1552 -0
  262. package/src/session/retry.ts +86 -0
  263. package/src/session/revert.ts +108 -0
  264. package/src/session/sensitive-filter.test.ts +327 -0
  265. package/src/session/sensitive-filter.ts +466 -0
  266. package/src/session/status.ts +76 -0
  267. package/src/session/summary.ts +209 -0
  268. package/src/session/system.ts +122 -0
  269. package/src/session/todo.ts +37 -0
  270. package/src/share/share-next.ts +222 -0
  271. package/src/share/share.ts +87 -0
  272. package/src/shell/shell.ts +67 -0
  273. package/src/skill/index.ts +1 -0
  274. package/src/skill/skill.ts +83 -0
  275. package/src/snapshot/index.ts +197 -0
  276. package/src/storage/storage.ts +226 -0
  277. package/src/tests/agent.test.ts +308 -0
  278. package/src/tests/build-guards.test.ts +267 -0
  279. package/src/tests/config.test.ts +664 -0
  280. package/src/tests/tool-registry.test.ts +589 -0
  281. package/src/tool/bash.ts +314 -0
  282. package/src/tool/bash.txt +158 -0
  283. package/src/tool/batch.ts +175 -0
  284. package/src/tool/batch.txt +24 -0
  285. package/src/tool/codesearch.ts +184 -0
  286. package/src/tool/codesearch.txt +12 -0
  287. package/src/tool/edit.ts +675 -0
  288. package/src/tool/edit.txt +10 -0
  289. package/src/tool/glob.ts +65 -0
  290. package/src/tool/glob.txt +6 -0
  291. package/src/tool/grep.ts +121 -0
  292. package/src/tool/grep.txt +8 -0
  293. package/src/tool/invalid.ts +17 -0
  294. package/src/tool/ls.ts +110 -0
  295. package/src/tool/ls.txt +1 -0
  296. package/src/tool/lsp-diagnostics.ts +26 -0
  297. package/src/tool/lsp-diagnostics.txt +1 -0
  298. package/src/tool/lsp-hover.ts +31 -0
  299. package/src/tool/lsp-hover.txt +1 -0
  300. package/src/tool/lsp.ts +87 -0
  301. package/src/tool/lsp.txt +19 -0
  302. package/src/tool/multiedit.ts +46 -0
  303. package/src/tool/multiedit.txt +41 -0
  304. package/src/tool/patch.ts +233 -0
  305. package/src/tool/patch.txt +1 -0
  306. package/src/tool/read.ts +219 -0
  307. package/src/tool/read.txt +12 -0
  308. package/src/tool/registry.ts +162 -0
  309. package/src/tool/skill.ts +100 -0
  310. package/src/tool/task.ts +136 -0
  311. package/src/tool/task.txt +51 -0
  312. package/src/tool/todo.ts +39 -0
  313. package/src/tool/todoread.txt +14 -0
  314. package/src/tool/todowrite.txt +167 -0
  315. package/src/tool/tool.ts +71 -0
  316. package/src/tool/webfetch.ts +198 -0
  317. package/src/tool/webfetch.txt +13 -0
  318. package/src/tool/websearch.ts +268 -0
  319. package/src/tool/websearch.txt +13 -0
  320. package/src/tool/write.ts +110 -0
  321. package/src/tool/write.txt +8 -0
  322. package/src/util/archive.ts +16 -0
  323. package/src/util/color.ts +19 -0
  324. package/src/util/context.ts +25 -0
  325. package/src/util/defer.ts +12 -0
  326. package/src/util/eventloop.ts +20 -0
  327. package/src/util/filesystem.ts +83 -0
  328. package/src/util/fn.ts +11 -0
  329. package/src/util/iife.ts +3 -0
  330. package/src/util/keybind.ts +102 -0
  331. package/src/util/lazy.ts +11 -0
  332. package/src/util/license.ts +362 -0
  333. package/src/util/locale.ts +81 -0
  334. package/src/util/lock.ts +98 -0
  335. package/src/util/log.ts +180 -0
  336. package/src/util/queue.ts +32 -0
  337. package/src/util/rpc.ts +42 -0
  338. package/src/util/scrap.ts +10 -0
  339. package/src/util/signal.ts +12 -0
  340. package/src/util/timeout.ts +14 -0
  341. package/src/util/token.ts +7 -0
  342. package/src/util/wildcard.ts +54 -0
  343. package/sst-env.d.ts +9 -0
  344. package/test/agent/agent.test.ts +146 -0
  345. package/test/bun.test.ts +53 -0
  346. package/test/cli/cmd/acp.test.ts +144 -0
  347. package/test/cli/cmd/run.test.ts +250 -0
  348. package/test/cli/github-remote.test.ts +80 -0
  349. package/test/config/agent-color.test.ts +66 -0
  350. package/test/config/config.test.ts +536 -0
  351. package/test/config/markdown.test.ts +89 -0
  352. package/test/file/ignore.test.ts +10 -0
  353. package/test/fixture/fixture.ts +37 -0
  354. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  355. package/test/helpers.ts +172 -0
  356. package/test/ide/ide.test.ts +82 -0
  357. package/test/installation/installation.test.ts +143 -0
  358. package/test/keybind.test.ts +421 -0
  359. package/test/lsp/client.test.ts +95 -0
  360. package/test/mcp/headers.test.ts +153 -0
  361. package/test/patch/patch.test.ts +348 -0
  362. package/test/preload.ts +57 -0
  363. package/test/project/project.test.ts +74 -0
  364. package/test/provider/provider.test.ts +74 -0
  365. package/test/provider/transform.test.ts +411 -0
  366. package/test/session/retry.test.ts +111 -0
  367. package/test/session/session.test.ts +71 -0
  368. package/test/skill/skill.test.ts +131 -0
  369. package/test/snapshot/snapshot.test.ts +940 -0
  370. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  371. package/test/tool/bash.test.ts +434 -0
  372. package/test/tool/grep.test.ts +108 -0
  373. package/test/tool/patch.test.ts +259 -0
  374. package/test/tool/read.test.ts +42 -0
  375. package/test/util/iife.test.ts +36 -0
  376. package/test/util/lazy.test.ts +50 -0
  377. package/test/util/license.test.ts +235 -0
  378. package/test/util/timeout.test.ts +21 -0
  379. package/test/util/wildcard.test.ts +55 -0
  380. package/tsconfig.json +16 -0
  381. package/update-versions.ps1 +65 -0
@@ -0,0 +1,146 @@
1
+ import { RGBA } from "@opentui/core"
2
+
3
+ // Default terminal dimensions when detection fails or returns bogus values
4
+ const DEFAULT_WIDTH = 80
5
+ const DEFAULT_HEIGHT = 24
6
+
7
+ // Maximum reasonable terminal dimensions (beyond this is likely bogus)
8
+ const MAX_REASONABLE_WIDTH = 500
9
+ const MAX_REASONABLE_HEIGHT = 200
10
+
11
+ /**
12
+ * Sanitize terminal dimensions to handle edge cases where detection fails.
13
+ * Returns sensible defaults (80x24) when dimensions are bogus (e.g., 131072x1).
14
+ */
15
+ export function sanitizeDimensions(dims: { width: number; height: number }): { width: number; height: number } {
16
+ const width = dims.width
17
+ const height = dims.height
18
+
19
+ // Check for bogus dimensions: too small, too large, or zero
20
+ const isBogus =
21
+ width <= 0 ||
22
+ height <= 0 ||
23
+ width > MAX_REASONABLE_WIDTH ||
24
+ height > MAX_REASONABLE_HEIGHT ||
25
+ !Number.isFinite(width) ||
26
+ !Number.isFinite(height)
27
+
28
+ if (isBogus) {
29
+ return { width: DEFAULT_WIDTH, height: DEFAULT_HEIGHT }
30
+ }
31
+
32
+ return { width, height }
33
+ }
34
+
35
+ export namespace Terminal {
36
+ export type Colors = Awaited<ReturnType<typeof colors>>
37
+ /**
38
+ * Query terminal colors including background, foreground, and palette (0-15).
39
+ * Uses OSC escape sequences to retrieve actual terminal color values.
40
+ *
41
+ * Note: OSC 4 (palette) queries may not work through tmux as responses are filtered.
42
+ * OSC 10/11 (foreground/background) typically work in most environments.
43
+ *
44
+ * Returns an object with background, foreground, and colors array.
45
+ * Any query that fails will be null/empty.
46
+ */
47
+ export async function colors(): Promise<{
48
+ background: RGBA | null
49
+ foreground: RGBA | null
50
+ colors: RGBA[]
51
+ }> {
52
+ if (!process.stdin.isTTY) return { background: null, foreground: null, colors: [] }
53
+
54
+ return new Promise((resolve) => {
55
+ let background: RGBA | null = null
56
+ let foreground: RGBA | null = null
57
+ const paletteColors: RGBA[] = []
58
+ let timeout: NodeJS.Timeout
59
+
60
+ const cleanup = () => {
61
+ process.stdin.setRawMode(false)
62
+ process.stdin.removeListener("data", handler)
63
+ clearTimeout(timeout)
64
+ }
65
+
66
+ const parseColor = (colorStr: string): RGBA | null => {
67
+ if (colorStr.startsWith("rgb:")) {
68
+ const parts = colorStr.substring(4).split("/")
69
+ return RGBA.fromInts(
70
+ parseInt(parts[0], 16) >> 8, // Convert 16-bit to 8-bit
71
+ parseInt(parts[1], 16) >> 8,
72
+ parseInt(parts[2], 16) >> 8,
73
+ 255,
74
+ )
75
+ }
76
+ if (colorStr.startsWith("#")) {
77
+ return RGBA.fromHex(colorStr)
78
+ }
79
+ if (colorStr.startsWith("rgb(")) {
80
+ const parts = colorStr.substring(4, colorStr.length - 1).split(",")
81
+ return RGBA.fromInts(parseInt(parts[0]), parseInt(parts[1]), parseInt(parts[2]), 255)
82
+ }
83
+ return null
84
+ }
85
+
86
+ const handler = (data: Buffer) => {
87
+ const str = data.toString()
88
+
89
+ // Match OSC 11 (background color)
90
+ const bgMatch = str.match(/\x1b]11;([^\x07\x1b]+)/)
91
+ if (bgMatch) {
92
+ background = parseColor(bgMatch[1])
93
+ }
94
+
95
+ // Match OSC 10 (foreground color)
96
+ const fgMatch = str.match(/\x1b]10;([^\x07\x1b]+)/)
97
+ if (fgMatch) {
98
+ foreground = parseColor(fgMatch[1])
99
+ }
100
+
101
+ // Match OSC 4 (palette colors)
102
+ const paletteMatches = str.matchAll(/\x1b]4;(\d+);([^\x07\x1b]+)/g)
103
+ for (const match of paletteMatches) {
104
+ const index = parseInt(match[1])
105
+ const color = parseColor(match[2])
106
+ if (color) paletteColors[index] = color
107
+ }
108
+
109
+ // Return immediately if we have all 16 palette colors
110
+ if (paletteColors.filter((c) => c !== undefined).length === 16) {
111
+ cleanup()
112
+ resolve({ background, foreground, colors: paletteColors })
113
+ }
114
+ }
115
+
116
+ process.stdin.setRawMode(true)
117
+ process.stdin.on("data", handler)
118
+
119
+ // Query background (OSC 11)
120
+ process.stdout.write("\x1b]11;?\x07")
121
+ // Query foreground (OSC 10)
122
+ process.stdout.write("\x1b]10;?\x07")
123
+ // Query palette colors 0-15 (OSC 4)
124
+ for (let i = 0; i < 16; i++) {
125
+ process.stdout.write(`\x1b]4;${i};?\x07`)
126
+ }
127
+
128
+ timeout = setTimeout(() => {
129
+ cleanup()
130
+ resolve({ background, foreground, colors: paletteColors })
131
+ }, 1000)
132
+ })
133
+ }
134
+
135
+ export async function getTerminalBackgroundColor(): Promise<"dark" | "light"> {
136
+ const result = await colors()
137
+ if (!result.background) return "dark"
138
+
139
+ const { r, g, b } = result.background
140
+ // Calculate luminance using relative luminance formula
141
+ const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255
142
+
143
+ // Determine if dark or light based on luminance threshold
144
+ return luminance > 0.5 ? "light" : "dark"
145
+ }
146
+ }
@@ -0,0 +1,63 @@
1
+ import { Installation } from "@/installation"
2
+ import { Server } from "@/server/server"
3
+ import { Log } from "@/util/log"
4
+ import { Instance } from "@/project/instance"
5
+ import { InstanceBootstrap } from "@/project/bootstrap"
6
+ import { Rpc } from "@/util/rpc"
7
+ import { upgrade } from "@/cli/upgrade"
8
+ import type { BunWebSocketData } from "hono/bun"
9
+
10
+ await Log.init({
11
+ print: process.argv.includes("--print-logs"),
12
+ dev: Installation.isLocal(),
13
+ level: (() => {
14
+ if (Installation.isLocal()) return "DEBUG"
15
+ return "INFO"
16
+ })(),
17
+ })
18
+
19
+ process.on("unhandledRejection", (e) => {
20
+ Log.Default.error("rejection", {
21
+ e: e instanceof Error ? e.message : e,
22
+ })
23
+ })
24
+
25
+ process.on("uncaughtException", (e) => {
26
+ Log.Default.error("exception", {
27
+ e: e instanceof Error ? e.message : e,
28
+ })
29
+ })
30
+
31
+ let server: Bun.Server<BunWebSocketData>
32
+ export const rpc = {
33
+ async server(input: { port: number; hostname: string }) {
34
+ if (server) await server.stop(true)
35
+ try {
36
+ server = Server.listen(input)
37
+ return {
38
+ url: server.url.toString(),
39
+ }
40
+ } catch (e) {
41
+ console.error(e)
42
+ throw e
43
+ }
44
+ },
45
+ async checkUpgrade(input: { directory: string }) {
46
+ await Instance.provide({
47
+ directory: input.directory,
48
+ init: InstanceBootstrap,
49
+ fn: async () => {
50
+ await upgrade().catch(() => {})
51
+ },
52
+ })
53
+ },
54
+ async shutdown() {
55
+ Log.Default.info("worker shutting down")
56
+ await Instance.disposeAll()
57
+ // TODO: this should be awaited, but ws connections are
58
+ // causing this to hang, need to revisit this
59
+ server.stop(true)
60
+ },
61
+ }
62
+
63
+ Rpc.listen(rpc)
@@ -0,0 +1,344 @@
1
+ import type { Argv } from "yargs"
2
+ import { UI } from "../ui"
3
+ import * as prompts from "@clack/prompts"
4
+ import { Installation } from "../../installation"
5
+ import { Global } from "../../global"
6
+ import { $ } from "bun"
7
+ import fs from "fs/promises"
8
+ import path from "path"
9
+ import os from "os"
10
+
11
+ interface UninstallArgs {
12
+ keepConfig: boolean
13
+ keepData: boolean
14
+ dryRun: boolean
15
+ force: boolean
16
+ }
17
+
18
+ interface RemovalTargets {
19
+ directories: Array<{ path: string; label: string; keep: boolean }>
20
+ shellConfig: string | null
21
+ binary: string | null
22
+ }
23
+
24
+ export const UninstallCommand = {
25
+ command: "uninstall",
26
+ describe: "uninstall rird and remove all related files",
27
+ builder: (yargs: Argv) =>
28
+ yargs
29
+ .option("keep-config", {
30
+ alias: "c",
31
+ type: "boolean",
32
+ describe: "keep configuration files",
33
+ default: false,
34
+ })
35
+ .option("keep-data", {
36
+ alias: "d",
37
+ type: "boolean",
38
+ describe: "keep session data and snapshots",
39
+ default: false,
40
+ })
41
+ .option("dry-run", {
42
+ type: "boolean",
43
+ describe: "show what would be removed without removing",
44
+ default: false,
45
+ })
46
+ .option("force", {
47
+ alias: "f",
48
+ type: "boolean",
49
+ describe: "skip confirmation prompts",
50
+ default: false,
51
+ }),
52
+
53
+ handler: async (args: UninstallArgs) => {
54
+ UI.empty()
55
+ UI.println(UI.logo(" "))
56
+ UI.empty()
57
+ prompts.intro("Uninstall RIRD")
58
+
59
+ const method = await Installation.method()
60
+ prompts.log.info(`Installation method: ${method}`)
61
+
62
+ const targets = await collectRemovalTargets(args, method)
63
+
64
+ await showRemovalSummary(targets, method)
65
+
66
+ if (!args.force && !args.dryRun) {
67
+ const confirm = await prompts.confirm({
68
+ message: "Are you sure you want to uninstall?",
69
+ initialValue: false,
70
+ })
71
+ if (!confirm || prompts.isCancel(confirm)) {
72
+ prompts.outro("Cancelled")
73
+ return
74
+ }
75
+ }
76
+
77
+ if (args.dryRun) {
78
+ prompts.log.warn("Dry run - no changes made")
79
+ prompts.outro("Done")
80
+ return
81
+ }
82
+
83
+ await executeUninstall(method, targets)
84
+
85
+ prompts.outro("Done")
86
+ },
87
+ }
88
+
89
+ async function collectRemovalTargets(args: UninstallArgs, method: Installation.Method): Promise<RemovalTargets> {
90
+ const directories: RemovalTargets["directories"] = [
91
+ { path: Global.Path.data, label: "Data", keep: args.keepData },
92
+ { path: Global.Path.cache, label: "Cache", keep: false },
93
+ { path: Global.Path.config, label: "Config", keep: args.keepConfig },
94
+ { path: Global.Path.state, label: "State", keep: false },
95
+ ]
96
+
97
+ const shellConfig = method === "curl" ? await getShellConfigFile() : null
98
+ const binary = method === "curl" ? process.execPath : null
99
+
100
+ return { directories, shellConfig, binary }
101
+ }
102
+
103
+ async function showRemovalSummary(targets: RemovalTargets, method: Installation.Method) {
104
+ prompts.log.message("The following will be removed:")
105
+
106
+ for (const dir of targets.directories) {
107
+ const exists = await fs
108
+ .access(dir.path)
109
+ .then(() => true)
110
+ .catch(() => false)
111
+ if (!exists) continue
112
+
113
+ const size = await getDirectorySize(dir.path)
114
+ const sizeStr = formatSize(size)
115
+ const status = dir.keep ? UI.Style.TEXT_DIM + "(keeping)" : ""
116
+ const prefix = dir.keep ? "○" : "✓"
117
+
118
+ prompts.log.info(` ${prefix} ${dir.label}: ${shortenPath(dir.path)} ${UI.Style.TEXT_DIM}(${sizeStr})${status}`)
119
+ }
120
+
121
+ if (targets.binary) {
122
+ prompts.log.info(` ✓ Binary: ${shortenPath(targets.binary)}`)
123
+ }
124
+
125
+ if (targets.shellConfig) {
126
+ prompts.log.info(` ✓ Shell PATH in ${shortenPath(targets.shellConfig)}`)
127
+ }
128
+
129
+ if (method !== "curl" && method !== "unknown") {
130
+ const cmds: Record<string, string> = {
131
+ npm: "npm uninstall -g rird-ai",
132
+ pnpm: "pnpm uninstall -g rird-ai",
133
+ bun: "bun remove -g rird-ai",
134
+ yarn: "yarn global remove rird-ai",
135
+ brew: "brew uninstall rird",
136
+ }
137
+ prompts.log.info(` ✓ Package: ${cmds[method] || method}`)
138
+ }
139
+ }
140
+
141
+ async function executeUninstall(method: Installation.Method, targets: RemovalTargets) {
142
+ const spinner = prompts.spinner()
143
+ const errors: string[] = []
144
+
145
+ for (const dir of targets.directories) {
146
+ if (dir.keep) {
147
+ prompts.log.step(`Skipping ${dir.label} (--keep-${dir.label.toLowerCase()})`)
148
+ continue
149
+ }
150
+
151
+ const exists = await fs
152
+ .access(dir.path)
153
+ .then(() => true)
154
+ .catch(() => false)
155
+ if (!exists) continue
156
+
157
+ spinner.start(`Removing ${dir.label}...`)
158
+ const err = await fs.rm(dir.path, { recursive: true, force: true }).catch((e) => e)
159
+ if (err) {
160
+ spinner.stop(`Failed to remove ${dir.label}`, 1)
161
+ errors.push(`${dir.label}: ${err.message}`)
162
+ continue
163
+ }
164
+ spinner.stop(`Removed ${dir.label}`)
165
+ }
166
+
167
+ if (targets.shellConfig) {
168
+ spinner.start("Cleaning shell config...")
169
+ const err = await cleanShellConfig(targets.shellConfig).catch((e) => e)
170
+ if (err) {
171
+ spinner.stop("Failed to clean shell config", 1)
172
+ errors.push(`Shell config: ${err.message}`)
173
+ } else {
174
+ spinner.stop("Cleaned shell config")
175
+ }
176
+ }
177
+
178
+ if (method !== "curl" && method !== "unknown") {
179
+ const cmds: Record<string, string[]> = {
180
+ npm: ["npm", "uninstall", "-g", "rird-ai"],
181
+ pnpm: ["pnpm", "uninstall", "-g", "rird-ai"],
182
+ bun: ["bun", "remove", "-g", "rird-ai"],
183
+ yarn: ["yarn", "global", "remove", "rird-ai"],
184
+ brew: ["brew", "uninstall", "rird"],
185
+ }
186
+
187
+ const cmd = cmds[method]
188
+ if (cmd) {
189
+ spinner.start(`Running ${cmd.join(" ")}...`)
190
+ const result = await $`${cmd}`.quiet().nothrow()
191
+ if (result.exitCode !== 0) {
192
+ spinner.stop(`Package manager uninstall failed`, 1)
193
+ prompts.log.warn(`You may need to run manually: ${cmd.join(" ")}`)
194
+ errors.push(`Package manager: exit code ${result.exitCode}`)
195
+ } else {
196
+ spinner.stop("Package removed")
197
+ }
198
+ }
199
+ }
200
+
201
+ if (method === "curl" && targets.binary) {
202
+ UI.empty()
203
+ prompts.log.message("To finish removing the binary, run:")
204
+ prompts.log.info(` rm "${targets.binary}"`)
205
+
206
+ const binDir = path.dirname(targets.binary)
207
+ if (binDir.includes(".RIRD")) {
208
+ prompts.log.info(` rmdir "${binDir}" 2>/dev/null`)
209
+ }
210
+ }
211
+
212
+ if (errors.length > 0) {
213
+ UI.empty()
214
+ prompts.log.warn("Some operations failed:")
215
+ for (const err of errors) {
216
+ prompts.log.error(` ${err}`)
217
+ }
218
+ }
219
+
220
+ UI.empty()
221
+ prompts.log.success("Thank you for using RIRD!")
222
+ }
223
+
224
+ async function getShellConfigFile(): Promise<string | null> {
225
+ const shell = path.basename(process.env.SHELL || "bash")
226
+ const home = os.homedir()
227
+ const xdgConfig = process.env.XDG_CONFIG_HOME || path.join(home, ".config")
228
+
229
+ const configFiles: Record<string, string[]> = {
230
+ fish: [path.join(xdgConfig, "fish", "config.fish")],
231
+ zsh: [
232
+ path.join(home, ".zshrc"),
233
+ path.join(home, ".zshenv"),
234
+ path.join(xdgConfig, "zsh", ".zshrc"),
235
+ path.join(xdgConfig, "zsh", ".zshenv"),
236
+ ],
237
+ bash: [
238
+ path.join(home, ".bashrc"),
239
+ path.join(home, ".bash_profile"),
240
+ path.join(home, ".profile"),
241
+ path.join(xdgConfig, "bash", ".bashrc"),
242
+ path.join(xdgConfig, "bash", ".bash_profile"),
243
+ ],
244
+ ash: [path.join(home, ".ashrc"), path.join(home, ".profile")],
245
+ sh: [path.join(home, ".profile")],
246
+ }
247
+
248
+ const candidates = configFiles[shell] || configFiles.bash
249
+
250
+ for (const file of candidates) {
251
+ const exists = await fs
252
+ .access(file)
253
+ .then(() => true)
254
+ .catch(() => false)
255
+ if (!exists) continue
256
+
257
+ const content = await Bun.file(file)
258
+ .text()
259
+ .catch(() => "")
260
+ if (content.includes("# RIRD") || content.includes(".RIRD/bin")) {
261
+ return file
262
+ }
263
+ }
264
+
265
+ return null
266
+ }
267
+
268
+ async function cleanShellConfig(file: string) {
269
+ const content = await Bun.file(file).text()
270
+ const lines = content.split("\n")
271
+
272
+ const filtered: string[] = []
273
+ let skip = false
274
+
275
+ for (const line of lines) {
276
+ const trimmed = line.trim()
277
+
278
+ if (trimmed === "# RIRD") {
279
+ skip = true
280
+ continue
281
+ }
282
+
283
+ if (skip) {
284
+ skip = false
285
+ if (trimmed.includes(".RIRD/bin") || trimmed.includes("fish_add_path")) {
286
+ continue
287
+ }
288
+ }
289
+
290
+ if (
291
+ (trimmed.startsWith("export PATH=") && trimmed.includes(".RIRD/bin")) ||
292
+ (trimmed.startsWith("fish_add_path") && trimmed.includes(".RIRD"))
293
+ ) {
294
+ continue
295
+ }
296
+
297
+ filtered.push(line)
298
+ }
299
+
300
+ while (filtered.length > 0 && filtered[filtered.length - 1].trim() === "") {
301
+ filtered.pop()
302
+ }
303
+
304
+ const output = filtered.join("\n") + "\n"
305
+ await Bun.write(file, output)
306
+ }
307
+
308
+ async function getDirectorySize(dir: string): Promise<number> {
309
+ let total = 0
310
+
311
+ const walk = async (current: string) => {
312
+ const entries = await fs.readdir(current, { withFileTypes: true }).catch(() => [])
313
+
314
+ for (const entry of entries) {
315
+ const full = path.join(current, entry.name)
316
+ if (entry.isDirectory()) {
317
+ await walk(full)
318
+ continue
319
+ }
320
+ if (entry.isFile()) {
321
+ const stat = await fs.stat(full).catch(() => null)
322
+ if (stat) total += stat.size
323
+ }
324
+ }
325
+ }
326
+
327
+ await walk(dir)
328
+ return total
329
+ }
330
+
331
+ function formatSize(bytes: number): string {
332
+ if (bytes < 1024) return `${bytes} B`
333
+ if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`
334
+ if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} MB`
335
+ return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)} GB`
336
+ }
337
+
338
+ function shortenPath(p: string): string {
339
+ const home = os.homedir()
340
+ if (p.startsWith(home)) {
341
+ return p.replace(home, "~")
342
+ }
343
+ return p
344
+ }
@@ -0,0 +1,127 @@
1
+ import type { Argv } from "yargs"
2
+ import { UI } from "../ui"
3
+ import * as prompts from "@clack/prompts"
4
+ import { Installation } from "../../installation"
5
+ import { Log } from "../../util/log"
6
+ import { InstallationError } from "../../lib/errors"
7
+ import { handleError } from "../../lib/error-handler"
8
+
9
+ const log = Log.create({ service: "upgrade-command" })
10
+
11
+ export const UpgradeCommand = {
12
+ command: "upgrade [target]",
13
+ aliases: ["update"],
14
+ describe: "upgrade rird to the latest or a specific version",
15
+ builder: (yargs: Argv) => {
16
+ return yargs
17
+ .positional("target", {
18
+ describe: "version to upgrade to, for ex '0.1.48' or 'v0.1.48'",
19
+ type: "string",
20
+ })
21
+ .option("method", {
22
+ alias: "m",
23
+ describe: "installation method to use",
24
+ type: "string",
25
+ choices: ["curl", "npm", "pnpm", "bun", "brew"],
26
+ })
27
+ },
28
+ handler: async (args: { target?: string; method?: string }) => {
29
+ try {
30
+ UI.empty()
31
+ UI.println(UI.logo(" "))
32
+ UI.empty()
33
+ prompts.intro("Upgrade")
34
+ const detectedMethod = await Installation.method()
35
+ const method = (args.method as Installation.Method) ?? detectedMethod
36
+ if (method === "unknown") {
37
+ prompts.log.warn(`rird is installed to ${process.execPath} and may be managed by a package manager`)
38
+ const install = await prompts.select({
39
+ message: "Install anyways?",
40
+ options: [
41
+ { label: "Yes", value: true },
42
+ { label: "No", value: false },
43
+ ],
44
+ initialValue: true,
45
+ })
46
+ if (!install) {
47
+ prompts.outro("Done")
48
+ return
49
+ }
50
+ }
51
+ prompts.log.info("Using method: " + method)
52
+
53
+ let target: string
54
+ try {
55
+ target = args.target ? args.target.replace(/^v/, "") : await Installation.latest()
56
+ } catch (latestErr) {
57
+ log.warn("Could not fetch latest version from registry", {
58
+ error: (latestErr as Error).message,
59
+ })
60
+ prompts.log.warn("Could not fetch latest version from registry")
61
+ prompts.log.info("Attempting force reinstall to fix broken installation...")
62
+ const spinner = prompts.spinner()
63
+ spinner.start("Reinstalling...")
64
+ const err = await Installation.forceReinstall().catch((err) => err)
65
+ if (err) {
66
+ spinner.stop("Reinstall failed", 1)
67
+ log.error("Force reinstall failed", {
68
+ error: err instanceof Error ? err.message : String(err),
69
+ })
70
+ if (err instanceof Installation.UpgradeFailedError) prompts.log.error(err.data.stderr)
71
+ else if (err instanceof Error) prompts.log.error(err.message)
72
+ prompts.log.info("Manual fix: npm install -g rird-ai@latest")
73
+ } else {
74
+ spinner.stop("Reinstall complete")
75
+ log.info("Force reinstall completed successfully")
76
+ }
77
+ prompts.outro("Done")
78
+ return
79
+ }
80
+
81
+ if (Installation.VERSION === target) {
82
+ prompts.log.warn(`rird upgrade skipped: ${target} is already installed`)
83
+ prompts.outro("Done")
84
+ return
85
+ }
86
+
87
+ prompts.log.info(`From ${Installation.VERSION} -> ${target}`)
88
+ const spinner = prompts.spinner()
89
+ spinner.start("Upgrading...")
90
+ let err = await Installation.upgrade(method, target).catch((err) => err)
91
+
92
+ if (err) {
93
+ log.error("Standard upgrade failed", {
94
+ method,
95
+ target,
96
+ error: err instanceof Error ? err.message : String(err),
97
+ })
98
+ spinner.stop("Standard upgrade failed, trying force reinstall...", 1)
99
+ const forceSpinner = prompts.spinner()
100
+ forceSpinner.start("Reinstalling...")
101
+ err = await Installation.forceReinstall().catch((err) => err)
102
+ if (err) {
103
+ log.error("Force reinstall also failed", {
104
+ error: err instanceof Error ? err.message : String(err),
105
+ })
106
+ forceSpinner.stop("Reinstall failed", 1)
107
+ if (err instanceof Installation.UpgradeFailedError) prompts.log.error(err.data.stderr)
108
+ else if (err instanceof Error) prompts.log.error(err.message)
109
+ prompts.log.info("Manual fix: npm install -g rird-ai@latest")
110
+ throw new InstallationError("Upgrade and reinstall both failed", {
111
+ code: "UPGRADE_FAILED",
112
+ originalError: err as Error,
113
+ details: { method, target },
114
+ })
115
+ }
116
+ log.info("Force reinstall completed after initial upgrade failure")
117
+ forceSpinner.stop("Reinstall complete")
118
+ } else {
119
+ log.info("Upgrade completed successfully", { target })
120
+ spinner.stop("Upgrade complete")
121
+ }
122
+ prompts.outro("Done")
123
+ } catch (error) {
124
+ await handleError(error, "upgrade-command", { exitOnError: true })
125
+ }
126
+ },
127
+ }