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,256 @@
1
+ import { cmd } from "./cmd"
2
+ import * as prompts from "@clack/prompts"
3
+ import { UI } from "../ui"
4
+ import { Global } from "../../global"
5
+ import { Agent } from "../../agent/agent"
6
+ import { Provider } from "../../provider/provider"
7
+ import path from "path"
8
+ import fs from "fs/promises"
9
+ import matter from "gray-matter"
10
+ import { Instance } from "../../project/instance"
11
+ import { EOL } from "os"
12
+ import type { Argv } from "yargs"
13
+
14
+ type AgentMode = "all" | "primary" | "subagent"
15
+
16
+ const AVAILABLE_TOOLS = [
17
+ "bash",
18
+ "read",
19
+ "write",
20
+ "edit",
21
+ "list",
22
+ "glob",
23
+ "grep",
24
+ "webfetch",
25
+ "task",
26
+ "todowrite",
27
+ "todoread",
28
+ ]
29
+
30
+ const AgentCreateCommand = cmd({
31
+ command: "create",
32
+ describe: "create a new agent",
33
+ builder: (yargs: Argv) =>
34
+ yargs
35
+ .option("path", {
36
+ type: "string",
37
+ describe: "directory path to generate the agent file",
38
+ })
39
+ .option("description", {
40
+ type: "string",
41
+ describe: "what the agent should do",
42
+ })
43
+ .option("mode", {
44
+ type: "string",
45
+ describe: "agent mode",
46
+ choices: ["all", "primary", "subagent"] as const,
47
+ })
48
+ .option("tools", {
49
+ type: "string",
50
+ describe: `comma-separated list of tools to enable (default: all). Available: "${AVAILABLE_TOOLS.join(", ")}"`,
51
+ })
52
+ .option("model", {
53
+ type: "string",
54
+ alias: ["m"],
55
+ describe: "model to use in the format of provider/model",
56
+ }),
57
+ async handler(args) {
58
+ await Instance.provide({
59
+ directory: process.cwd(),
60
+ async fn() {
61
+ const cliPath = args.path
62
+ const cliDescription = args.description
63
+ const cliMode = args.mode as AgentMode | undefined
64
+ const cliTools = args.tools
65
+
66
+ const isFullyNonInteractive = cliPath && cliDescription && cliMode && cliTools !== undefined
67
+
68
+ if (!isFullyNonInteractive) {
69
+ UI.empty()
70
+ prompts.intro("Create agent")
71
+ }
72
+
73
+ const project = Instance.project
74
+
75
+ // Determine scope/path
76
+ let targetPath: string
77
+ if (cliPath) {
78
+ targetPath = path.join(cliPath, "agent")
79
+ } else {
80
+ let scope: "global" | "project" = "global"
81
+ if (project.vcs === "git") {
82
+ const scopeResult = await prompts.select({
83
+ message: "Location",
84
+ options: [
85
+ {
86
+ label: "Current project",
87
+ value: "project" as const,
88
+ hint: Instance.worktree,
89
+ },
90
+ {
91
+ label: "Global",
92
+ value: "global" as const,
93
+ hint: Global.Path.config,
94
+ },
95
+ ],
96
+ })
97
+ if (prompts.isCancel(scopeResult)) throw new UI.CancelledError()
98
+ scope = scopeResult
99
+ }
100
+ targetPath = path.join(
101
+ scope === "global" ? Global.Path.config : path.join(Instance.worktree, ".RIRD"),
102
+ "agent",
103
+ )
104
+ }
105
+
106
+ // Get description
107
+ let description: string
108
+ if (cliDescription) {
109
+ description = cliDescription
110
+ } else {
111
+ const query = await prompts.text({
112
+ message: "Description",
113
+ placeholder: "What should this agent do?",
114
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
115
+ })
116
+ if (prompts.isCancel(query)) throw new UI.CancelledError()
117
+ description = query
118
+ }
119
+
120
+ // Generate agent
121
+ const spinner = prompts.spinner()
122
+ spinner.start("Generating agent configuration...")
123
+ const model = args.model ? Provider.parseModel(args.model) : undefined
124
+ const generated = await Agent.generate({ description, model }).catch((error) => {
125
+ spinner.stop(`LLM failed to generate agent: ${error.message}`, 1)
126
+ if (isFullyNonInteractive) process.exit(1)
127
+ throw new UI.CancelledError()
128
+ })
129
+ spinner.stop(`Agent ${generated.identifier} generated`)
130
+
131
+ // Select tools
132
+ let selectedTools: string[]
133
+ if (cliTools !== undefined) {
134
+ selectedTools = cliTools ? cliTools.split(",").map((t) => t.trim()) : AVAILABLE_TOOLS
135
+ } else {
136
+ const result = await prompts.multiselect({
137
+ message: "Select tools to enable",
138
+ options: AVAILABLE_TOOLS.map((tool) => ({
139
+ label: tool,
140
+ value: tool,
141
+ })),
142
+ initialValues: AVAILABLE_TOOLS,
143
+ })
144
+ if (prompts.isCancel(result)) throw new UI.CancelledError()
145
+ selectedTools = result
146
+ }
147
+
148
+ // Get mode
149
+ let mode: AgentMode
150
+ if (cliMode) {
151
+ mode = cliMode
152
+ } else {
153
+ const modeResult = await prompts.select({
154
+ message: "Agent mode",
155
+ options: [
156
+ {
157
+ label: "All",
158
+ value: "all" as const,
159
+ hint: "Can function in both primary and subagent roles",
160
+ },
161
+ {
162
+ label: "Primary",
163
+ value: "primary" as const,
164
+ hint: "Acts as a primary/main agent",
165
+ },
166
+ {
167
+ label: "Subagent",
168
+ value: "subagent" as const,
169
+ hint: "Can be used as a subagent by other agents",
170
+ },
171
+ ],
172
+ initialValue: "all" as const,
173
+ })
174
+ if (prompts.isCancel(modeResult)) throw new UI.CancelledError()
175
+ mode = modeResult
176
+ }
177
+
178
+ // Build tools config
179
+ const tools: Record<string, boolean> = {}
180
+ for (const tool of AVAILABLE_TOOLS) {
181
+ if (!selectedTools.includes(tool)) {
182
+ tools[tool] = false
183
+ }
184
+ }
185
+
186
+ // Build frontmatter
187
+ const frontmatter: {
188
+ description: string
189
+ mode: AgentMode
190
+ tools?: Record<string, boolean>
191
+ } = {
192
+ description: generated.whenToUse,
193
+ mode,
194
+ }
195
+ if (Object.keys(tools).length > 0) {
196
+ frontmatter.tools = tools
197
+ }
198
+
199
+ // Write file
200
+ const content = matter.stringify(generated.systemPrompt, frontmatter)
201
+ const filePath = path.join(targetPath, `${generated.identifier}.md`)
202
+
203
+ await fs.mkdir(targetPath, { recursive: true })
204
+
205
+ const file = Bun.file(filePath)
206
+ if (await file.exists()) {
207
+ if (isFullyNonInteractive) {
208
+ console.error(`Error: Agent file already exists: ${filePath}`)
209
+ process.exit(1)
210
+ }
211
+ prompts.log.error(`Agent file already exists: ${filePath}`)
212
+ throw new UI.CancelledError()
213
+ }
214
+
215
+ await Bun.write(filePath, content)
216
+
217
+ if (isFullyNonInteractive) {
218
+ console.log(filePath)
219
+ } else {
220
+ prompts.log.success(`Agent created: ${filePath}`)
221
+ prompts.outro("Done")
222
+ }
223
+ },
224
+ })
225
+ },
226
+ })
227
+
228
+ const AgentListCommand = cmd({
229
+ command: "list",
230
+ describe: "list all available agents",
231
+ async handler() {
232
+ await Instance.provide({
233
+ directory: process.cwd(),
234
+ async fn() {
235
+ const agents = await Agent.list()
236
+ const sortedAgents = agents.sort((a, b) => {
237
+ if (a.native !== b.native) {
238
+ return a.native ? -1 : 1
239
+ }
240
+ return a.name.localeCompare(b.name)
241
+ })
242
+
243
+ for (const agent of sortedAgents) {
244
+ process.stdout.write(`${agent.name} (${agent.mode})${EOL}`)
245
+ }
246
+ },
247
+ })
248
+ },
249
+ })
250
+
251
+ export const AgentCommand = cmd({
252
+ command: "agent",
253
+ describe: "manage agents",
254
+ builder: (yargs) => yargs.command(AgentCreateCommand).command(AgentListCommand).demandCommand(),
255
+ async handler() {},
256
+ })
@@ -0,0 +1,412 @@
1
+ import { Auth } from "../../auth"
2
+ import { cmd } from "./cmd"
3
+ import * as prompts from "@clack/prompts"
4
+ import { UI } from "../ui"
5
+ import { ModelsDev } from "../../provider/models"
6
+ import { map, pipe, sortBy, values } from "remeda"
7
+ import path from "path"
8
+ import os from "os"
9
+ import { Config } from "../../config/config"
10
+ import { Global } from "../../global"
11
+ import { Plugin } from "../../plugin"
12
+ import { Instance } from "../../project/instance"
13
+ import type { Hooks } from "@opencode-ai/plugin"
14
+
15
+ // Map ALL provider names to RIRD-branded names
16
+ // Whitelist known providers, map everything else to RIRD
17
+ const KNOWN_PROVIDERS = new Set([
18
+ "anthropic", "openai", "google", "gemini", "mistral", "azure",
19
+ "amazon", "bedrock", "aws", "vertex", "github", "copilot",
20
+ "openrouter", "groq", "together", "fireworks", "replicate",
21
+ "cohere", "ai21", "deepseek", "perplexity", "xai", "grok"
22
+ ])
23
+
24
+ function getRirdProviderName(providerName: string): string {
25
+ if (!providerName) return "RIRD"
26
+ const lower = providerName.toLowerCase()
27
+ for (const known of KNOWN_PROVIDERS) {
28
+ if (lower.includes(known)) return providerName
29
+ }
30
+ return "RIRD"
31
+ }
32
+
33
+ type PluginAuth = NonNullable<Hooks["auth"]>
34
+
35
+ /**
36
+ * Handle plugin-based authentication flow.
37
+ * Returns true if auth was handled, false if it should fall through to default handling.
38
+ */
39
+ async function handlePluginAuth(plugin: { auth: PluginAuth }, provider: string): Promise<boolean> {
40
+ let index = 0
41
+ if (plugin.auth.methods.length > 1) {
42
+ const method = await prompts.select({
43
+ message: "Login method",
44
+ options: [
45
+ ...plugin.auth.methods.map((x, index) => ({
46
+ label: x.label,
47
+ value: index.toString(),
48
+ })),
49
+ ],
50
+ })
51
+ if (prompts.isCancel(method)) throw new UI.CancelledError()
52
+ index = parseInt(method)
53
+ }
54
+ const method = plugin.auth.methods[index]
55
+
56
+ // Handle prompts for all auth types
57
+ await new Promise((resolve) => setTimeout(resolve, 10))
58
+ const inputs: Record<string, string> = {}
59
+ if (method.prompts) {
60
+ for (const prompt of method.prompts) {
61
+ if (prompt.condition && !prompt.condition(inputs)) {
62
+ continue
63
+ }
64
+ if (prompt.type === "select") {
65
+ const value = await prompts.select({
66
+ message: prompt.message,
67
+ options: prompt.options,
68
+ })
69
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
70
+ inputs[prompt.key] = value
71
+ } else {
72
+ const value = await prompts.text({
73
+ message: prompt.message,
74
+ placeholder: prompt.placeholder,
75
+ validate: prompt.validate ? (v) => prompt.validate!(v ?? "") : undefined,
76
+ })
77
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
78
+ inputs[prompt.key] = value
79
+ }
80
+ }
81
+ }
82
+
83
+ if (method.type === "oauth") {
84
+ const authorize = await method.authorize(inputs)
85
+
86
+ if (authorize.url) {
87
+ prompts.log.info("Go to: " + authorize.url)
88
+ }
89
+
90
+ if (authorize.method === "auto") {
91
+ if (authorize.instructions) {
92
+ prompts.log.info(authorize.instructions)
93
+ }
94
+ const spinner = prompts.spinner()
95
+ spinner.start("Waiting for authorization...")
96
+ const result = await authorize.callback()
97
+ if (result.type === "failed") {
98
+ spinner.stop("Failed to authorize", 1)
99
+ }
100
+ if (result.type === "success") {
101
+ const saveProvider = result.provider ?? provider
102
+ if ("refresh" in result) {
103
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
104
+ await Auth.set(saveProvider, {
105
+ type: "oauth",
106
+ refresh,
107
+ access,
108
+ expires,
109
+ ...extraFields,
110
+ })
111
+ }
112
+ if ("key" in result) {
113
+ await Auth.set(saveProvider, {
114
+ type: "api",
115
+ key: result.key,
116
+ })
117
+ }
118
+ spinner.stop("Login successful")
119
+ }
120
+ }
121
+
122
+ if (authorize.method === "code") {
123
+ const code = await prompts.text({
124
+ message: "Paste the authorization code here: ",
125
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
126
+ })
127
+ if (prompts.isCancel(code)) throw new UI.CancelledError()
128
+ const result = await authorize.callback(code)
129
+ if (result.type === "failed") {
130
+ prompts.log.error("Failed to authorize")
131
+ }
132
+ if (result.type === "success") {
133
+ const saveProvider = result.provider ?? provider
134
+ if ("refresh" in result) {
135
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
136
+ await Auth.set(saveProvider, {
137
+ type: "oauth",
138
+ refresh,
139
+ access,
140
+ expires,
141
+ ...extraFields,
142
+ })
143
+ }
144
+ if ("key" in result) {
145
+ await Auth.set(saveProvider, {
146
+ type: "api",
147
+ key: result.key,
148
+ })
149
+ }
150
+ prompts.log.success("Login successful")
151
+ }
152
+ }
153
+
154
+ prompts.outro("Done")
155
+ return true
156
+ }
157
+
158
+ if (method.type === "api") {
159
+ if (method.authorize) {
160
+ const result = await method.authorize(inputs)
161
+ if (result.type === "failed") {
162
+ prompts.log.error("Failed to authorize")
163
+ }
164
+ if (result.type === "success") {
165
+ const saveProvider = result.provider ?? provider
166
+ await Auth.set(saveProvider, {
167
+ type: "api",
168
+ key: result.key,
169
+ })
170
+ prompts.log.success("Login successful")
171
+ }
172
+ prompts.outro("Done")
173
+ return true
174
+ }
175
+ }
176
+
177
+ return false
178
+ }
179
+
180
+ export const AuthCommand = cmd({
181
+ command: "auth",
182
+ describe: "manage credentials",
183
+ builder: (yargs) => yargs,
184
+ async handler() {
185
+ UI.empty()
186
+ UI.println("RIRD uses your license key for authentication.")
187
+ UI.println("Run: rird activate YOUR_LICENSE_KEY")
188
+ },
189
+ })
190
+
191
+ export const AuthListCommand = cmd({
192
+ command: "list",
193
+ aliases: ["ls"],
194
+ describe: "list providers",
195
+ async handler() {
196
+ UI.empty()
197
+ const authPath = path.join(Global.Path.data, "auth.json")
198
+ const homedir = os.homedir()
199
+ const displayPath = authPath.startsWith(homedir) ? authPath.replace(homedir, "~") : authPath
200
+ prompts.intro(`Credentials ${UI.Style.TEXT_DIM}${displayPath}`)
201
+ const results = Object.entries(await Auth.all())
202
+ const database = await ModelsDev.get()
203
+
204
+ for (const [providerID, result] of results) {
205
+ const name = database[providerID]?.name || providerID
206
+ prompts.log.info(`${name} ${UI.Style.TEXT_DIM}${result.type}`)
207
+ }
208
+
209
+ prompts.outro(`${results.length} credentials`)
210
+
211
+ // Environment variables section
212
+ const activeEnvVars: Array<{ provider: string; envVar: string }> = []
213
+
214
+ for (const [providerID, provider] of Object.entries(database)) {
215
+ for (const envVar of provider.env) {
216
+ if (process.env[envVar]) {
217
+ activeEnvVars.push({
218
+ provider: getRirdProviderName(provider.name || providerID),
219
+ envVar,
220
+ })
221
+ }
222
+ }
223
+ }
224
+
225
+ if (activeEnvVars.length > 0) {
226
+ UI.empty()
227
+ prompts.intro("Environment")
228
+
229
+ for (const { provider, envVar } of activeEnvVars) {
230
+ prompts.log.info(`${provider} ${UI.Style.TEXT_DIM}${envVar}`)
231
+ }
232
+
233
+ prompts.outro(`${activeEnvVars.length} environment variable` + (activeEnvVars.length === 1 ? "" : "s"))
234
+ }
235
+ },
236
+ })
237
+
238
+ export const AuthLoginCommand = cmd({
239
+ command: "login [url]",
240
+ describe: "log in to a provider",
241
+ builder: (yargs) =>
242
+ yargs.positional("url", {
243
+ describe: "RIRD auth provider",
244
+ type: "string",
245
+ }),
246
+ async handler(args) {
247
+ await Instance.provide({
248
+ directory: process.cwd(),
249
+ async fn() {
250
+ UI.empty()
251
+ prompts.intro("Add credential")
252
+ if (args.url) {
253
+ const wellknown = await fetch(`${args.url}/.well-known/RIRD`).then((x) => x.json() as any)
254
+ prompts.log.info(`Running \`${wellknown.auth.command.join(" ")}\``)
255
+ const proc = Bun.spawn({
256
+ cmd: wellknown.auth.command,
257
+ stdout: "pipe",
258
+ })
259
+ const exit = await proc.exited
260
+ if (exit !== 0) {
261
+ prompts.log.error("Failed")
262
+ prompts.outro("Done")
263
+ return
264
+ }
265
+ const token = await new Response(proc.stdout).text()
266
+ await Auth.set(args.url, {
267
+ type: "wellknown",
268
+ key: wellknown.auth.env,
269
+ token: token.trim(),
270
+ })
271
+ prompts.log.success("Logged into " + args.url)
272
+ prompts.outro("Done")
273
+ return
274
+ }
275
+ await ModelsDev.refresh().catch(() => {})
276
+
277
+ const config = await Config.get()
278
+
279
+ const disabled = new Set(config.disabled_providers ?? [])
280
+ const enabled = config.enabled_providers ? new Set(config.enabled_providers) : undefined
281
+
282
+ const providers = await ModelsDev.get().then((x) => {
283
+ const filtered: Record<string, (typeof x)[string]> = {}
284
+ for (const [key, value] of Object.entries(x)) {
285
+ if ((enabled ? enabled.has(key) : true) && !disabled.has(key)) {
286
+ filtered[key] = value
287
+ }
288
+ }
289
+ return filtered
290
+ })
291
+
292
+ const priority: Record<string, number> = {
293
+ RIRD: 0,
294
+ anthropic: 1,
295
+ "github-copilot": 2,
296
+ openai: 3,
297
+ google: 4,
298
+ openrouter: 5,
299
+ vercel: 6,
300
+ }
301
+ let provider = await prompts.autocomplete({
302
+ message: "Select provider",
303
+ maxItems: 8,
304
+ options: [
305
+ ...pipe(
306
+ providers,
307
+ values(),
308
+ sortBy(
309
+ (x) => priority[x.id] ?? 99,
310
+ (x) => x.name ?? x.id,
311
+ ),
312
+ map((x) => ({
313
+ label: x.name,
314
+ value: x.id,
315
+ hint: {
316
+ RIRD: "recommended",
317
+ anthropic: "Claude Max or API key",
318
+ }[x.id],
319
+ })),
320
+ ),
321
+ {
322
+ value: "other",
323
+ label: "Other",
324
+ },
325
+ ],
326
+ })
327
+
328
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
329
+
330
+ const plugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
331
+ if (plugin && plugin.auth) {
332
+ const handled = await handlePluginAuth({ auth: plugin.auth }, provider)
333
+ if (handled) return
334
+ }
335
+
336
+ if (provider === "other") {
337
+ provider = await prompts.text({
338
+ message: "Enter provider id",
339
+ validate: (x) => (x && x.match(/^[0-9a-z-]+$/) ? undefined : "a-z, 0-9 and hyphens only"),
340
+ })
341
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
342
+ provider = provider.replace(/^@ai-sdk\//, "")
343
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
344
+
345
+ // Check if a plugin provides auth for this custom provider
346
+ const customPlugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
347
+ if (customPlugin && customPlugin.auth) {
348
+ const handled = await handlePluginAuth({ auth: customPlugin.auth }, provider)
349
+ if (handled) return
350
+ }
351
+
352
+ prompts.log.warn(
353
+ `This only stores a credential for ${provider} - you will need to configure it in rird.json, check the docs for examples.`,
354
+ )
355
+ }
356
+
357
+ if (provider === "amazon-bedrock") {
358
+ prompts.log.info(
359
+ "Amazon bedrock can be configured with standard AWS environment variables like AWS_BEARER_TOKEN_BEDROCK, AWS_PROFILE or AWS_ACCESS_KEY_ID",
360
+ )
361
+ prompts.outro("Done")
362
+ return
363
+ }
364
+
365
+ if (provider === "rird") {
366
+ prompts.log.info("Create an api key at https://rird.ai")
367
+ }
368
+
369
+ if (provider === "vercel") {
370
+ prompts.log.info("You can create an api key at https://vercel.link/ai-gateway-token")
371
+ }
372
+
373
+ const key = await prompts.password({
374
+ message: "Enter your API key",
375
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
376
+ })
377
+ if (prompts.isCancel(key)) throw new UI.CancelledError()
378
+ await Auth.set(provider, {
379
+ type: "api",
380
+ key,
381
+ })
382
+
383
+ prompts.outro("Done")
384
+ },
385
+ })
386
+ },
387
+ })
388
+
389
+ export const AuthLogoutCommand = cmd({
390
+ command: "logout",
391
+ describe: "log out from a configured provider",
392
+ async handler() {
393
+ UI.empty()
394
+ const credentials = await Auth.all().then((x) => Object.entries(x))
395
+ prompts.intro("Remove credential")
396
+ if (credentials.length === 0) {
397
+ prompts.log.error("No credentials found")
398
+ return
399
+ }
400
+ const database = await ModelsDev.get()
401
+ const providerID = await prompts.select({
402
+ message: "Select provider",
403
+ options: credentials.map(([key, value]) => ({
404
+ label: (database[key]?.name || key) + UI.Style.TEXT_DIM + " (" + value.type + ")",
405
+ value: key,
406
+ })),
407
+ })
408
+ if (prompts.isCancel(providerID)) throw new UI.CancelledError()
409
+ await Auth.remove(providerID)
410
+ prompts.outro("Logout successful")
411
+ },
412
+ })
@@ -0,0 +1,7 @@
1
+ import type { CommandModule } from "yargs"
2
+
3
+ type WithDoubleDash<T> = T & { "--"?: string[] }
4
+
5
+ export function cmd<T, U>(input: CommandModule<T, WithDoubleDash<U>>) {
6
+ return input
7
+ }
@@ -0,0 +1,15 @@
1
+ import { EOL } from "os"
2
+ import { Config } from "../../../config/config"
3
+ import { bootstrap } from "../../bootstrap"
4
+ import { cmd } from "../cmd"
5
+
6
+ export const ConfigCommand = cmd({
7
+ command: "config",
8
+ builder: (yargs) => yargs,
9
+ async handler() {
10
+ await bootstrap(process.cwd(), async () => {
11
+ const config = await Config.get()
12
+ process.stdout.write(JSON.stringify(config, null, 2) + EOL)
13
+ })
14
+ },
15
+ })