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,315 @@
1
+ import { readableStreamToText } from "bun"
2
+ import { BunProc } from "../bun"
3
+ import { Instance } from "../project/instance"
4
+ import { Filesystem } from "../util/filesystem"
5
+ import { Flag } from "@/flag/flag"
6
+
7
+ export interface Info {
8
+ name: string
9
+ command: string[]
10
+ environment?: Record<string, string>
11
+ extensions: string[]
12
+ enabled(): Promise<boolean>
13
+ }
14
+
15
+ export const gofmt: Info = {
16
+ name: "gofmt",
17
+ command: ["gofmt", "-w", "$FILE"],
18
+ extensions: [".go"],
19
+ async enabled() {
20
+ return Bun.which("gofmt") !== null
21
+ },
22
+ }
23
+
24
+ export const mix: Info = {
25
+ name: "mix",
26
+ command: ["mix", "format", "$FILE"],
27
+ extensions: [".ex", ".exs", ".eex", ".heex", ".leex", ".neex", ".sface"],
28
+ async enabled() {
29
+ return Bun.which("mix") !== null
30
+ },
31
+ }
32
+
33
+ export const prettier: Info = {
34
+ name: "prettier",
35
+ command: [BunProc.which(), "x", "prettier", "--write", "$FILE"],
36
+ environment: {
37
+ BUN_BE_BUN: "1",
38
+ },
39
+ extensions: [
40
+ ".js",
41
+ ".jsx",
42
+ ".mjs",
43
+ ".cjs",
44
+ ".ts",
45
+ ".tsx",
46
+ ".mts",
47
+ ".cts",
48
+ ".html",
49
+ ".htm",
50
+ ".css",
51
+ ".scss",
52
+ ".sass",
53
+ ".less",
54
+ ".vue",
55
+ ".svelte",
56
+ ".json",
57
+ ".jsonc",
58
+ ".yaml",
59
+ ".yml",
60
+ ".toml",
61
+ ".xml",
62
+ ".md",
63
+ ".mdx",
64
+ ".graphql",
65
+ ".gql",
66
+ ],
67
+ async enabled() {
68
+ const items = await Filesystem.findUp("package.json", Instance.directory, Instance.worktree)
69
+ for (const item of items) {
70
+ const json = await Bun.file(item).json()
71
+ if (json.dependencies?.prettier) return true
72
+ if (json.devDependencies?.prettier) return true
73
+ }
74
+ return false
75
+ },
76
+ }
77
+
78
+ export const oxfmt: Info = {
79
+ name: "oxfmt",
80
+ command: [BunProc.which(), "x", "oxfmt", "$FILE"],
81
+ environment: {
82
+ BUN_BE_BUN: "1",
83
+ },
84
+ extensions: [".js", ".jsx", ".mjs", ".cjs", ".ts", ".tsx", ".mts", ".cts"],
85
+ async enabled() {
86
+ if (!Flag.RIRD_EXPERIMENTAL_OXFMT) return false
87
+ const items = await Filesystem.findUp("package.json", Instance.directory, Instance.worktree)
88
+ for (const item of items) {
89
+ const json = await Bun.file(item).json()
90
+ if (json.dependencies?.oxfmt) return true
91
+ if (json.devDependencies?.oxfmt) return true
92
+ }
93
+ return false
94
+ },
95
+ }
96
+
97
+ export const biome: Info = {
98
+ name: "biome",
99
+ command: [BunProc.which(), "x", "@biomejs/biome", "format", "--write", "$FILE"],
100
+ environment: {
101
+ BUN_BE_BUN: "1",
102
+ },
103
+ extensions: [
104
+ ".js",
105
+ ".jsx",
106
+ ".mjs",
107
+ ".cjs",
108
+ ".ts",
109
+ ".tsx",
110
+ ".mts",
111
+ ".cts",
112
+ ".html",
113
+ ".htm",
114
+ ".css",
115
+ ".scss",
116
+ ".sass",
117
+ ".less",
118
+ ".vue",
119
+ ".svelte",
120
+ ".json",
121
+ ".jsonc",
122
+ ".yaml",
123
+ ".yml",
124
+ ".toml",
125
+ ".xml",
126
+ ".md",
127
+ ".mdx",
128
+ ".graphql",
129
+ ".gql",
130
+ ],
131
+ async enabled() {
132
+ const configs = ["biome.json", "biome.jsonc"]
133
+ for (const config of configs) {
134
+ const found = await Filesystem.findUp(config, Instance.directory, Instance.worktree)
135
+ if (found.length > 0) {
136
+ return true
137
+ }
138
+ }
139
+ return false
140
+ },
141
+ }
142
+
143
+ export const zig: Info = {
144
+ name: "zig",
145
+ command: ["zig", "fmt", "$FILE"],
146
+ extensions: [".zig", ".zon"],
147
+ async enabled() {
148
+ return Bun.which("zig") !== null
149
+ },
150
+ }
151
+
152
+ export const clang: Info = {
153
+ name: "clang-format",
154
+ command: ["clang-format", "-i", "$FILE"],
155
+ extensions: [".c", ".cc", ".cpp", ".cxx", ".c++", ".h", ".hh", ".hpp", ".hxx", ".h++", ".ino", ".C", ".H"],
156
+ async enabled() {
157
+ const items = await Filesystem.findUp(".clang-format", Instance.directory, Instance.worktree)
158
+ return items.length > 0
159
+ },
160
+ }
161
+
162
+ export const ktlint: Info = {
163
+ name: "ktlint",
164
+ command: ["ktlint", "-F", "$FILE"],
165
+ extensions: [".kt", ".kts"],
166
+ async enabled() {
167
+ return Bun.which("ktlint") !== null
168
+ },
169
+ }
170
+
171
+ export const ruff: Info = {
172
+ name: "ruff",
173
+ command: ["ruff", "format", "$FILE"],
174
+ extensions: [".py", ".pyi"],
175
+ async enabled() {
176
+ if (!Bun.which("ruff")) return false
177
+ const configs = ["pyproject.toml", "ruff.toml", ".ruff.toml"]
178
+ for (const config of configs) {
179
+ const found = await Filesystem.findUp(config, Instance.directory, Instance.worktree)
180
+ if (found.length > 0) {
181
+ if (config === "pyproject.toml") {
182
+ const content = await Bun.file(found[0]).text()
183
+ if (content.includes("[tool.ruff]")) return true
184
+ } else {
185
+ return true
186
+ }
187
+ }
188
+ }
189
+ const deps = ["requirements.txt", "pyproject.toml", "Pipfile"]
190
+ for (const dep of deps) {
191
+ const found = await Filesystem.findUp(dep, Instance.directory, Instance.worktree)
192
+ if (found.length > 0) {
193
+ const content = await Bun.file(found[0]).text()
194
+ if (content.includes("ruff")) return true
195
+ }
196
+ }
197
+ return false
198
+ },
199
+ }
200
+
201
+ export const rlang: Info = {
202
+ name: "air",
203
+ command: ["air", "format", "$FILE"],
204
+ extensions: [".R"],
205
+ async enabled() {
206
+ const airPath = Bun.which("air")
207
+ if (airPath == null) return false
208
+
209
+ try {
210
+ const proc = Bun.spawn(["air", "--help"], {
211
+ stdout: "pipe",
212
+ stderr: "pipe",
213
+ })
214
+ await proc.exited
215
+ const output = await readableStreamToText(proc.stdout)
216
+
217
+ // Check for "Air: An R language server and formatter"
218
+ const firstLine = output.split("\n")[0]
219
+ const hasR = firstLine.includes("R language")
220
+ const hasFormatter = firstLine.includes("formatter")
221
+ return hasR && hasFormatter
222
+ } catch (error) {
223
+ return false
224
+ }
225
+ },
226
+ }
227
+
228
+ export const uvformat: Info = {
229
+ name: "uv format",
230
+ command: ["uv", "format", "--", "$FILE"],
231
+ extensions: [".py", ".pyi"],
232
+ async enabled() {
233
+ if (await ruff.enabled()) return false
234
+ if (Bun.which("uv") !== null) {
235
+ const proc = Bun.spawn(["uv", "format", "--help"], { stderr: "pipe", stdout: "pipe" })
236
+ const code = await proc.exited
237
+ return code === 0
238
+ }
239
+ return false
240
+ },
241
+ }
242
+
243
+ export const rubocop: Info = {
244
+ name: "rubocop",
245
+ command: ["rubocop", "--autocorrect", "$FILE"],
246
+ extensions: [".rb", ".rake", ".gemspec", ".ru"],
247
+ async enabled() {
248
+ return Bun.which("rubocop") !== null
249
+ },
250
+ }
251
+
252
+ export const standardrb: Info = {
253
+ name: "standardrb",
254
+ command: ["standardrb", "--fix", "$FILE"],
255
+ extensions: [".rb", ".rake", ".gemspec", ".ru"],
256
+ async enabled() {
257
+ return Bun.which("standardrb") !== null
258
+ },
259
+ }
260
+
261
+ export const htmlbeautifier: Info = {
262
+ name: "htmlbeautifier",
263
+ command: ["htmlbeautifier", "$FILE"],
264
+ extensions: [".erb", ".html.erb"],
265
+ async enabled() {
266
+ return Bun.which("htmlbeautifier") !== null
267
+ },
268
+ }
269
+
270
+ export const dart: Info = {
271
+ name: "dart",
272
+ command: ["dart", "format", "$FILE"],
273
+ extensions: [".dart"],
274
+ async enabled() {
275
+ return Bun.which("dart") !== null
276
+ },
277
+ }
278
+
279
+ export const ocamlformat: Info = {
280
+ name: "ocamlformat",
281
+ command: ["ocamlformat", "-i", "$FILE"],
282
+ extensions: [".ml", ".mli"],
283
+ async enabled() {
284
+ if (!Bun.which("ocamlformat")) return false
285
+ const items = await Filesystem.findUp(".ocamlformat", Instance.directory, Instance.worktree)
286
+ return items.length > 0
287
+ },
288
+ }
289
+
290
+ export const terraform: Info = {
291
+ name: "terraform",
292
+ command: ["terraform", "fmt", "$FILE"],
293
+ extensions: [".tf", ".tfvars"],
294
+ async enabled() {
295
+ return Bun.which("terraform") !== null
296
+ },
297
+ }
298
+
299
+ export const latexindent: Info = {
300
+ name: "latexindent",
301
+ command: ["latexindent", "-w", "-s", "$FILE"],
302
+ extensions: [".tex"],
303
+ async enabled() {
304
+ return Bun.which("latexindent") !== null
305
+ },
306
+ }
307
+
308
+ export const gleam: Info = {
309
+ name: "gleam",
310
+ command: ["gleam", "format", "$FILE"],
311
+ extensions: [".gleam"],
312
+ async enabled() {
313
+ return Bun.which("gleam") !== null
314
+ },
315
+ }
@@ -0,0 +1,137 @@
1
+ import { Bus } from "../bus"
2
+ import { File } from "../file"
3
+ import { Log } from "../util/log"
4
+ import path from "path"
5
+ import z from "zod"
6
+
7
+ import * as Formatter from "./formatter"
8
+ import { Config } from "../config/config"
9
+ import { mergeDeep } from "remeda"
10
+ import { Instance } from "../project/instance"
11
+
12
+ export namespace Format {
13
+ const log = Log.create({ service: "format" })
14
+
15
+ export const Status = z
16
+ .object({
17
+ name: z.string(),
18
+ extensions: z.string().array(),
19
+ enabled: z.boolean(),
20
+ })
21
+ .meta({
22
+ ref: "FormatterStatus",
23
+ })
24
+ export type Status = z.infer<typeof Status>
25
+
26
+ const state = Instance.state(async () => {
27
+ const enabled: Record<string, boolean> = {}
28
+ const cfg = await Config.get()
29
+
30
+ const formatters: Record<string, Formatter.Info> = {}
31
+ if (cfg.formatter === false) {
32
+ log.info("all formatters are disabled")
33
+ return {
34
+ enabled,
35
+ formatters,
36
+ }
37
+ }
38
+
39
+ for (const item of Object.values(Formatter)) {
40
+ formatters[item.name] = item
41
+ }
42
+ for (const [name, item] of Object.entries(cfg.formatter ?? {})) {
43
+ if (item.disabled) {
44
+ delete formatters[name]
45
+ continue
46
+ }
47
+ const result: Formatter.Info = mergeDeep(formatters[name] ?? {}, {
48
+ command: [],
49
+ extensions: [],
50
+ ...item,
51
+ })
52
+
53
+ if (result.command.length === 0) continue
54
+
55
+ result.enabled = async () => true
56
+ result.name = name
57
+ formatters[name] = result
58
+ }
59
+
60
+ return {
61
+ enabled,
62
+ formatters,
63
+ }
64
+ })
65
+
66
+ async function isEnabled(item: Formatter.Info) {
67
+ const s = await state()
68
+ let status = s.enabled[item.name]
69
+ if (status === undefined) {
70
+ status = await item.enabled()
71
+ s.enabled[item.name] = status
72
+ }
73
+ return status
74
+ }
75
+
76
+ async function getFormatter(ext: string) {
77
+ const formatters = await state().then((x) => x.formatters)
78
+ const result = []
79
+ for (const item of Object.values(formatters)) {
80
+ log.info("checking", { name: item.name, ext })
81
+ if (!item.extensions.includes(ext)) continue
82
+ if (!(await isEnabled(item))) continue
83
+ log.info("enabled", { name: item.name, ext })
84
+ result.push(item)
85
+ }
86
+ return result
87
+ }
88
+
89
+ export async function status() {
90
+ const s = await state()
91
+ const result: Status[] = []
92
+ for (const formatter of Object.values(s.formatters)) {
93
+ const enabled = await isEnabled(formatter)
94
+ result.push({
95
+ name: formatter.name,
96
+ extensions: formatter.extensions,
97
+ enabled,
98
+ })
99
+ }
100
+ return result
101
+ }
102
+
103
+ export function init() {
104
+ log.info("init")
105
+ Bus.subscribe(File.Event.Edited, async (payload) => {
106
+ const file = payload.properties.file
107
+ log.info("formatting", { file })
108
+ const ext = path.extname(file)
109
+
110
+ for (const item of await getFormatter(ext)) {
111
+ log.info("running", { command: item.command })
112
+ try {
113
+ const proc = Bun.spawn({
114
+ cmd: item.command.map((x) => x.replace("$FILE", file)),
115
+ cwd: Instance.directory,
116
+ env: { ...process.env, ...item.environment },
117
+ stdout: "ignore",
118
+ stderr: "ignore",
119
+ })
120
+ const exit = await proc.exited
121
+ if (exit !== 0)
122
+ log.error("failed", {
123
+ command: item.command,
124
+ ...item.environment,
125
+ })
126
+ } catch (error) {
127
+ log.error("failed to format file", {
128
+ error,
129
+ command: item.command,
130
+ ...item.environment,
131
+ file,
132
+ })
133
+ }
134
+ }
135
+ })
136
+ }
137
+ }
@@ -0,0 +1,52 @@
1
+ import fs from "fs/promises"
2
+ import { xdgData, xdgCache, xdgConfig, xdgState } from "xdg-basedir"
3
+ import path from "path"
4
+ import os from "os"
5
+
6
+ const app = "rird"
7
+
8
+ const data = path.join(xdgData!, app)
9
+ const cache = path.join(xdgCache!, app)
10
+ const config = path.join(xdgConfig!, app)
11
+ const state = path.join(xdgState!, app)
12
+
13
+ export namespace Global {
14
+ export const Path = {
15
+ home: os.homedir(),
16
+ data,
17
+ bin: path.join(data, "bin"),
18
+ log: path.join(data, "log"),
19
+ cache,
20
+ config,
21
+ state,
22
+ } as const
23
+ }
24
+
25
+ await Promise.all([
26
+ fs.mkdir(Global.Path.data, { recursive: true }),
27
+ fs.mkdir(Global.Path.config, { recursive: true }),
28
+ fs.mkdir(Global.Path.state, { recursive: true }),
29
+ fs.mkdir(Global.Path.log, { recursive: true }),
30
+ fs.mkdir(Global.Path.bin, { recursive: true }),
31
+ ])
32
+
33
+ const CACHE_VERSION = "14"
34
+
35
+ const version = await Bun.file(path.join(Global.Path.cache, "version"))
36
+ .text()
37
+ .catch(() => "0")
38
+
39
+ if (version !== CACHE_VERSION) {
40
+ try {
41
+ const contents = await fs.readdir(Global.Path.cache)
42
+ await Promise.all(
43
+ contents.map((item) =>
44
+ fs.rm(path.join(Global.Path.cache, item), {
45
+ recursive: true,
46
+ force: true,
47
+ }),
48
+ ),
49
+ )
50
+ } catch (e) {}
51
+ await Bun.file(path.join(Global.Path.cache, "version")).write(CACHE_VERSION)
52
+ }
package/src/id/id.ts ADDED
@@ -0,0 +1,73 @@
1
+ import z from "zod"
2
+ import { randomBytes } from "crypto"
3
+
4
+ export namespace Identifier {
5
+ const prefixes = {
6
+ session: "ses",
7
+ message: "msg",
8
+ permission: "per",
9
+ user: "usr",
10
+ part: "prt",
11
+ pty: "pty",
12
+ } as const
13
+
14
+ export function schema(prefix: keyof typeof prefixes) {
15
+ return z.string().startsWith(prefixes[prefix])
16
+ }
17
+
18
+ const LENGTH = 26
19
+
20
+ // State for monotonic ID generation
21
+ let lastTimestamp = 0
22
+ let counter = 0
23
+
24
+ export function ascending(prefix: keyof typeof prefixes, given?: string) {
25
+ return generateID(prefix, false, given)
26
+ }
27
+
28
+ export function descending(prefix: keyof typeof prefixes, given?: string) {
29
+ return generateID(prefix, true, given)
30
+ }
31
+
32
+ function generateID(prefix: keyof typeof prefixes, descending: boolean, given?: string): string {
33
+ if (!given) {
34
+ return create(prefix, descending)
35
+ }
36
+
37
+ if (!given.startsWith(prefixes[prefix])) {
38
+ throw new Error(`ID ${given} does not start with ${prefixes[prefix]}`)
39
+ }
40
+ return given
41
+ }
42
+
43
+ function randomBase62(length: number): string {
44
+ const chars = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"
45
+ let result = ""
46
+ const bytes = randomBytes(length)
47
+ for (let i = 0; i < length; i++) {
48
+ result += chars[bytes[i] % 62]
49
+ }
50
+ return result
51
+ }
52
+
53
+ export function create(prefix: keyof typeof prefixes, descending: boolean, timestamp?: number): string {
54
+ const currentTimestamp = timestamp ?? Date.now()
55
+
56
+ if (currentTimestamp !== lastTimestamp) {
57
+ lastTimestamp = currentTimestamp
58
+ counter = 0
59
+ }
60
+ counter++
61
+
62
+ let now = BigInt(currentTimestamp) * BigInt(0x1000) + BigInt(counter)
63
+
64
+ now = descending ? ~now : now
65
+
66
+ const timeBytes = Buffer.alloc(6)
67
+ for (let i = 0; i < 6; i++) {
68
+ timeBytes[i] = Number((now >> BigInt(40 - 8 * i)) & BigInt(0xff))
69
+ }
70
+
71
+ return prefixes[prefix] + "_" + timeBytes.toString("hex") + randomBase62(LENGTH - 12)
72
+ }
73
+ }
@@ -0,0 +1,76 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import { spawn } from "bun"
4
+ import z from "zod"
5
+ import { NamedError } from "@opencode-ai/util/error"
6
+ import { Log } from "../util/log"
7
+
8
+ const SUPPORTED_IDES = [
9
+ { name: "Windsurf" as const, cmd: "windsurf" },
10
+ { name: "Visual Studio Code - Insiders" as const, cmd: "code-insiders" },
11
+ { name: "Visual Studio Code" as const, cmd: "code" },
12
+ { name: "Cursor" as const, cmd: "cursor" },
13
+ { name: "VSCodium" as const, cmd: "codium" },
14
+ ]
15
+
16
+ export namespace Ide {
17
+ const log = Log.create({ service: "ide" })
18
+
19
+ export const Event = {
20
+ Installed: BusEvent.define(
21
+ "ide.installed",
22
+ z.object({
23
+ ide: z.string(),
24
+ }),
25
+ ),
26
+ }
27
+
28
+ export const AlreadyInstalledError = NamedError.create("AlreadyInstalledError", z.object({}))
29
+
30
+ export const InstallFailedError = NamedError.create(
31
+ "InstallFailedError",
32
+ z.object({
33
+ stderr: z.string(),
34
+ }),
35
+ )
36
+
37
+ export function ide() {
38
+ if (process.env["TERM_PROGRAM"] === "vscode") {
39
+ const v = process.env["GIT_ASKPASS"]
40
+ for (const ide of SUPPORTED_IDES) {
41
+ if (v?.includes(ide.name)) return ide.name
42
+ }
43
+ }
44
+ return "unknown"
45
+ }
46
+
47
+ export function alreadyInstalled() {
48
+ return process.env["OPENCODE_CALLER"] === "vscode" || process.env["OPENCODE_CALLER"] === "vscode-insiders"
49
+ }
50
+
51
+ export async function install(ide: (typeof SUPPORTED_IDES)[number]["name"]) {
52
+ const cmd = SUPPORTED_IDES.find((i) => i.name === ide)?.cmd
53
+ if (!cmd) throw new Error(`Unknown IDE: ${ide}`)
54
+
55
+ const p = spawn([cmd, "--install-extension", "sst-dev.opencode"], {
56
+ stdout: "pipe",
57
+ stderr: "pipe",
58
+ })
59
+ await p.exited
60
+ const stdout = await new Response(p.stdout).text()
61
+ const stderr = await new Response(p.stderr).text()
62
+
63
+ log.info("installed", {
64
+ ide,
65
+ stdout,
66
+ stderr,
67
+ })
68
+
69
+ if (p.exitCode !== 0) {
70
+ throw new InstallFailedError({ stderr })
71
+ }
72
+ if (stdout.includes("already installed")) {
73
+ throw new AlreadyInstalledError({})
74
+ }
75
+ }
76
+ }