rird 1.0.200

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 (350) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +336 -0
  5. package/bin/pty-wrapper.js +285 -0
  6. package/bunfig.toml +4 -0
  7. package/facebook_ads_library.png +0 -0
  8. package/nul`nif +0 -0
  9. package/package.json +111 -0
  10. package/parsers-config.ts +239 -0
  11. package/rird-1.0.199.tgz +0 -0
  12. package/script/build-windows.ts +54 -0
  13. package/script/build.ts +167 -0
  14. package/script/postinstall.mjs +544 -0
  15. package/script/publish-registries.ts +187 -0
  16. package/script/publish.ts +72 -0
  17. package/script/schema.ts +47 -0
  18. package/src/acp/README.md +164 -0
  19. package/src/acp/agent.ts +1063 -0
  20. package/src/acp/session.ts +101 -0
  21. package/src/acp/types.ts +22 -0
  22. package/src/agent/agent.ts +367 -0
  23. package/src/agent/generate.txt +75 -0
  24. package/src/agent/prompt/compaction.txt +12 -0
  25. package/src/agent/prompt/explore.txt +18 -0
  26. package/src/agent/prompt/summary.txt +10 -0
  27. package/src/agent/prompt/title.txt +36 -0
  28. package/src/auth/index.ts +70 -0
  29. package/src/bun/index.ts +114 -0
  30. package/src/bus/bus-event.ts +43 -0
  31. package/src/bus/global.ts +10 -0
  32. package/src/bus/index.ts +105 -0
  33. package/src/cli/bootstrap.ts +17 -0
  34. package/src/cli/cmd/acp.ts +88 -0
  35. package/src/cli/cmd/agent.ts +256 -0
  36. package/src/cli/cmd/auth.ts +391 -0
  37. package/src/cli/cmd/cmd.ts +7 -0
  38. package/src/cli/cmd/debug/config.ts +15 -0
  39. package/src/cli/cmd/debug/file.ts +91 -0
  40. package/src/cli/cmd/debug/index.ts +43 -0
  41. package/src/cli/cmd/debug/lsp.ts +48 -0
  42. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  43. package/src/cli/cmd/debug/scrap.ts +15 -0
  44. package/src/cli/cmd/debug/skill.ts +15 -0
  45. package/src/cli/cmd/debug/snapshot.ts +48 -0
  46. package/src/cli/cmd/export.ts +88 -0
  47. package/src/cli/cmd/generate.ts +38 -0
  48. package/src/cli/cmd/github.ts +1400 -0
  49. package/src/cli/cmd/import.ts +98 -0
  50. package/src/cli/cmd/mcp.ts +654 -0
  51. package/src/cli/cmd/models.ts +77 -0
  52. package/src/cli/cmd/pr.ts +112 -0
  53. package/src/cli/cmd/run.ts +368 -0
  54. package/src/cli/cmd/serve.ts +31 -0
  55. package/src/cli/cmd/session.ts +106 -0
  56. package/src/cli/cmd/stats.ts +298 -0
  57. package/src/cli/cmd/tui/app.tsx +696 -0
  58. package/src/cli/cmd/tui/attach.ts +30 -0
  59. package/src/cli/cmd/tui/component/border.tsx +21 -0
  60. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  61. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  62. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  63. package/src/cli/cmd/tui/component/dialog-model.tsx +245 -0
  64. package/src/cli/cmd/tui/component/dialog-provider.tsx +224 -0
  65. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  66. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  67. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  68. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  69. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  70. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  71. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  72. package/src/cli/cmd/tui/component/logo.tsx +35 -0
  73. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +574 -0
  74. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  75. package/src/cli/cmd/tui/component/prompt/index.tsx +1090 -0
  76. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  77. package/src/cli/cmd/tui/component/tips.ts +27 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +14 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  85. package/src/cli/cmd/tui/context/local.tsx +354 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  95. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  96. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  97. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  98. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  99. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  100. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  101. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  102. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  103. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  104. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  105. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  107. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  108. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  111. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  112. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  113. package/src/cli/cmd/tui/context/theme/rird.json +245 -0
  114. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  115. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  116. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  117. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  118. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  119. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  120. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  121. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  122. package/src/cli/cmd/tui/event.ts +40 -0
  123. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  128. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  129. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  130. package/src/cli/cmd/tui/routes/session/index.tsx +1864 -0
  131. package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
  132. package/src/cli/cmd/tui/spawn.ts +60 -0
  133. package/src/cli/cmd/tui/thread.ts +142 -0
  134. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  135. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  136. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  137. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  138. package/src/cli/cmd/tui/ui/dialog-select.tsx +332 -0
  139. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  140. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  141. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  142. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  143. package/src/cli/cmd/tui/util/editor.ts +32 -0
  144. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  145. package/src/cli/cmd/tui/worker.ts +63 -0
  146. package/src/cli/cmd/uninstall.ts +344 -0
  147. package/src/cli/cmd/upgrade.ts +100 -0
  148. package/src/cli/cmd/web.ts +84 -0
  149. package/src/cli/error.ts +56 -0
  150. package/src/cli/ui.ts +84 -0
  151. package/src/cli/upgrade.ts +25 -0
  152. package/src/command/index.ts +80 -0
  153. package/src/command/template/initialize.txt +10 -0
  154. package/src/command/template/review.txt +97 -0
  155. package/src/config/config.ts +995 -0
  156. package/src/config/markdown.ts +41 -0
  157. package/src/env/index.ts +26 -0
  158. package/src/file/ignore.ts +83 -0
  159. package/src/file/index.ts +328 -0
  160. package/src/file/ripgrep.ts +393 -0
  161. package/src/file/time.ts +64 -0
  162. package/src/file/watcher.ts +103 -0
  163. package/src/flag/flag.ts +46 -0
  164. package/src/format/formatter.ts +315 -0
  165. package/src/format/index.ts +137 -0
  166. package/src/global/index.ts +52 -0
  167. package/src/id/id.ts +73 -0
  168. package/src/ide/index.ts +76 -0
  169. package/src/index.ts +240 -0
  170. package/src/installation/index.ts +239 -0
  171. package/src/lsp/client.ts +229 -0
  172. package/src/lsp/index.ts +485 -0
  173. package/src/lsp/language.ts +116 -0
  174. package/src/lsp/server.ts +1895 -0
  175. package/src/mcp/auth.ts +135 -0
  176. package/src/mcp/index.ts +690 -0
  177. package/src/mcp/oauth-callback.ts +200 -0
  178. package/src/mcp/oauth-provider.ts +154 -0
  179. package/src/patch/index.ts +622 -0
  180. package/src/permission/index.ts +199 -0
  181. package/src/plugin/index.ts +91 -0
  182. package/src/project/bootstrap.ts +31 -0
  183. package/src/project/instance.ts +78 -0
  184. package/src/project/project.ts +221 -0
  185. package/src/project/state.ts +65 -0
  186. package/src/project/vcs.ts +76 -0
  187. package/src/provider/auth.ts +143 -0
  188. package/src/provider/models-macro.ts +11 -0
  189. package/src/provider/models.ts +106 -0
  190. package/src/provider/provider.ts +1071 -0
  191. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  192. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  193. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  194. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  195. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  196. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  197. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  198. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  199. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  200. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  201. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  202. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  203. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  204. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  205. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  206. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  207. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  208. package/src/provider/transform.ts +455 -0
  209. package/src/pty/index.ts +231 -0
  210. package/src/security/guardrails.test.ts +341 -0
  211. package/src/security/guardrails.ts +558 -0
  212. package/src/security/index.ts +19 -0
  213. package/src/server/error.ts +36 -0
  214. package/src/server/project.ts +79 -0
  215. package/src/server/server.ts +2642 -0
  216. package/src/server/tui.ts +71 -0
  217. package/src/session/compaction.ts +223 -0
  218. package/src/session/index.ts +461 -0
  219. package/src/session/llm.ts +201 -0
  220. package/src/session/message-v2.ts +690 -0
  221. package/src/session/message.ts +189 -0
  222. package/src/session/processor.ts +409 -0
  223. package/src/session/prompt/act-switch.txt +5 -0
  224. package/src/session/prompt/anthropic-20250930.txt +166 -0
  225. package/src/session/prompt/anthropic.txt +85 -0
  226. package/src/session/prompt/anthropic_spoof.txt +1 -0
  227. package/src/session/prompt/beast.txt +103 -0
  228. package/src/session/prompt/codex.txt +304 -0
  229. package/src/session/prompt/copilot-gpt-5.txt +138 -0
  230. package/src/session/prompt/gemini.txt +85 -0
  231. package/src/session/prompt/max-steps.txt +16 -0
  232. package/src/session/prompt/plan-reminder-anthropic.txt +35 -0
  233. package/src/session/prompt/plan.txt +24 -0
  234. package/src/session/prompt/polaris.txt +84 -0
  235. package/src/session/prompt/qwen.txt +106 -0
  236. package/src/session/prompt.ts +1509 -0
  237. package/src/session/retry.ts +86 -0
  238. package/src/session/revert.ts +108 -0
  239. package/src/session/sensitive-filter.test.ts +327 -0
  240. package/src/session/sensitive-filter.ts +466 -0
  241. package/src/session/status.ts +76 -0
  242. package/src/session/summary.ts +194 -0
  243. package/src/session/system.ts +120 -0
  244. package/src/session/todo.ts +37 -0
  245. package/src/share/share-next.ts +194 -0
  246. package/src/share/share.ts +87 -0
  247. package/src/shell/shell.ts +67 -0
  248. package/src/skill/index.ts +1 -0
  249. package/src/skill/skill.ts +83 -0
  250. package/src/snapshot/index.ts +197 -0
  251. package/src/storage/storage.ts +226 -0
  252. package/src/tests/agent.test.ts +308 -0
  253. package/src/tests/build-guards.test.ts +267 -0
  254. package/src/tests/config.test.ts +664 -0
  255. package/src/tests/tool-registry.test.ts +589 -0
  256. package/src/tool/bash.ts +317 -0
  257. package/src/tool/bash.txt +158 -0
  258. package/src/tool/batch.ts +175 -0
  259. package/src/tool/batch.txt +24 -0
  260. package/src/tool/codesearch.ts +168 -0
  261. package/src/tool/codesearch.txt +12 -0
  262. package/src/tool/edit.ts +675 -0
  263. package/src/tool/edit.txt +10 -0
  264. package/src/tool/glob.ts +65 -0
  265. package/src/tool/glob.txt +6 -0
  266. package/src/tool/grep.ts +121 -0
  267. package/src/tool/grep.txt +8 -0
  268. package/src/tool/invalid.ts +17 -0
  269. package/src/tool/ls.ts +110 -0
  270. package/src/tool/ls.txt +1 -0
  271. package/src/tool/lsp-diagnostics.ts +26 -0
  272. package/src/tool/lsp-diagnostics.txt +1 -0
  273. package/src/tool/lsp-hover.ts +31 -0
  274. package/src/tool/lsp-hover.txt +1 -0
  275. package/src/tool/lsp.ts +87 -0
  276. package/src/tool/lsp.txt +19 -0
  277. package/src/tool/multiedit.ts +46 -0
  278. package/src/tool/multiedit.txt +41 -0
  279. package/src/tool/patch.ts +233 -0
  280. package/src/tool/patch.txt +1 -0
  281. package/src/tool/read.ts +219 -0
  282. package/src/tool/read.txt +12 -0
  283. package/src/tool/registry.ts +162 -0
  284. package/src/tool/skill.ts +100 -0
  285. package/src/tool/task.ts +136 -0
  286. package/src/tool/task.txt +51 -0
  287. package/src/tool/todo.ts +39 -0
  288. package/src/tool/todoread.txt +14 -0
  289. package/src/tool/todowrite.txt +167 -0
  290. package/src/tool/tool.ts +71 -0
  291. package/src/tool/webfetch.ts +198 -0
  292. package/src/tool/webfetch.txt +13 -0
  293. package/src/tool/websearch.ts +180 -0
  294. package/src/tool/websearch.txt +11 -0
  295. package/src/tool/write.ts +110 -0
  296. package/src/tool/write.txt +8 -0
  297. package/src/util/archive.ts +16 -0
  298. package/src/util/color.ts +19 -0
  299. package/src/util/context.ts +25 -0
  300. package/src/util/defer.ts +12 -0
  301. package/src/util/eventloop.ts +20 -0
  302. package/src/util/filesystem.ts +83 -0
  303. package/src/util/fn.ts +11 -0
  304. package/src/util/iife.ts +3 -0
  305. package/src/util/keybind.ts +102 -0
  306. package/src/util/lazy.ts +11 -0
  307. package/src/util/license.ts +325 -0
  308. package/src/util/locale.ts +81 -0
  309. package/src/util/lock.ts +98 -0
  310. package/src/util/log.ts +180 -0
  311. package/src/util/queue.ts +32 -0
  312. package/src/util/rpc.ts +42 -0
  313. package/src/util/scrap.ts +10 -0
  314. package/src/util/signal.ts +12 -0
  315. package/src/util/timeout.ts +14 -0
  316. package/src/util/token.ts +7 -0
  317. package/src/util/wildcard.ts +54 -0
  318. package/sst-env.d.ts +9 -0
  319. package/test/agent/agent.test.ts +146 -0
  320. package/test/bun.test.ts +53 -0
  321. package/test/cli/github-remote.test.ts +80 -0
  322. package/test/config/agent-color.test.ts +66 -0
  323. package/test/config/config.test.ts +535 -0
  324. package/test/config/markdown.test.ts +89 -0
  325. package/test/file/ignore.test.ts +10 -0
  326. package/test/fixture/fixture.ts +36 -0
  327. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  328. package/test/ide/ide.test.ts +82 -0
  329. package/test/keybind.test.ts +421 -0
  330. package/test/lsp/client.test.ts +95 -0
  331. package/test/mcp/headers.test.ts +153 -0
  332. package/test/patch/patch.test.ts +348 -0
  333. package/test/preload.ts +57 -0
  334. package/test/project/project.test.ts +72 -0
  335. package/test/provider/provider.test.ts +1809 -0
  336. package/test/provider/transform.test.ts +411 -0
  337. package/test/session/retry.test.ts +111 -0
  338. package/test/session/session.test.ts +71 -0
  339. package/test/skill/skill.test.ts +131 -0
  340. package/test/snapshot/snapshot.test.ts +939 -0
  341. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  342. package/test/tool/bash.test.ts +434 -0
  343. package/test/tool/grep.test.ts +108 -0
  344. package/test/tool/patch.test.ts +259 -0
  345. package/test/tool/read.test.ts +42 -0
  346. package/test/util/iife.test.ts +36 -0
  347. package/test/util/lazy.test.ts +50 -0
  348. package/test/util/timeout.test.ts +21 -0
  349. package/test/util/wildcard.test.ts +55 -0
  350. package/tsconfig.json +16 -0
@@ -0,0 +1,391 @@
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
+ type PluginAuth = NonNullable<Hooks["auth"]>
16
+
17
+ /**
18
+ * Handle plugin-based authentication flow.
19
+ * Returns true if auth was handled, false if it should fall through to default handling.
20
+ */
21
+ async function handlePluginAuth(plugin: { auth: PluginAuth }, provider: string): Promise<boolean> {
22
+ let index = 0
23
+ if (plugin.auth.methods.length > 1) {
24
+ const method = await prompts.select({
25
+ message: "Login method",
26
+ options: [
27
+ ...plugin.auth.methods.map((x, index) => ({
28
+ label: x.label,
29
+ value: index.toString(),
30
+ })),
31
+ ],
32
+ })
33
+ if (prompts.isCancel(method)) throw new UI.CancelledError()
34
+ index = parseInt(method)
35
+ }
36
+ const method = plugin.auth.methods[index]
37
+
38
+ // Handle prompts for all auth types
39
+ await new Promise((resolve) => setTimeout(resolve, 10))
40
+ const inputs: Record<string, string> = {}
41
+ if (method.prompts) {
42
+ for (const prompt of method.prompts) {
43
+ if (prompt.condition && !prompt.condition(inputs)) {
44
+ continue
45
+ }
46
+ if (prompt.type === "select") {
47
+ const value = await prompts.select({
48
+ message: prompt.message,
49
+ options: prompt.options,
50
+ })
51
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
52
+ inputs[prompt.key] = value
53
+ } else {
54
+ const value = await prompts.text({
55
+ message: prompt.message,
56
+ placeholder: prompt.placeholder,
57
+ validate: prompt.validate ? (v) => prompt.validate!(v ?? "") : undefined,
58
+ })
59
+ if (prompts.isCancel(value)) throw new UI.CancelledError()
60
+ inputs[prompt.key] = value
61
+ }
62
+ }
63
+ }
64
+
65
+ if (method.type === "oauth") {
66
+ const authorize = await method.authorize(inputs)
67
+
68
+ if (authorize.url) {
69
+ prompts.log.info("Go to: " + authorize.url)
70
+ }
71
+
72
+ if (authorize.method === "auto") {
73
+ if (authorize.instructions) {
74
+ prompts.log.info(authorize.instructions)
75
+ }
76
+ const spinner = prompts.spinner()
77
+ spinner.start("Waiting for authorization...")
78
+ const result = await authorize.callback()
79
+ if (result.type === "failed") {
80
+ spinner.stop("Failed to authorize", 1)
81
+ }
82
+ if (result.type === "success") {
83
+ const saveProvider = result.provider ?? provider
84
+ if ("refresh" in result) {
85
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
86
+ await Auth.set(saveProvider, {
87
+ type: "oauth",
88
+ refresh,
89
+ access,
90
+ expires,
91
+ ...extraFields,
92
+ })
93
+ }
94
+ if ("key" in result) {
95
+ await Auth.set(saveProvider, {
96
+ type: "api",
97
+ key: result.key,
98
+ })
99
+ }
100
+ spinner.stop("Login successful")
101
+ }
102
+ }
103
+
104
+ if (authorize.method === "code") {
105
+ const code = await prompts.text({
106
+ message: "Paste the authorization code here: ",
107
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
108
+ })
109
+ if (prompts.isCancel(code)) throw new UI.CancelledError()
110
+ const result = await authorize.callback(code)
111
+ if (result.type === "failed") {
112
+ prompts.log.error("Failed to authorize")
113
+ }
114
+ if (result.type === "success") {
115
+ const saveProvider = result.provider ?? provider
116
+ if ("refresh" in result) {
117
+ const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
118
+ await Auth.set(saveProvider, {
119
+ type: "oauth",
120
+ refresh,
121
+ access,
122
+ expires,
123
+ ...extraFields,
124
+ })
125
+ }
126
+ if ("key" in result) {
127
+ await Auth.set(saveProvider, {
128
+ type: "api",
129
+ key: result.key,
130
+ })
131
+ }
132
+ prompts.log.success("Login successful")
133
+ }
134
+ }
135
+
136
+ prompts.outro("Done")
137
+ return true
138
+ }
139
+
140
+ if (method.type === "api") {
141
+ if (method.authorize) {
142
+ const result = await method.authorize(inputs)
143
+ if (result.type === "failed") {
144
+ prompts.log.error("Failed to authorize")
145
+ }
146
+ if (result.type === "success") {
147
+ const saveProvider = result.provider ?? provider
148
+ await Auth.set(saveProvider, {
149
+ type: "api",
150
+ key: result.key,
151
+ })
152
+ prompts.log.success("Login successful")
153
+ }
154
+ prompts.outro("Done")
155
+ return true
156
+ }
157
+ }
158
+
159
+ return false
160
+ }
161
+
162
+ export const AuthCommand = cmd({
163
+ command: "auth",
164
+ describe: "manage credentials",
165
+ builder: (yargs) =>
166
+ yargs.command(AuthLoginCommand).command(AuthLogoutCommand).command(AuthListCommand).demandCommand(),
167
+ async handler() {},
168
+ })
169
+
170
+ export const AuthListCommand = cmd({
171
+ command: "list",
172
+ aliases: ["ls"],
173
+ describe: "list providers",
174
+ async handler() {
175
+ UI.empty()
176
+ const authPath = path.join(Global.Path.data, "auth.json")
177
+ const homedir = os.homedir()
178
+ const displayPath = authPath.startsWith(homedir) ? authPath.replace(homedir, "~") : authPath
179
+ prompts.intro(`Credentials ${UI.Style.TEXT_DIM}${displayPath}`)
180
+ const results = Object.entries(await Auth.all())
181
+ const database = await ModelsDev.get()
182
+
183
+ for (const [providerID, result] of results) {
184
+ const name = database[providerID]?.name || providerID
185
+ prompts.log.info(`${name} ${UI.Style.TEXT_DIM}${result.type}`)
186
+ }
187
+
188
+ prompts.outro(`${results.length} credentials`)
189
+
190
+ // Environment variables section
191
+ const activeEnvVars: Array<{ provider: string; envVar: string }> = []
192
+
193
+ for (const [providerID, provider] of Object.entries(database)) {
194
+ for (const envVar of provider.env) {
195
+ if (process.env[envVar]) {
196
+ activeEnvVars.push({
197
+ provider: provider.name || providerID,
198
+ envVar,
199
+ })
200
+ }
201
+ }
202
+ }
203
+
204
+ if (activeEnvVars.length > 0) {
205
+ UI.empty()
206
+ prompts.intro("Environment")
207
+
208
+ for (const { provider, envVar } of activeEnvVars) {
209
+ prompts.log.info(`${provider} ${UI.Style.TEXT_DIM}${envVar}`)
210
+ }
211
+
212
+ prompts.outro(`${activeEnvVars.length} environment variable` + (activeEnvVars.length === 1 ? "" : "s"))
213
+ }
214
+ },
215
+ })
216
+
217
+ export const AuthLoginCommand = cmd({
218
+ command: "login [url]",
219
+ describe: "log in to a provider",
220
+ builder: (yargs) =>
221
+ yargs.positional("url", {
222
+ describe: "RIRD auth provider",
223
+ type: "string",
224
+ }),
225
+ async handler(args) {
226
+ await Instance.provide({
227
+ directory: process.cwd(),
228
+ async fn() {
229
+ UI.empty()
230
+ prompts.intro("Add credential")
231
+ if (args.url) {
232
+ const wellknown = await fetch(`${args.url}/.well-known/opencode`).then((x) => x.json() as any)
233
+ prompts.log.info(`Running \`${wellknown.auth.command.join(" ")}\``)
234
+ const proc = Bun.spawn({
235
+ cmd: wellknown.auth.command,
236
+ stdout: "pipe",
237
+ })
238
+ const exit = await proc.exited
239
+ if (exit !== 0) {
240
+ prompts.log.error("Failed")
241
+ prompts.outro("Done")
242
+ return
243
+ }
244
+ const token = await new Response(proc.stdout).text()
245
+ await Auth.set(args.url, {
246
+ type: "wellknown",
247
+ key: wellknown.auth.env,
248
+ token: token.trim(),
249
+ })
250
+ prompts.log.success("Logged into " + args.url)
251
+ prompts.outro("Done")
252
+ return
253
+ }
254
+ await ModelsDev.refresh().catch(() => {})
255
+
256
+ const config = await Config.get()
257
+
258
+ const disabled = new Set(config.disabled_providers ?? [])
259
+ const enabled = config.enabled_providers ? new Set(config.enabled_providers) : undefined
260
+
261
+ const providers = await ModelsDev.get().then((x) => {
262
+ const filtered: Record<string, (typeof x)[string]> = {}
263
+ for (const [key, value] of Object.entries(x)) {
264
+ if ((enabled ? enabled.has(key) : true) && !disabled.has(key)) {
265
+ filtered[key] = value
266
+ }
267
+ }
268
+ return filtered
269
+ })
270
+
271
+ const priority: Record<string, number> = {
272
+ opencode: 0,
273
+ anthropic: 1,
274
+ "github-copilot": 2,
275
+ openai: 3,
276
+ google: 4,
277
+ openrouter: 5,
278
+ vercel: 6,
279
+ }
280
+ let provider = await prompts.autocomplete({
281
+ message: "Select provider",
282
+ maxItems: 8,
283
+ options: [
284
+ ...pipe(
285
+ providers,
286
+ values(),
287
+ sortBy(
288
+ (x) => priority[x.id] ?? 99,
289
+ (x) => x.name ?? x.id,
290
+ ),
291
+ map((x) => ({
292
+ label: x.name,
293
+ value: x.id,
294
+ hint: {
295
+ opencode: "recommended",
296
+ anthropic: "Claude Max or API key",
297
+ }[x.id],
298
+ })),
299
+ ),
300
+ {
301
+ value: "other",
302
+ label: "Other",
303
+ },
304
+ ],
305
+ })
306
+
307
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
308
+
309
+ const plugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
310
+ if (plugin && plugin.auth) {
311
+ const handled = await handlePluginAuth({ auth: plugin.auth }, provider)
312
+ if (handled) return
313
+ }
314
+
315
+ if (provider === "other") {
316
+ provider = await prompts.text({
317
+ message: "Enter provider id",
318
+ validate: (x) => (x && x.match(/^[0-9a-z-]+$/) ? undefined : "a-z, 0-9 and hyphens only"),
319
+ })
320
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
321
+ provider = provider.replace(/^@ai-sdk\//, "")
322
+ if (prompts.isCancel(provider)) throw new UI.CancelledError()
323
+
324
+ // Check if a plugin provides auth for this custom provider
325
+ const customPlugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
326
+ if (customPlugin && customPlugin.auth) {
327
+ const handled = await handlePluginAuth({ auth: customPlugin.auth }, provider)
328
+ if (handled) return
329
+ }
330
+
331
+ prompts.log.warn(
332
+ `This only stores a credential for ${provider} - you will need to configure it in rird.json, check the docs for examples.`,
333
+ )
334
+ }
335
+
336
+ if (provider === "amazon-bedrock") {
337
+ prompts.log.info(
338
+ "Amazon bedrock can be configured with standard AWS environment variables like AWS_BEARER_TOKEN_BEDROCK, AWS_PROFILE or AWS_ACCESS_KEY_ID",
339
+ )
340
+ prompts.outro("Done")
341
+ return
342
+ }
343
+
344
+ if (provider === "rird") {
345
+ prompts.log.info("Create an api key at https://rird.ai")
346
+ }
347
+
348
+ if (provider === "vercel") {
349
+ prompts.log.info("You can create an api key at https://vercel.link/ai-gateway-token")
350
+ }
351
+
352
+ const key = await prompts.password({
353
+ message: "Enter your API key",
354
+ validate: (x) => (x && x.length > 0 ? undefined : "Required"),
355
+ })
356
+ if (prompts.isCancel(key)) throw new UI.CancelledError()
357
+ await Auth.set(provider, {
358
+ type: "api",
359
+ key,
360
+ })
361
+
362
+ prompts.outro("Done")
363
+ },
364
+ })
365
+ },
366
+ })
367
+
368
+ export const AuthLogoutCommand = cmd({
369
+ command: "logout",
370
+ describe: "log out from a configured provider",
371
+ async handler() {
372
+ UI.empty()
373
+ const credentials = await Auth.all().then((x) => Object.entries(x))
374
+ prompts.intro("Remove credential")
375
+ if (credentials.length === 0) {
376
+ prompts.log.error("No credentials found")
377
+ return
378
+ }
379
+ const database = await ModelsDev.get()
380
+ const providerID = await prompts.select({
381
+ message: "Select provider",
382
+ options: credentials.map(([key, value]) => ({
383
+ label: (database[key]?.name || key) + UI.Style.TEXT_DIM + " (" + value.type + ")",
384
+ value: key,
385
+ })),
386
+ })
387
+ if (prompts.isCancel(providerID)) throw new UI.CancelledError()
388
+ await Auth.remove(providerID)
389
+ prompts.outro("Logout successful")
390
+ },
391
+ })
@@ -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
+ })
@@ -0,0 +1,91 @@
1
+ import { EOL } from "os"
2
+ import { File } from "../../../file"
3
+ import { bootstrap } from "../../bootstrap"
4
+ import { cmd } from "../cmd"
5
+ import { Ripgrep } from "@/file/ripgrep"
6
+
7
+ const FileSearchCommand = cmd({
8
+ command: "search <query>",
9
+ builder: (yargs) =>
10
+ yargs.positional("query", {
11
+ type: "string",
12
+ demandOption: true,
13
+ description: "Search query",
14
+ }),
15
+ async handler(args) {
16
+ await bootstrap(process.cwd(), async () => {
17
+ const results = await File.search({ query: args.query })
18
+ process.stdout.write(results.join(EOL) + EOL)
19
+ })
20
+ },
21
+ })
22
+
23
+ const FileReadCommand = cmd({
24
+ command: "read <path>",
25
+ builder: (yargs) =>
26
+ yargs.positional("path", {
27
+ type: "string",
28
+ demandOption: true,
29
+ description: "File path to read",
30
+ }),
31
+ async handler(args) {
32
+ await bootstrap(process.cwd(), async () => {
33
+ const content = await File.read(args.path)
34
+ process.stdout.write(JSON.stringify(content, null, 2) + EOL)
35
+ })
36
+ },
37
+ })
38
+
39
+ const FileStatusCommand = cmd({
40
+ command: "status",
41
+ builder: (yargs) => yargs,
42
+ async handler() {
43
+ await bootstrap(process.cwd(), async () => {
44
+ const status = await File.status()
45
+ process.stdout.write(JSON.stringify(status, null, 2) + EOL)
46
+ })
47
+ },
48
+ })
49
+
50
+ const FileListCommand = cmd({
51
+ command: "list <path>",
52
+ builder: (yargs) =>
53
+ yargs.positional("path", {
54
+ type: "string",
55
+ demandOption: true,
56
+ description: "File path to list",
57
+ }),
58
+ async handler(args) {
59
+ await bootstrap(process.cwd(), async () => {
60
+ const files = await File.list(args.path)
61
+ process.stdout.write(JSON.stringify(files, null, 2) + EOL)
62
+ })
63
+ },
64
+ })
65
+
66
+ const FileTreeCommand = cmd({
67
+ command: "tree [dir]",
68
+ builder: (yargs) =>
69
+ yargs.positional("dir", {
70
+ type: "string",
71
+ description: "Directory to tree",
72
+ default: process.cwd(),
73
+ }),
74
+ async handler(args) {
75
+ const files = await Ripgrep.tree({ cwd: args.dir, limit: 200 })
76
+ console.log(files)
77
+ },
78
+ })
79
+
80
+ export const FileCommand = cmd({
81
+ command: "file",
82
+ builder: (yargs) =>
83
+ yargs
84
+ .command(FileReadCommand)
85
+ .command(FileStatusCommand)
86
+ .command(FileListCommand)
87
+ .command(FileSearchCommand)
88
+ .command(FileTreeCommand)
89
+ .demandCommand(),
90
+ async handler() {},
91
+ })
@@ -0,0 +1,43 @@
1
+ import { Global } from "../../../global"
2
+ import { bootstrap } from "../../bootstrap"
3
+ import { cmd } from "../cmd"
4
+ import { ConfigCommand } from "./config"
5
+ import { FileCommand } from "./file"
6
+ import { LSPCommand } from "./lsp"
7
+ import { RipgrepCommand } from "./ripgrep"
8
+ import { ScrapCommand } from "./scrap"
9
+ import { SkillCommand } from "./skill"
10
+ import { SnapshotCommand } from "./snapshot"
11
+
12
+ export const DebugCommand = cmd({
13
+ command: "debug",
14
+ builder: (yargs) =>
15
+ yargs
16
+ .command(ConfigCommand)
17
+ .command(LSPCommand)
18
+ .command(RipgrepCommand)
19
+ .command(FileCommand)
20
+ .command(ScrapCommand)
21
+ .command(SkillCommand)
22
+ .command(SnapshotCommand)
23
+ .command(PathsCommand)
24
+ .command({
25
+ command: "wait",
26
+ async handler() {
27
+ await bootstrap(process.cwd(), async () => {
28
+ await new Promise((resolve) => setTimeout(resolve, 1_000 * 60 * 60 * 24))
29
+ })
30
+ },
31
+ })
32
+ .demandCommand(),
33
+ async handler() {},
34
+ })
35
+
36
+ const PathsCommand = cmd({
37
+ command: "paths",
38
+ handler() {
39
+ for (const [key, value] of Object.entries(Global.Path)) {
40
+ console.log(key.padEnd(10), value)
41
+ }
42
+ },
43
+ })
@@ -0,0 +1,48 @@
1
+ import { LSP } from "../../../lsp"
2
+ import { bootstrap } from "../../bootstrap"
3
+ import { cmd } from "../cmd"
4
+ import { Log } from "../../../util/log"
5
+ import { EOL } from "os"
6
+
7
+ export const LSPCommand = cmd({
8
+ command: "lsp",
9
+ builder: (yargs) =>
10
+ yargs.command(DiagnosticsCommand).command(SymbolsCommand).command(DocumentSymbolsCommand).demandCommand(),
11
+ async handler() {},
12
+ })
13
+
14
+ const DiagnosticsCommand = cmd({
15
+ command: "diagnostics <file>",
16
+ builder: (yargs) => yargs.positional("file", { type: "string", demandOption: true }),
17
+ async handler(args) {
18
+ await bootstrap(process.cwd(), async () => {
19
+ await LSP.touchFile(args.file, true)
20
+ await Bun.sleep(1000)
21
+ process.stdout.write(JSON.stringify(await LSP.diagnostics(), null, 2) + EOL)
22
+ })
23
+ },
24
+ })
25
+
26
+ export const SymbolsCommand = cmd({
27
+ command: "symbols <query>",
28
+ builder: (yargs) => yargs.positional("query", { type: "string", demandOption: true }),
29
+ async handler(args) {
30
+ await bootstrap(process.cwd(), async () => {
31
+ using _ = Log.Default.time("symbols")
32
+ const results = await LSP.workspaceSymbol(args.query)
33
+ process.stdout.write(JSON.stringify(results, null, 2) + EOL)
34
+ })
35
+ },
36
+ })
37
+
38
+ export const DocumentSymbolsCommand = cmd({
39
+ command: "document-symbols <uri>",
40
+ builder: (yargs) => yargs.positional("uri", { type: "string", demandOption: true }),
41
+ async handler(args) {
42
+ await bootstrap(process.cwd(), async () => {
43
+ using _ = Log.Default.time("document-symbols")
44
+ const results = await LSP.documentSymbol(args.uri)
45
+ process.stdout.write(JSON.stringify(results, null, 2) + EOL)
46
+ })
47
+ },
48
+ })
@@ -0,0 +1,83 @@
1
+ import { EOL } from "os"
2
+ import { Ripgrep } from "../../../file/ripgrep"
3
+ import { Instance } from "../../../project/instance"
4
+ import { bootstrap } from "../../bootstrap"
5
+ import { cmd } from "../cmd"
6
+
7
+ export const RipgrepCommand = cmd({
8
+ command: "rg",
9
+ builder: (yargs) => yargs.command(TreeCommand).command(FilesCommand).command(SearchCommand).demandCommand(),
10
+ async handler() {},
11
+ })
12
+
13
+ const TreeCommand = cmd({
14
+ command: "tree",
15
+ builder: (yargs) =>
16
+ yargs.option("limit", {
17
+ type: "number",
18
+ }),
19
+ async handler(args) {
20
+ await bootstrap(process.cwd(), async () => {
21
+ process.stdout.write((await Ripgrep.tree({ cwd: Instance.directory, limit: args.limit })) + EOL)
22
+ })
23
+ },
24
+ })
25
+
26
+ const FilesCommand = cmd({
27
+ command: "files",
28
+ builder: (yargs) =>
29
+ yargs
30
+ .option("query", {
31
+ type: "string",
32
+ description: "Filter files by query",
33
+ })
34
+ .option("glob", {
35
+ type: "string",
36
+ description: "Glob pattern to match files",
37
+ })
38
+ .option("limit", {
39
+ type: "number",
40
+ description: "Limit number of results",
41
+ }),
42
+ async handler(args) {
43
+ await bootstrap(process.cwd(), async () => {
44
+ const files: string[] = []
45
+ for await (const file of Ripgrep.files({
46
+ cwd: Instance.directory,
47
+ glob: args.glob ? [args.glob] : undefined,
48
+ })) {
49
+ files.push(file)
50
+ if (args.limit && files.length >= args.limit) break
51
+ }
52
+ process.stdout.write(files.join(EOL) + EOL)
53
+ })
54
+ },
55
+ })
56
+
57
+ const SearchCommand = cmd({
58
+ command: "search <pattern>",
59
+ builder: (yargs) =>
60
+ yargs
61
+ .positional("pattern", {
62
+ type: "string",
63
+ demandOption: true,
64
+ description: "Search pattern",
65
+ })
66
+ .option("glob", {
67
+ type: "array",
68
+ description: "File glob patterns",
69
+ })
70
+ .option("limit", {
71
+ type: "number",
72
+ description: "Limit number of results",
73
+ }),
74
+ async handler(args) {
75
+ const results = await Ripgrep.search({
76
+ cwd: process.cwd(),
77
+ pattern: args.pattern,
78
+ glob: args.glob as string[] | undefined,
79
+ limit: args.limit,
80
+ })
81
+ process.stdout.write(JSON.stringify(results, null, 2) + EOL)
82
+ },
83
+ })
@@ -0,0 +1,15 @@
1
+ import { EOL } from "os"
2
+ import { Project } from "../../../project/project"
3
+ import { Log } from "../../../util/log"
4
+ import { cmd } from "../cmd"
5
+
6
+ export const ScrapCommand = cmd({
7
+ command: "scrap",
8
+ builder: (yargs) => yargs,
9
+ async handler() {
10
+ const timer = Log.Default.time("scrap")
11
+ const list = await Project.list()
12
+ process.stdout.write(JSON.stringify(list, null, 2) + EOL)
13
+ timer.stop()
14
+ },
15
+ })