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,393 @@
1
+ // Ripgrep utility functions
2
+ import path from "path"
3
+ import { Global } from "../global"
4
+ import fs from "fs/promises"
5
+ import z from "zod"
6
+ import { NamedError } from "@opencode-ai/util/error"
7
+ import { lazy } from "../util/lazy"
8
+ import { $ } from "bun"
9
+
10
+ import { ZipReader, BlobReader, BlobWriter } from "@zip.js/zip.js"
11
+ import { Log } from "@/util/log"
12
+
13
+ export namespace Ripgrep {
14
+ const log = Log.create({ service: "ripgrep" })
15
+ const Stats = z.object({
16
+ elapsed: z.object({
17
+ secs: z.number(),
18
+ nanos: z.number(),
19
+ human: z.string(),
20
+ }),
21
+ searches: z.number(),
22
+ searches_with_match: z.number(),
23
+ bytes_searched: z.number(),
24
+ bytes_printed: z.number(),
25
+ matched_lines: z.number(),
26
+ matches: z.number(),
27
+ })
28
+
29
+ const Begin = z.object({
30
+ type: z.literal("begin"),
31
+ data: z.object({
32
+ path: z.object({
33
+ text: z.string(),
34
+ }),
35
+ }),
36
+ })
37
+
38
+ export const Match = z.object({
39
+ type: z.literal("match"),
40
+ data: z.object({
41
+ path: z.object({
42
+ text: z.string(),
43
+ }),
44
+ lines: z.object({
45
+ text: z.string(),
46
+ }),
47
+ line_number: z.number(),
48
+ absolute_offset: z.number(),
49
+ submatches: z.array(
50
+ z.object({
51
+ match: z.object({
52
+ text: z.string(),
53
+ }),
54
+ start: z.number(),
55
+ end: z.number(),
56
+ }),
57
+ ),
58
+ }),
59
+ })
60
+
61
+ const End = z.object({
62
+ type: z.literal("end"),
63
+ data: z.object({
64
+ path: z.object({
65
+ text: z.string(),
66
+ }),
67
+ binary_offset: z.number().nullable(),
68
+ stats: Stats,
69
+ }),
70
+ })
71
+
72
+ const Summary = z.object({
73
+ type: z.literal("summary"),
74
+ data: z.object({
75
+ elapsed_total: z.object({
76
+ human: z.string(),
77
+ nanos: z.number(),
78
+ secs: z.number(),
79
+ }),
80
+ stats: Stats,
81
+ }),
82
+ })
83
+
84
+ const Result = z.union([Begin, Match, End, Summary])
85
+
86
+ export type Result = z.infer<typeof Result>
87
+ export type Match = z.infer<typeof Match>
88
+ export type Begin = z.infer<typeof Begin>
89
+ export type End = z.infer<typeof End>
90
+ export type Summary = z.infer<typeof Summary>
91
+ const PLATFORM = {
92
+ "arm64-darwin": { platform: "aarch64-apple-darwin", extension: "tar.gz" },
93
+ "arm64-linux": {
94
+ platform: "aarch64-unknown-linux-gnu",
95
+ extension: "tar.gz",
96
+ },
97
+ "x64-darwin": { platform: "x86_64-apple-darwin", extension: "tar.gz" },
98
+ "x64-linux": { platform: "x86_64-unknown-linux-musl", extension: "tar.gz" },
99
+ "x64-win32": { platform: "x86_64-pc-windows-msvc", extension: "zip" },
100
+ } as const
101
+
102
+ export const ExtractionFailedError = NamedError.create(
103
+ "RipgrepExtractionFailedError",
104
+ z.object({
105
+ filepath: z.string(),
106
+ stderr: z.string(),
107
+ }),
108
+ )
109
+
110
+ export const UnsupportedPlatformError = NamedError.create(
111
+ "RipgrepUnsupportedPlatformError",
112
+ z.object({
113
+ platform: z.string(),
114
+ }),
115
+ )
116
+
117
+ export const DownloadFailedError = NamedError.create(
118
+ "RipgrepDownloadFailedError",
119
+ z.object({
120
+ url: z.string(),
121
+ status: z.number(),
122
+ }),
123
+ )
124
+
125
+ const state = lazy(async () => {
126
+ let filepath = Bun.which("rg")
127
+ if (filepath) return { filepath }
128
+ filepath = path.join(Global.Path.bin, "rg" + (process.platform === "win32" ? ".exe" : ""))
129
+
130
+ const file = Bun.file(filepath)
131
+ if (!(await file.exists())) {
132
+ const platformKey = `${process.arch}-${process.platform}` as keyof typeof PLATFORM
133
+ const config = PLATFORM[platformKey]
134
+ if (!config) throw new UnsupportedPlatformError({ platform: platformKey })
135
+
136
+ const version = "14.1.1"
137
+ const filename = `ripgrep-${version}-${config.platform}.${config.extension}`
138
+ const url = `https://github.com/BurntSushi/ripgrep/releases/download/${version}/${filename}`
139
+
140
+ const response = await fetch(url)
141
+ if (!response.ok) throw new DownloadFailedError({ url, status: response.status })
142
+
143
+ const buffer = await response.arrayBuffer()
144
+ const archivePath = path.join(Global.Path.bin, filename)
145
+ await Bun.write(archivePath, buffer)
146
+ if (config.extension === "tar.gz") {
147
+ const args = ["tar", "-xzf", archivePath, "--strip-components=1"]
148
+
149
+ if (platformKey.endsWith("-darwin")) args.push("--include=*/rg")
150
+ if (platformKey.endsWith("-linux")) args.push("--wildcards", "*/rg")
151
+
152
+ const proc = Bun.spawn(args, {
153
+ cwd: Global.Path.bin,
154
+ stderr: "pipe",
155
+ stdout: "pipe",
156
+ })
157
+ await proc.exited
158
+ if (proc.exitCode !== 0)
159
+ throw new ExtractionFailedError({
160
+ filepath,
161
+ stderr: await Bun.readableStreamToText(proc.stderr),
162
+ })
163
+ }
164
+ if (config.extension === "zip") {
165
+ if (config.extension === "zip") {
166
+ const zipFileReader = new ZipReader(new BlobReader(new Blob([await Bun.file(archivePath).arrayBuffer()])))
167
+ const entries = await zipFileReader.getEntries()
168
+ let rgEntry: any
169
+ for (const entry of entries) {
170
+ if (entry.filename.endsWith("rg.exe")) {
171
+ rgEntry = entry
172
+ break
173
+ }
174
+ }
175
+
176
+ if (!rgEntry) {
177
+ throw new ExtractionFailedError({
178
+ filepath: archivePath,
179
+ stderr: "rg.exe not found in zip archive",
180
+ })
181
+ }
182
+
183
+ const rgBlob = await rgEntry.getData(new BlobWriter())
184
+ if (!rgBlob) {
185
+ throw new ExtractionFailedError({
186
+ filepath: archivePath,
187
+ stderr: "Failed to extract rg.exe from zip archive",
188
+ })
189
+ }
190
+ await Bun.write(filepath, await rgBlob.arrayBuffer())
191
+ await zipFileReader.close()
192
+ }
193
+ }
194
+ await fs.unlink(archivePath)
195
+ if (!platformKey.endsWith("-win32")) await fs.chmod(filepath, 0o755)
196
+ }
197
+
198
+ return {
199
+ filepath,
200
+ }
201
+ })
202
+
203
+ export async function filepath() {
204
+ const { filepath } = await state()
205
+ return filepath
206
+ }
207
+
208
+ export async function* files(input: { cwd: string; glob?: string[] }) {
209
+ const args = [await filepath(), "--files", "--follow", "--hidden", "--glob=!.git/*"]
210
+ if (input.glob) {
211
+ for (const g of input.glob) {
212
+ args.push(`--glob=${g}`)
213
+ }
214
+ }
215
+
216
+ // Bun.spawn should throw this, but it incorrectly reports that the executable does not exist.
217
+ // See https://github.com/oven-sh/bun/issues/24012
218
+ if (!(await fs.stat(input.cwd).catch(() => undefined))?.isDirectory()) {
219
+ throw Object.assign(new Error(`No such file or directory: '${input.cwd}'`), {
220
+ code: "ENOENT",
221
+ errno: -2,
222
+ path: input.cwd,
223
+ })
224
+ }
225
+
226
+ const proc = Bun.spawn(args, {
227
+ cwd: input.cwd,
228
+ stdout: "pipe",
229
+ stderr: "ignore",
230
+ maxBuffer: 1024 * 1024 * 20,
231
+ })
232
+
233
+ const reader = proc.stdout.getReader()
234
+ const decoder = new TextDecoder()
235
+ let buffer = ""
236
+
237
+ try {
238
+ while (true) {
239
+ const { done, value } = await reader.read()
240
+ if (done) break
241
+
242
+ buffer += decoder.decode(value, { stream: true })
243
+ // Handle both Unix (\n) and Windows (\r\n) line endings
244
+ const lines = buffer.split(/\r?\n/)
245
+ buffer = lines.pop() || ""
246
+
247
+ for (const line of lines) {
248
+ if (line) yield line
249
+ }
250
+ }
251
+
252
+ if (buffer) yield buffer
253
+ } finally {
254
+ reader.releaseLock()
255
+ await proc.exited
256
+ }
257
+ }
258
+
259
+ export async function tree(input: { cwd: string; limit?: number }) {
260
+ log.info("tree", input)
261
+ const files = await Array.fromAsync(Ripgrep.files({ cwd: input.cwd }))
262
+ interface Node {
263
+ path: string[]
264
+ children: Node[]
265
+ }
266
+
267
+ function getPath(node: Node, parts: string[], create: boolean) {
268
+ if (parts.length === 0) return node
269
+ let current = node
270
+ for (const part of parts) {
271
+ let existing = current.children.find((x) => x.path.at(-1) === part)
272
+ if (!existing) {
273
+ if (!create) return
274
+ existing = {
275
+ path: current.path.concat(part),
276
+ children: [],
277
+ }
278
+ current.children.push(existing)
279
+ }
280
+ current = existing
281
+ }
282
+ return current
283
+ }
284
+
285
+ const root: Node = {
286
+ path: [],
287
+ children: [],
288
+ }
289
+ for (const file of files) {
290
+ if (file.includes(".opencode")) continue
291
+ const parts = file.split(path.sep)
292
+ getPath(root, parts, true)
293
+ }
294
+
295
+ function sort(node: Node) {
296
+ node.children.sort((a, b) => {
297
+ if (!a.children.length && b.children.length) return 1
298
+ if (!b.children.length && a.children.length) return -1
299
+ return a.path.at(-1)!.localeCompare(b.path.at(-1)!)
300
+ })
301
+ for (const child of node.children) {
302
+ sort(child)
303
+ }
304
+ }
305
+ sort(root)
306
+
307
+ let current = [root]
308
+ const result: Node = {
309
+ path: [],
310
+ children: [],
311
+ }
312
+
313
+ let processed = 0
314
+ const limit = input.limit ?? 50
315
+ while (current.length > 0) {
316
+ const next = []
317
+ for (const node of current) {
318
+ if (node.children.length) next.push(...node.children)
319
+ }
320
+ const max = Math.max(...current.map((x) => x.children.length))
321
+ for (let i = 0; i < max && processed < limit; i++) {
322
+ for (const node of current) {
323
+ const child = node.children[i]
324
+ if (!child) continue
325
+ getPath(result, child.path, true)
326
+ processed++
327
+ if (processed >= limit) break
328
+ }
329
+ }
330
+ if (processed >= limit) {
331
+ for (const node of [...current, ...next]) {
332
+ const compare = getPath(result, node.path, false)
333
+ if (!compare) continue
334
+ if (compare?.children.length !== node.children.length) {
335
+ const diff = node.children.length - compare.children.length
336
+ compare.children.push({
337
+ path: compare.path.concat(`[${diff} truncated]`),
338
+ children: [],
339
+ })
340
+ }
341
+ }
342
+ break
343
+ }
344
+ current = next
345
+ }
346
+
347
+ const lines: string[] = []
348
+
349
+ function render(node: Node, depth: number) {
350
+ const indent = "\t".repeat(depth)
351
+ lines.push(indent + node.path.at(-1) + (node.children.length ? "/" : ""))
352
+ for (const child of node.children) {
353
+ render(child, depth + 1)
354
+ }
355
+ }
356
+ result.children.map((x) => render(x, 0))
357
+
358
+ return lines.join("\n")
359
+ }
360
+
361
+ export async function search(input: { cwd: string; pattern: string; glob?: string[]; limit?: number }) {
362
+ const args = [`${await filepath()}`, "--json", "--hidden", "--glob='!.git/*'"]
363
+
364
+ if (input.glob) {
365
+ for (const g of input.glob) {
366
+ args.push(`--glob=${g}`)
367
+ }
368
+ }
369
+
370
+ if (input.limit) {
371
+ args.push(`--max-count=${input.limit}`)
372
+ }
373
+
374
+ args.push("--")
375
+ args.push(input.pattern)
376
+
377
+ const command = args.join(" ")
378
+ const result = await $`${{ raw: command }}`.cwd(input.cwd).quiet().nothrow()
379
+ if (result.exitCode !== 0) {
380
+ return []
381
+ }
382
+
383
+ // Handle both Unix (\n) and Windows (\r\n) line endings
384
+ const lines = result.text().trim().split(/\r?\n/).filter(Boolean)
385
+ // Parse JSON lines from ripgrep output
386
+
387
+ return lines
388
+ .map((line) => JSON.parse(line))
389
+ .map((parsed) => Result.parse(parsed))
390
+ .filter((r) => r.type === "match")
391
+ .map((r) => r.data)
392
+ }
393
+ }
@@ -0,0 +1,64 @@
1
+ import { Instance } from "../project/instance"
2
+ import { Log } from "../util/log"
3
+
4
+ export namespace FileTime {
5
+ const log = Log.create({ service: "file.time" })
6
+ // Per-session read times plus per-file write locks.
7
+ // All tools that overwrite existing files should run their
8
+ // assert/read/write/update sequence inside withLock(filepath, ...)
9
+ // so concurrent writes to the same file are serialized.
10
+ export const state = Instance.state(() => {
11
+ const read: {
12
+ [sessionID: string]: {
13
+ [path: string]: Date | undefined
14
+ }
15
+ } = {}
16
+ const locks = new Map<string, Promise<void>>()
17
+ return {
18
+ read,
19
+ locks,
20
+ }
21
+ })
22
+
23
+ export function read(sessionID: string, file: string) {
24
+ log.info("read", { sessionID, file })
25
+ const { read } = state()
26
+ read[sessionID] = read[sessionID] || {}
27
+ read[sessionID][file] = new Date()
28
+ }
29
+
30
+ export function get(sessionID: string, file: string) {
31
+ return state().read[sessionID]?.[file]
32
+ }
33
+
34
+ export async function withLock<T>(filepath: string, fn: () => Promise<T>): Promise<T> {
35
+ const current = state()
36
+ const currentLock = current.locks.get(filepath) ?? Promise.resolve()
37
+ let release: () => void = () => {}
38
+ const nextLock = new Promise<void>((resolve) => {
39
+ release = resolve
40
+ })
41
+ const chained = currentLock.then(() => nextLock)
42
+ current.locks.set(filepath, chained)
43
+ await currentLock
44
+ try {
45
+ return await fn()
46
+ } finally {
47
+ release()
48
+ if (current.locks.get(filepath) === chained) {
49
+ current.locks.delete(filepath)
50
+ }
51
+ }
52
+ }
53
+
54
+ export async function assert(sessionID: string, filepath: string) {
55
+ const time = get(sessionID, filepath)
56
+ if (!time) throw new Error(`You must read the file ${filepath} before overwriting it. Use the Read tool first`)
57
+ const stats = await Bun.file(filepath).stat()
58
+ if (stats.mtime.getTime() > time.getTime()) {
59
+ throw new Error(
60
+ `File ${filepath} has been modified since it was last read.\nLast modification: ${stats.mtime.toISOString()}\nLast read: ${time.toISOString()}\n\nPlease read the file again before modifying it.`,
61
+ )
62
+ }
63
+ }
64
+ }
@@ -0,0 +1,103 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import z from "zod"
4
+ import { Instance } from "../project/instance"
5
+ import { Log } from "../util/log"
6
+ import { FileIgnore } from "./ignore"
7
+ import { Config } from "../config/config"
8
+ import path from "path"
9
+ // @ts-ignore
10
+ import { createWrapper } from "@parcel/watcher/wrapper"
11
+ import { lazy } from "@/util/lazy"
12
+ import type ParcelWatcher from "@parcel/watcher"
13
+ import { $ } from "bun"
14
+ import { Flag } from "@/flag/flag"
15
+
16
+ declare const OPENCODE_LIBC: string | undefined
17
+
18
+ export namespace FileWatcher {
19
+ const log = Log.create({ service: "file.watcher" })
20
+
21
+ export const Event = {
22
+ Updated: BusEvent.define(
23
+ "file.watcher.updated",
24
+ z.object({
25
+ file: z.string(),
26
+ event: z.union([z.literal("add"), z.literal("change"), z.literal("unlink")]),
27
+ }),
28
+ ),
29
+ }
30
+
31
+ const watcher = lazy(() => {
32
+ const binding = require(
33
+ `@parcel/watcher-${process.platform}-${process.arch}${process.platform === "linux" ? `-${OPENCODE_LIBC || "glibc"}` : ""}`,
34
+ )
35
+ return createWrapper(binding) as typeof import("@parcel/watcher")
36
+ })
37
+
38
+ const state = Instance.state(
39
+ async () => {
40
+ if (Instance.project.vcs !== "git") return {}
41
+ log.info("init")
42
+ const cfg = await Config.get()
43
+ const backend = (() => {
44
+ if (process.platform === "win32") return "windows"
45
+ if (process.platform === "darwin") return "fs-events"
46
+ if (process.platform === "linux") return "inotify"
47
+ })()
48
+ if (!backend) {
49
+ log.error("watcher backend not supported", { platform: process.platform })
50
+ return {}
51
+ }
52
+ log.info("watcher backend", { platform: process.platform, backend })
53
+ const subscribe: ParcelWatcher.SubscribeCallback = (err, evts) => {
54
+ if (err) return
55
+ for (const evt of evts) {
56
+ if (evt.type === "create") Bus.publish(Event.Updated, { file: evt.path, event: "add" })
57
+ if (evt.type === "update") Bus.publish(Event.Updated, { file: evt.path, event: "change" })
58
+ if (evt.type === "delete") Bus.publish(Event.Updated, { file: evt.path, event: "unlink" })
59
+ }
60
+ }
61
+
62
+ const subs: ParcelWatcher.AsyncSubscription[] = []
63
+ const cfgIgnores = cfg.watcher?.ignore ?? []
64
+
65
+ if (Flag.RIRD_EXPERIMENTAL_FILEWATCHER) {
66
+ subs.push(
67
+ await watcher().subscribe(Instance.directory, subscribe, {
68
+ ignore: [...FileIgnore.PATTERNS, ...cfgIgnores],
69
+ backend,
70
+ }),
71
+ )
72
+ }
73
+
74
+ const vcsDir = await $`git rev-parse --git-dir`
75
+ .quiet()
76
+ .nothrow()
77
+ .cwd(Instance.worktree)
78
+ .text()
79
+ .then((x) => path.resolve(Instance.worktree, x.trim()))
80
+ if (vcsDir && !cfgIgnores.includes(".git") && !cfgIgnores.includes(vcsDir)) {
81
+ subs.push(
82
+ await watcher().subscribe(vcsDir, subscribe, {
83
+ ignore: ["hooks", "info", "logs", "objects", "refs", "worktrees", "modules", "lfs"],
84
+ backend,
85
+ }),
86
+ )
87
+ }
88
+
89
+ return { subs }
90
+ },
91
+ async (state) => {
92
+ if (!state.subs) return
93
+ await Promise.all(state.subs.map((sub) => sub?.unsubscribe()))
94
+ },
95
+ )
96
+
97
+ export function init() {
98
+ if (Flag.RIRD_EXPERIMENTAL_DISABLE_FILEWATCHER) {
99
+ return
100
+ }
101
+ state()
102
+ }
103
+ }
@@ -0,0 +1,46 @@
1
+ export namespace Flag {
2
+ export const RIRD_AUTO_SHARE = truthy("RIRD_AUTO_SHARE")
3
+ export const RIRD_GIT_BASH_PATH = process.env["RIRD_GIT_BASH_PATH"]
4
+ export const RIRD_CONFIG = process.env["RIRD_CONFIG"]
5
+ export const RIRD_CONFIG_DIR = process.env["RIRD_CONFIG_DIR"]
6
+ export const RIRD_CONFIG_CONTENT = process.env["RIRD_CONFIG_CONTENT"]
7
+ export const RIRD_DISABLE_AUTOUPDATE = truthy("RIRD_DISABLE_AUTOUPDATE")
8
+ export const RIRD_DISABLE_PRUNE = truthy("RIRD_DISABLE_PRUNE")
9
+ export const RIRD_DISABLE_TERMINAL_TITLE = truthy("RIRD_DISABLE_TERMINAL_TITLE")
10
+ export const RIRD_PERMISSION = process.env["RIRD_PERMISSION"]
11
+ export const RIRD_DISABLE_DEFAULT_PLUGINS = truthy("RIRD_DISABLE_DEFAULT_PLUGINS")
12
+ export const RIRD_DISABLE_LSP_DOWNLOAD = truthy("RIRD_DISABLE_LSP_DOWNLOAD")
13
+ export const RIRD_ENABLE_EXPERIMENTAL_MODELS = truthy("RIRD_ENABLE_EXPERIMENTAL_MODELS")
14
+ export const RIRD_DISABLE_AUTOCOMPACT = truthy("RIRD_DISABLE_AUTOCOMPACT")
15
+ export const RIRD_DISABLE_MODELS_FETCH = truthy("RIRD_DISABLE_MODELS_FETCH")
16
+ export const RIRD_FAKE_VCS = process.env["RIRD_FAKE_VCS"]
17
+ export const RIRD_CLIENT = process.env["RIRD_CLIENT"] ?? "cli"
18
+
19
+ // Experimental
20
+ export const RIRD_EXPERIMENTAL = truthy("RIRD_EXPERIMENTAL")
21
+ export const RIRD_EXPERIMENTAL_FILEWATCHER = truthy("RIRD_EXPERIMENTAL_FILEWATCHER")
22
+ export const RIRD_EXPERIMENTAL_DISABLE_FILEWATCHER = truthy("RIRD_EXPERIMENTAL_DISABLE_FILEWATCHER")
23
+ export const RIRD_EXPERIMENTAL_ICON_DISCOVERY =
24
+ RIRD_EXPERIMENTAL || truthy("RIRD_EXPERIMENTAL_ICON_DISCOVERY")
25
+ export const RIRD_EXPERIMENTAL_DISABLE_COPY_ON_SELECT = truthy("RIRD_EXPERIMENTAL_DISABLE_COPY_ON_SELECT")
26
+ export const RIRD_ENABLE_EXA =
27
+ truthy("RIRD_ENABLE_EXA") || RIRD_EXPERIMENTAL || truthy("RIRD_EXPERIMENTAL_EXA")
28
+ export const RIRD_EXPERIMENTAL_BASH_MAX_OUTPUT_LENGTH = number("RIRD_EXPERIMENTAL_BASH_MAX_OUTPUT_LENGTH")
29
+ export const RIRD_EXPERIMENTAL_BASH_DEFAULT_TIMEOUT_MS = number("RIRD_EXPERIMENTAL_BASH_DEFAULT_TIMEOUT_MS")
30
+ export const RIRD_EXPERIMENTAL_OUTPUT_TOKEN_MAX = number("RIRD_EXPERIMENTAL_OUTPUT_TOKEN_MAX")
31
+ export const RIRD_EXPERIMENTAL_OXFMT = RIRD_EXPERIMENTAL || truthy("RIRD_EXPERIMENTAL_OXFMT")
32
+ export const RIRD_EXPERIMENTAL_LSP_TY = truthy("RIRD_EXPERIMENTAL_LSP_TY")
33
+ export const RIRD_EXPERIMENTAL_LSP_TOOL = RIRD_EXPERIMENTAL || truthy("RIRD_EXPERIMENTAL_LSP_TOOL")
34
+
35
+ function truthy(key: string) {
36
+ const value = process.env[key]?.toLowerCase()
37
+ return value === "true" || value === "1"
38
+ }
39
+
40
+ function number(key: string) {
41
+ const value = process.env[key]
42
+ if (!value) return undefined
43
+ const parsed = Number(value)
44
+ return Number.isInteger(parsed) && parsed > 0 ? parsed : undefined
45
+ }
46
+ }