rird 2.1.231 → 2.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (381) hide show
  1. package/AGENTS.md +86 -0
  2. package/COMPLETED_TEST_SUITE.txt +280 -0
  3. package/Dockerfile +18 -0
  4. package/README.md +397 -6
  5. package/RIRD_ERROR_HANDLING_SUMMARY.md +307 -0
  6. package/TESTING.md +512 -0
  7. package/TEST_IMPLEMENTATION_REPORT.md +463 -0
  8. package/TEST_SUITE.md +307 -0
  9. package/TEST_SUMMARY.txt +380 -0
  10. package/bin/rird-perf.js +37 -0
  11. package/bin/rird.js +43 -8
  12. package/bunfig.toml +4 -0
  13. package/create-wrapper.ps1 +51 -0
  14. package/docs/ARCHITECTURE.md +768 -0
  15. package/docs/CLI_REFERENCE.md +681 -0
  16. package/docs/DOCUMENTATION_MANIFEST.md +392 -0
  17. package/docs/INDEX.md +295 -0
  18. package/docs/PRODUCTION_SETUP.md +633 -0
  19. package/docs/TROUBLESHOOTING.md +914 -0
  20. package/facebook_ads_library.png +0 -0
  21. package/nul +0 -0
  22. package/nul`nif +0 -0
  23. package/package.json +104 -15
  24. package/parsers-config.ts +239 -0
  25. package/rird-1.0.199.tgz +0 -0
  26. package/rird-1.0.205.tgz +0 -0
  27. package/script/build-windows.ts +56 -0
  28. package/script/build.ts +165 -0
  29. package/{postinstall.mjs → script/postinstall.mjs} +47 -68
  30. package/script/publish-registries.ts +187 -0
  31. package/script/publish.ts +85 -0
  32. package/script/schema.ts +47 -0
  33. package/src/acp/README.md +164 -0
  34. package/src/acp/agent.ts +1063 -0
  35. package/src/acp/session.ts +101 -0
  36. package/src/acp/types.ts +22 -0
  37. package/src/agent/agent.ts +367 -0
  38. package/src/agent/generate.txt +75 -0
  39. package/src/agent/prompt/compaction.txt +12 -0
  40. package/src/agent/prompt/explore.txt +18 -0
  41. package/src/agent/prompt/summary.txt +10 -0
  42. package/src/agent/prompt/title.txt +36 -0
  43. package/src/auth/index.ts +70 -0
  44. package/src/bun/index.ts +114 -0
  45. package/src/bus/bus-event.ts +43 -0
  46. package/src/bus/global.ts +10 -0
  47. package/src/bus/index.ts +105 -0
  48. package/src/cli/bootstrap.ts +17 -0
  49. package/src/cli/cmd/acp.ts +104 -0
  50. package/src/cli/cmd/activate.ts +50 -0
  51. package/src/cli/cmd/agent.ts +256 -0
  52. package/src/cli/cmd/auth.ts +412 -0
  53. package/src/cli/cmd/cmd.ts +7 -0
  54. package/src/cli/cmd/debug/config.ts +15 -0
  55. package/src/cli/cmd/debug/file.ts +91 -0
  56. package/src/cli/cmd/debug/index.ts +43 -0
  57. package/src/cli/cmd/debug/lsp.ts +48 -0
  58. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  59. package/src/cli/cmd/debug/scrap.ts +15 -0
  60. package/src/cli/cmd/debug/skill.ts +15 -0
  61. package/src/cli/cmd/debug/snapshot.ts +48 -0
  62. package/src/cli/cmd/export.ts +88 -0
  63. package/src/cli/cmd/generate.ts +38 -0
  64. package/src/cli/cmd/github.ts +1400 -0
  65. package/src/cli/cmd/import.ts +98 -0
  66. package/src/cli/cmd/mcp.ts +654 -0
  67. package/src/cli/cmd/models.ts +68 -0
  68. package/src/cli/cmd/pr.ts +112 -0
  69. package/src/cli/cmd/run.ts +434 -0
  70. package/src/cli/cmd/serve.ts +31 -0
  71. package/src/cli/cmd/session.ts +106 -0
  72. package/src/cli/cmd/stats.ts +298 -0
  73. package/src/cli/cmd/tui/app.tsx +694 -0
  74. package/src/cli/cmd/tui/attach.ts +30 -0
  75. package/src/cli/cmd/tui/component/border.tsx +21 -0
  76. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  77. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  78. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  79. package/src/cli/cmd/tui/component/dialog-model.tsx +236 -0
  80. package/src/cli/cmd/tui/component/dialog-provider.tsx +240 -0
  81. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  82. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  83. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  84. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  85. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  86. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  87. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  88. package/src/cli/cmd/tui/component/logo.tsx +48 -0
  89. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +574 -0
  90. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  91. package/src/cli/cmd/tui/component/prompt/index.tsx +1087 -0
  92. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  93. package/src/cli/cmd/tui/component/tips.ts +27 -0
  94. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  95. package/src/cli/cmd/tui/context/args.tsx +14 -0
  96. package/src/cli/cmd/tui/context/directory.ts +13 -0
  97. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  98. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  99. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  100. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  101. package/src/cli/cmd/tui/context/local.tsx +345 -0
  102. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  103. package/src/cli/cmd/tui/context/route.tsx +46 -0
  104. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  105. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  106. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  107. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  108. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  109. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  110. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  111. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  112. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  114. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  115. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  116. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  117. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  118. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  119. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  120. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  121. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  122. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  123. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  124. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  125. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  126. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  127. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  128. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  129. package/src/cli/cmd/tui/context/theme/rird.json +245 -0
  130. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  131. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  132. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  133. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  134. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  135. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  136. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  137. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  138. package/src/cli/cmd/tui/event.ts +40 -0
  139. package/src/cli/cmd/tui/hooks/use-safe-terminal-dimensions.ts +12 -0
  140. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  141. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  142. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  143. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  144. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  145. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  146. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  147. package/src/cli/cmd/tui/routes/session/index.tsx +1876 -0
  148. package/src/cli/cmd/tui/routes/session/sidebar.tsx +320 -0
  149. package/src/cli/cmd/tui/spawn.ts +60 -0
  150. package/src/cli/cmd/tui/thread.ts +142 -0
  151. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  152. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  153. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  154. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  155. package/src/cli/cmd/tui/ui/dialog-select.tsx +333 -0
  156. package/src/cli/cmd/tui/ui/dialog.tsx +171 -0
  157. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  158. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  159. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  160. package/src/cli/cmd/tui/util/editor.ts +32 -0
  161. package/src/cli/cmd/tui/util/terminal.ts +146 -0
  162. package/src/cli/cmd/tui/worker.ts +63 -0
  163. package/src/cli/cmd/uninstall.ts +344 -0
  164. package/src/cli/cmd/upgrade.ts +127 -0
  165. package/src/cli/cmd/web.ts +84 -0
  166. package/src/cli/error.ts +69 -0
  167. package/src/cli/ui.ts +101 -0
  168. package/src/cli/upgrade.ts +28 -0
  169. package/src/command/index.ts +80 -0
  170. package/src/command/template/initialize.txt +10 -0
  171. package/src/command/template/review.txt +97 -0
  172. package/src/config/config.ts +994 -0
  173. package/src/config/markdown.ts +41 -0
  174. package/src/env/index.ts +26 -0
  175. package/src/file/ignore.ts +83 -0
  176. package/src/file/index.ts +328 -0
  177. package/src/file/ripgrep.ts +393 -0
  178. package/src/file/time.ts +64 -0
  179. package/src/file/watcher.ts +103 -0
  180. package/src/flag/flag.ts +84 -0
  181. package/src/format/formatter.ts +315 -0
  182. package/src/format/index.ts +137 -0
  183. package/src/global/index.ts +101 -0
  184. package/src/id/id.ts +73 -0
  185. package/src/ide/index.ts +76 -0
  186. package/src/index.ts +297 -0
  187. package/src/index.ts.backup +271 -0
  188. package/src/installation/index.ts +258 -0
  189. package/src/lib/IMPLEMENTATION_NOTES.md +345 -0
  190. package/src/lib/error-handler.ts +225 -0
  191. package/src/lib/error-testing-guide.md +258 -0
  192. package/src/lib/errors.ts +285 -0
  193. package/src/lib/performance.ts +70 -0
  194. package/src/lib/telemetry.ts +282 -0
  195. package/src/lsp/client.ts +229 -0
  196. package/src/lsp/index.ts +485 -0
  197. package/src/lsp/language.ts +116 -0
  198. package/src/lsp/server.ts +1895 -0
  199. package/src/mcp/auth.ts +135 -0
  200. package/src/mcp/index.ts +1117 -0
  201. package/src/mcp/intent-analyzer.ts +376 -0
  202. package/src/mcp/oauth-callback.ts +200 -0
  203. package/src/mcp/oauth-provider.ts +154 -0
  204. package/src/patch/index.ts +632 -0
  205. package/src/permission/index.ts +199 -0
  206. package/src/plugin/index.ts +91 -0
  207. package/src/project/bootstrap.ts +33 -0
  208. package/src/project/instance.ts +78 -0
  209. package/src/project/project.ts +236 -0
  210. package/src/project/state.ts +65 -0
  211. package/src/project/vcs.ts +76 -0
  212. package/src/provider/auth.ts +143 -0
  213. package/src/provider/models-macro.ts +55 -0
  214. package/src/provider/models.ts +161 -0
  215. package/src/provider/provider.ts +1109 -0
  216. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  217. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  218. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  219. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  220. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  221. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  222. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  223. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  224. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  225. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  226. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  227. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  228. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  229. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  230. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  231. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  232. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  233. package/src/provider/transform.ts +455 -0
  234. package/src/pty/index.ts +231 -0
  235. package/src/security/guardrails.test.ts +341 -0
  236. package/src/security/guardrails.ts +570 -0
  237. package/src/security/index.ts +19 -0
  238. package/src/server/error.ts +36 -0
  239. package/src/server/project.ts +79 -0
  240. package/src/server/server.ts +2641 -0
  241. package/src/server/tui.ts +71 -0
  242. package/src/session/compaction.ts +228 -0
  243. package/src/session/index.ts +464 -0
  244. package/src/session/llm.ts +201 -0
  245. package/src/session/message-v2.ts +695 -0
  246. package/src/session/message.ts +189 -0
  247. package/src/session/processor.ts +409 -0
  248. package/src/session/prompt/act-switch.txt +5 -0
  249. package/src/session/prompt/anthropic-20250930.txt +166 -0
  250. package/src/session/prompt/anthropic.txt +63 -0
  251. package/src/session/prompt/anthropic_spoof.txt +1 -0
  252. package/src/session/prompt/beast.txt +76 -0
  253. package/src/session/prompt/codex.txt +304 -0
  254. package/src/session/prompt/copilot-gpt-5.txt +137 -0
  255. package/src/session/prompt/gemini.txt +62 -0
  256. package/src/session/prompt/max-steps.txt +16 -0
  257. package/src/session/prompt/plan-reminder-anthropic.txt +35 -0
  258. package/src/session/prompt/plan.txt +24 -0
  259. package/src/session/prompt/polaris.txt +88 -0
  260. package/src/session/prompt/qwen.txt +59 -0
  261. package/src/session/prompt.ts +1552 -0
  262. package/src/session/retry.ts +86 -0
  263. package/src/session/revert.ts +108 -0
  264. package/src/session/sensitive-filter.test.ts +327 -0
  265. package/src/session/sensitive-filter.ts +466 -0
  266. package/src/session/status.ts +76 -0
  267. package/src/session/summary.ts +209 -0
  268. package/src/session/system.ts +122 -0
  269. package/src/session/todo.ts +37 -0
  270. package/src/share/share-next.ts +222 -0
  271. package/src/share/share.ts +87 -0
  272. package/src/shell/shell.ts +67 -0
  273. package/src/skill/index.ts +1 -0
  274. package/src/skill/skill.ts +83 -0
  275. package/src/snapshot/index.ts +197 -0
  276. package/src/storage/storage.ts +226 -0
  277. package/src/tests/agent.test.ts +308 -0
  278. package/src/tests/build-guards.test.ts +267 -0
  279. package/src/tests/config.test.ts +664 -0
  280. package/src/tests/tool-registry.test.ts +589 -0
  281. package/src/tool/bash.ts +314 -0
  282. package/src/tool/bash.txt +158 -0
  283. package/src/tool/batch.ts +175 -0
  284. package/src/tool/batch.txt +24 -0
  285. package/src/tool/codesearch.ts +184 -0
  286. package/src/tool/codesearch.txt +12 -0
  287. package/src/tool/edit.ts +675 -0
  288. package/src/tool/edit.txt +10 -0
  289. package/src/tool/glob.ts +65 -0
  290. package/src/tool/glob.txt +6 -0
  291. package/src/tool/grep.ts +121 -0
  292. package/src/tool/grep.txt +8 -0
  293. package/src/tool/invalid.ts +17 -0
  294. package/src/tool/ls.ts +110 -0
  295. package/src/tool/ls.txt +1 -0
  296. package/src/tool/lsp-diagnostics.ts +26 -0
  297. package/src/tool/lsp-diagnostics.txt +1 -0
  298. package/src/tool/lsp-hover.ts +31 -0
  299. package/src/tool/lsp-hover.txt +1 -0
  300. package/src/tool/lsp.ts +87 -0
  301. package/src/tool/lsp.txt +19 -0
  302. package/src/tool/multiedit.ts +46 -0
  303. package/src/tool/multiedit.txt +41 -0
  304. package/src/tool/patch.ts +233 -0
  305. package/src/tool/patch.txt +1 -0
  306. package/src/tool/read.ts +219 -0
  307. package/src/tool/read.txt +12 -0
  308. package/src/tool/registry.ts +162 -0
  309. package/src/tool/skill.ts +100 -0
  310. package/src/tool/task.ts +136 -0
  311. package/src/tool/task.txt +51 -0
  312. package/src/tool/todo.ts +39 -0
  313. package/src/tool/todoread.txt +14 -0
  314. package/src/tool/todowrite.txt +167 -0
  315. package/src/tool/tool.ts +71 -0
  316. package/src/tool/webfetch.ts +198 -0
  317. package/src/tool/webfetch.txt +13 -0
  318. package/src/tool/websearch.ts +268 -0
  319. package/src/tool/websearch.txt +13 -0
  320. package/src/tool/write.ts +110 -0
  321. package/src/tool/write.txt +8 -0
  322. package/src/util/archive.ts +16 -0
  323. package/src/util/color.ts +19 -0
  324. package/src/util/context.ts +25 -0
  325. package/src/util/defer.ts +12 -0
  326. package/src/util/eventloop.ts +20 -0
  327. package/src/util/filesystem.ts +83 -0
  328. package/src/util/fn.ts +11 -0
  329. package/src/util/iife.ts +3 -0
  330. package/src/util/keybind.ts +102 -0
  331. package/src/util/lazy.ts +11 -0
  332. package/src/util/license.ts +362 -0
  333. package/src/util/locale.ts +81 -0
  334. package/src/util/lock.ts +98 -0
  335. package/src/util/log.ts +180 -0
  336. package/src/util/queue.ts +32 -0
  337. package/src/util/rpc.ts +42 -0
  338. package/src/util/scrap.ts +10 -0
  339. package/src/util/signal.ts +12 -0
  340. package/src/util/timeout.ts +14 -0
  341. package/src/util/token.ts +7 -0
  342. package/src/util/wildcard.ts +54 -0
  343. package/sst-env.d.ts +9 -0
  344. package/test/agent/agent.test.ts +146 -0
  345. package/test/bun.test.ts +53 -0
  346. package/test/cli/cmd/acp.test.ts +144 -0
  347. package/test/cli/cmd/run.test.ts +250 -0
  348. package/test/cli/github-remote.test.ts +80 -0
  349. package/test/config/agent-color.test.ts +66 -0
  350. package/test/config/config.test.ts +536 -0
  351. package/test/config/markdown.test.ts +89 -0
  352. package/test/file/ignore.test.ts +10 -0
  353. package/test/fixture/fixture.ts +37 -0
  354. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  355. package/test/helpers.ts +172 -0
  356. package/test/ide/ide.test.ts +82 -0
  357. package/test/installation/installation.test.ts +143 -0
  358. package/test/keybind.test.ts +421 -0
  359. package/test/lsp/client.test.ts +95 -0
  360. package/test/mcp/headers.test.ts +153 -0
  361. package/test/patch/patch.test.ts +348 -0
  362. package/test/preload.ts +57 -0
  363. package/test/project/project.test.ts +74 -0
  364. package/test/provider/provider.test.ts +74 -0
  365. package/test/provider/transform.test.ts +411 -0
  366. package/test/session/retry.test.ts +111 -0
  367. package/test/session/session.test.ts +71 -0
  368. package/test/skill/skill.test.ts +131 -0
  369. package/test/snapshot/snapshot.test.ts +940 -0
  370. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  371. package/test/tool/bash.test.ts +434 -0
  372. package/test/tool/grep.test.ts +108 -0
  373. package/test/tool/patch.test.ts +259 -0
  374. package/test/tool/read.test.ts +42 -0
  375. package/test/util/iife.test.ts +36 -0
  376. package/test/util/lazy.test.ts +50 -0
  377. package/test/util/license.test.ts +235 -0
  378. package/test/util/timeout.test.ts +21 -0
  379. package/test/util/wildcard.test.ts +55 -0
  380. package/tsconfig.json +16 -0
  381. package/update-versions.ps1 +65 -0
@@ -0,0 +1,209 @@
1
+ import { Provider } from "@/provider/provider"
2
+
3
+ import { fn } from "@/util/fn"
4
+ import z from "zod"
5
+ import { Session } from "."
6
+
7
+ import { MessageV2 } from "./message-v2"
8
+ import { Identifier } from "@/id/id"
9
+ import { Snapshot } from "@/snapshot"
10
+
11
+ import { Log } from "@/util/log"
12
+ import path from "path"
13
+ import { Instance } from "@/project/instance"
14
+ import { Storage } from "@/storage/storage"
15
+ import { Bus } from "@/bus"
16
+
17
+ import { LLM } from "./llm"
18
+ import { Agent } from "@/agent/agent"
19
+
20
+ export namespace SessionSummary {
21
+ const log = Log.create({ service: "session.summary" })
22
+
23
+ export const summarize = fn(
24
+ z.object({
25
+ sessionID: z.string(),
26
+ messageID: z.string(),
27
+ }),
28
+ async (input) => {
29
+ const all = await Session.messages({ sessionID: input.sessionID })
30
+ await Promise.all([
31
+ summarizeSession({ sessionID: input.sessionID, messages: all }),
32
+ summarizeMessage({ messageID: input.messageID, messages: all }),
33
+ ])
34
+ },
35
+ )
36
+
37
+ async function summarizeSession(input: { sessionID: string; messages: MessageV2.WithParts[] }) {
38
+ const files = new Set(
39
+ input.messages
40
+ .flatMap((x) => x.parts)
41
+ .filter((x) => x.type === "patch")
42
+ .flatMap((x) => x.files)
43
+ .map((x) => path.relative(Instance.worktree, x)),
44
+ )
45
+ const diffs = await computeDiff({ messages: input.messages }).then((x) =>
46
+ x.filter((x) => {
47
+ return files.has(x.file)
48
+ }),
49
+ )
50
+ await Session.update(input.sessionID, (draft) => {
51
+ draft.summary = {
52
+ additions: diffs.reduce((sum, x) => sum + x.additions, 0),
53
+ deletions: diffs.reduce((sum, x) => sum + x.deletions, 0),
54
+ files: diffs.length,
55
+ }
56
+ })
57
+ await Storage.write(["session_diff", input.sessionID], diffs)
58
+ Bus.publish(Session.Event.Diff, {
59
+ sessionID: input.sessionID,
60
+ diff: diffs,
61
+ })
62
+ }
63
+
64
+ async function summarizeMessage(input: { messageID: string; messages: MessageV2.WithParts[] }) {
65
+ const messages = input.messages.filter(
66
+ (m) => m.info.id === input.messageID || (m.info.role === "assistant" && m.info.parentID === input.messageID),
67
+ )
68
+ const msgWithParts = messages.find((m) => m.info.id === input.messageID)!
69
+ const userMsg = msgWithParts.info as MessageV2.User
70
+ const diffs = await computeDiff({ messages })
71
+ userMsg.summary = {
72
+ ...userMsg.summary,
73
+ diffs,
74
+ }
75
+ await Session.updateMessage(userMsg)
76
+
77
+ const assistantMsg = messages.find((m) => m.info.role === "assistant")!.info as MessageV2.Assistant
78
+ const small = await (async () => {
79
+ try {
80
+ return (
81
+ (await Provider.getSmallModel(assistantMsg.providerID)) ??
82
+ (await Provider.getModel(assistantMsg.providerID, assistantMsg.modelID))
83
+ )
84
+ } catch {}
85
+ const fallback = await Provider.defaultModel()
86
+ return await Provider.getModel(fallback.providerID, fallback.modelID)
87
+ })()
88
+
89
+ const textPart = msgWithParts.parts.find((p) => p.type === "text" && !p.synthetic) as MessageV2.TextPart
90
+ if (textPart && !userMsg.summary?.title) {
91
+ /* DISABLED FOR TESTING
92
+ const agent = await Agent.get("title")
93
+ const stream = await LLM.stream({
94
+ agent,
95
+ user: userMsg,
96
+ tools: {},
97
+ model: agent.model ? await Provider.getModel(agent.model.providerID, agent.model.modelID) : small,
98
+ small: true,
99
+ messages: [
100
+ {
101
+ role: "user" as const,
102
+ content: `
103
+ The following is the text to summarize:
104
+ <text>
105
+ ${textPart?.text ?? ""}
106
+ </text>
107
+ `,
108
+ },
109
+ ],
110
+ abort: new AbortController().signal,
111
+ sessionID: userMsg.sessionID,
112
+ system: [],
113
+ retries: 3,
114
+ })
115
+ const result = await stream.text
116
+ log.info("title", { title: result })
117
+ userMsg.summary.title = result
118
+ await Session.updateMessage(userMsg)
119
+ */
120
+ }
121
+
122
+ if (
123
+ messages.some(
124
+ (m) =>
125
+ m.info.role === "assistant" && m.parts.some((p) => p.type === "step-finish" && p.reason !== "tool-calls"),
126
+ )
127
+ ) {
128
+ if (diffs.length > 0) {
129
+ for (const msg of messages) {
130
+ for (const part of msg.parts) {
131
+ if (part.type === "tool" && part.state.status === "completed") {
132
+ part.state.output = "[TOOL OUTPUT PRUNED]"
133
+ }
134
+ }
135
+ }
136
+ const summaryAgent = await Agent.get("summary")
137
+ const summaryModel = await (async () => {
138
+ if (!summaryAgent.model) return undefined
139
+ try {
140
+ return await Provider.getModel(summaryAgent.model.providerID, summaryAgent.model.modelID)
141
+ } catch {}
142
+ const fallback = await Provider.defaultModel()
143
+ return await Provider.getModel(fallback.providerID, fallback.modelID)
144
+ })()
145
+ const stream = await LLM.stream({
146
+ agent: summaryAgent,
147
+ user: userMsg,
148
+ tools: {},
149
+ model: summaryModel ?? small,
150
+ small: true,
151
+ messages: [
152
+ ...MessageV2.toModelMessage(messages, { sessionID: userMsg.sessionID }),
153
+ {
154
+ role: "user" as const,
155
+ content: `Summarize the above conversation according to your system prompts.`,
156
+ },
157
+ ],
158
+ abort: new AbortController().signal,
159
+ sessionID: userMsg.sessionID,
160
+ system: [],
161
+ retries: 3,
162
+ })
163
+ const result = await stream.text
164
+ if (result) {
165
+ userMsg.summary.body = result
166
+ }
167
+ }
168
+ await Session.updateMessage(userMsg)
169
+ }
170
+ }
171
+
172
+ export const diff = fn(
173
+ z.object({
174
+ sessionID: Identifier.schema("session"),
175
+ messageID: Identifier.schema("message").optional(),
176
+ }),
177
+ async (input) => {
178
+ return Storage.read<Snapshot.FileDiff[]>(["session_diff", input.sessionID]).catch(() => [])
179
+ },
180
+ )
181
+
182
+ async function computeDiff(input: { messages: MessageV2.WithParts[] }) {
183
+ let from: string | undefined
184
+ let to: string | undefined
185
+
186
+ // scan assistant messages to find earliest from and latest to
187
+ // snapshot
188
+ for (const item of input.messages) {
189
+ if (!from) {
190
+ for (const part of item.parts) {
191
+ if (part.type === "step-start" && part.snapshot) {
192
+ from = part.snapshot
193
+ break
194
+ }
195
+ }
196
+ }
197
+
198
+ for (const part of item.parts) {
199
+ if (part.type === "step-finish" && part.snapshot) {
200
+ to = part.snapshot
201
+ break
202
+ }
203
+ }
204
+ }
205
+
206
+ if (from && to) return Snapshot.diffFull(from, to)
207
+ return []
208
+ }
209
+ }
@@ -0,0 +1,122 @@
1
+ import { Ripgrep } from "../file/ripgrep"
2
+ import { Global } from "../global"
3
+ import { Filesystem } from "../util/filesystem"
4
+ import { Config } from "../config/config"
5
+ import { Skill } from "../skill"
6
+
7
+ import { Instance } from "../project/instance"
8
+ import path from "path"
9
+ import os from "os"
10
+
11
+ import PROMPT_ANTHROPIC from "./prompt/anthropic.txt"
12
+ import PROMPT_ANTHROPIC_WITHOUT_TODO from "./prompt/qwen.txt"
13
+ import PROMPT_POLARIS from "./prompt/polaris.txt"
14
+ import PROMPT_BEAST from "./prompt/beast.txt"
15
+ import PROMPT_GEMINI from "./prompt/gemini.txt"
16
+ import PROMPT_ANTHROPIC_SPOOF from "./prompt/anthropic_spoof.txt"
17
+
18
+ import PROMPT_CODEX from "./prompt/codex.txt"
19
+ import type { Provider } from "@/provider/provider"
20
+
21
+ export namespace SystemPrompt {
22
+ export function header(providerID: string) {
23
+ if (providerID.includes("anthropic")) return [PROMPT_ANTHROPIC_SPOOF.trim()]
24
+ return []
25
+ }
26
+
27
+ export function provider(model: Provider.Model) {
28
+ if (model.api.id.includes("gpt-5")) return [PROMPT_CODEX]
29
+ if (model.api.id.includes("gpt-") || model.api.id.includes("o1") || model.api.id.includes("o3"))
30
+ return [PROMPT_BEAST]
31
+ if (model.api.id.includes("gemini-")) return [PROMPT_GEMINI]
32
+ if (model.api.id.includes("claude")) return [PROMPT_ANTHROPIC]
33
+ if (model.api.id.includes("polaris-alpha")) return [PROMPT_POLARIS]
34
+ return [PROMPT_ANTHROPIC_WITHOUT_TODO]
35
+ }
36
+
37
+ export async function environment() {
38
+ const project = Instance.project
39
+ return [
40
+ [
41
+ `Here is some useful information about the environment you are running in:`,
42
+ `<env>`,
43
+ ` Working directory: ${Instance.directory}`,
44
+ ` Is directory a git repo: ${project.vcs === "git" ? "yes" : "no"}`,
45
+ ` Platform: ${process.platform}`,
46
+ ` Today's date: ${new Date().toDateString()}`,
47
+ `</env>`,
48
+ `<files>`,
49
+ ` ${project.vcs === "git"
50
+ ? await Ripgrep.tree({
51
+ cwd: Instance.directory,
52
+ limit: 200,
53
+ })
54
+ : ""
55
+ }`,
56
+ `</files>`,
57
+ ].join("\n"),
58
+ ]
59
+ }
60
+
61
+ const LOCAL_RULE_FILES = [
62
+ "AGENTS.md",
63
+ "RIRD.md",
64
+ "RIRD.md",
65
+ "CONTEXT.md", // deprecated
66
+ ]
67
+ const GLOBAL_RULE_FILES = [
68
+ path.join(Global.Path.config, "AGENTS.md"),
69
+ path.join(os.homedir(), ".rird", "RIRD.md"),
70
+ path.join(os.homedir(), ".RIRD", "RIRD.md"),
71
+ path.join(os.homedir(), ".RIRD", "AGENTS.md"),
72
+ ]
73
+
74
+ export async function custom() {
75
+ const config = await Config.get()
76
+ const paths = new Set<string>()
77
+
78
+ for (const localRuleFile of LOCAL_RULE_FILES) {
79
+ const matches = await Filesystem.findUp(localRuleFile, Instance.directory, Instance.worktree)
80
+ if (matches.length > 0) {
81
+ matches.forEach((path) => paths.add(path))
82
+ break
83
+ }
84
+ }
85
+
86
+ for (const globalRuleFile of GLOBAL_RULE_FILES) {
87
+ if (await Bun.file(globalRuleFile).exists()) {
88
+ paths.add(globalRuleFile)
89
+ break
90
+ }
91
+ }
92
+
93
+ if (config.instructions) {
94
+ for (let instruction of config.instructions) {
95
+ if (instruction.startsWith("~/")) {
96
+ instruction = path.join(os.homedir(), instruction.slice(2))
97
+ }
98
+ let matches: string[] = []
99
+ if (path.isAbsolute(instruction)) {
100
+ matches = await Array.fromAsync(
101
+ new Bun.Glob(path.basename(instruction)).scan({
102
+ cwd: path.dirname(instruction),
103
+ absolute: true,
104
+ onlyFiles: true,
105
+ }),
106
+ ).catch(() => [])
107
+ } else {
108
+ matches = await Filesystem.globUp(instruction, Instance.directory, Instance.worktree).catch(() => [])
109
+ }
110
+ matches.forEach((path) => paths.add(path))
111
+ }
112
+ }
113
+
114
+ const found = Array.from(paths).map((p) =>
115
+ Bun.file(p)
116
+ .text()
117
+ .catch(() => "")
118
+ .then((x) => "Instructions from: " + p + "\n" + x),
119
+ )
120
+ return Promise.all(found).then((result) => result.filter(Boolean))
121
+ }
122
+ }
@@ -0,0 +1,37 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import z from "zod"
4
+ import { Storage } from "../storage/storage"
5
+
6
+ export namespace Todo {
7
+ export const Info = z
8
+ .object({
9
+ content: z.string().describe("Brief description of the task"),
10
+ status: z.string().describe("Current status of the task: pending, in_progress, completed, cancelled"),
11
+ priority: z.string().describe("Priority level of the task: high, medium, low"),
12
+ id: z.string().describe("Unique identifier for the todo item"),
13
+ })
14
+ .meta({ ref: "Todo" })
15
+ export type Info = z.infer<typeof Info>
16
+
17
+ export const Event = {
18
+ Updated: BusEvent.define(
19
+ "todo.updated",
20
+ z.object({
21
+ sessionID: z.string(),
22
+ todos: z.array(Info),
23
+ }),
24
+ ),
25
+ }
26
+
27
+ export async function update(input: { sessionID: string; todos: Info[] }) {
28
+ await Storage.write(["todo", input.sessionID], input.todos)
29
+ Bus.publish(Event.Updated, input)
30
+ }
31
+
32
+ export async function get(sessionID: string) {
33
+ return Storage.read<Info[]>(["todo", sessionID])
34
+ .then((x) => x || [])
35
+ .catch(() => [])
36
+ }
37
+ }
@@ -0,0 +1,222 @@
1
+ import { Bus } from "@/bus"
2
+ import { Config } from "@/config/config"
3
+ import { ulid } from "ulid"
4
+ import { Provider } from "@/provider/provider"
5
+ import { Session } from "@/session"
6
+ import { MessageV2 } from "@/session/message-v2"
7
+ import { Storage } from "@/storage/storage"
8
+ import { Log } from "@/util/log"
9
+ import type * as SDK from "@opencode-ai/sdk/v2"
10
+
11
+ export namespace ShareNext {
12
+ const log = Log.create({ service: "share-next" })
13
+
14
+ async function url() {
15
+ return process.env["RIRD_API"] ?? Config.get().then((x) => x.enterprise?.url ?? "https://api.rird.ai")
16
+ }
17
+
18
+ export async function init() {
19
+ return // Share disabled - no backend
20
+
21
+ Bus.subscribe(Session.Event.Updated, async (evt) => {
22
+ await sync(evt.properties.info.id, [
23
+ {
24
+ type: "session",
25
+ data: evt.properties.info,
26
+ },
27
+ ])
28
+ })
29
+ Bus.subscribe(MessageV2.Event.Updated, async (evt) => {
30
+ await sync(evt.properties.info.sessionID, [
31
+ {
32
+ type: "message",
33
+ data: evt.properties.info,
34
+ },
35
+ ])
36
+ if (evt.properties.info.role === "user") {
37
+ await sync(evt.properties.info.sessionID, [
38
+ {
39
+ type: "model",
40
+ data: [
41
+ await Provider.getModel(evt.properties.info.model.providerID, evt.properties.info.model.modelID).then(
42
+ (m) => m,
43
+ ),
44
+ ],
45
+ },
46
+ ])
47
+ }
48
+ })
49
+ Bus.subscribe(MessageV2.Event.PartUpdated, async (evt) => {
50
+ await sync(evt.properties.part.sessionID, [
51
+ {
52
+ type: "part",
53
+ data: evt.properties.part,
54
+ },
55
+ ])
56
+ })
57
+ Bus.subscribe(Session.Event.Diff, async (evt) => {
58
+ await sync(evt.properties.sessionID, [
59
+ {
60
+ type: "session_diff",
61
+ data: evt.properties.diff,
62
+ },
63
+ ])
64
+ })
65
+ }
66
+
67
+ export type CreateResult = { id: string; url: string; secret: string } | { error: string }
68
+
69
+ export async function create(sessionID: string): Promise<CreateResult> {
70
+ log.info("share disabled", { sessionID })
71
+ return { error: "Share feature coming soon" }
72
+
73
+ log.info("creating share", { sessionID })
74
+ const headers: Record<string, string> = {
75
+ "Content-Type": "application/json",
76
+ }
77
+ if (process.env.RIRD_API_KEY) {
78
+ headers["Authorization"] = `Bearer ${process.env.RIRD_API_KEY}`
79
+ }
80
+
81
+ const result = await fetch(`${await url()}/api/share`, {
82
+ method: "POST",
83
+ headers,
84
+ body: JSON.stringify({ sessionID: sessionID }),
85
+ })
86
+ .then((x) => x.json())
87
+ .then((x) => x as { id: string; url: string; secret: string })
88
+ await Storage.write(["session_share", sessionID], result)
89
+ fullSync(sessionID)
90
+ return result
91
+ }
92
+
93
+ async function get(sessionID: string) {
94
+ try {
95
+ return await Storage.read<{
96
+ id: string
97
+ secret: string
98
+ url: string
99
+ }>(["session_share", sessionID])
100
+ } catch {
101
+ // Session not shared - return undefined
102
+ return undefined
103
+ }
104
+ }
105
+
106
+ type Data =
107
+ | {
108
+ type: "session"
109
+ data: SDK.Session
110
+ }
111
+ | {
112
+ type: "message"
113
+ data: SDK.Message
114
+ }
115
+ | {
116
+ type: "part"
117
+ data: SDK.Part
118
+ }
119
+ | {
120
+ type: "session_diff"
121
+ data: SDK.FileDiff[]
122
+ }
123
+ | {
124
+ type: "model"
125
+ data: SDK.Model[]
126
+ }
127
+
128
+ const queue = new Map<string, { timeout: NodeJS.Timeout; data: Map<string, Data> }>()
129
+ async function sync(sessionID: string, data: Data[]) {
130
+ const existing = queue.get(sessionID)
131
+ if (existing) {
132
+ for (const item of data) {
133
+ existing.data.set("id" in item ? (item.id as string) : ulid(), item)
134
+ }
135
+ return
136
+ }
137
+
138
+ const dataMap = new Map<string, Data>()
139
+ for (const item of data) {
140
+ dataMap.set("id" in item ? (item.id as string) : ulid(), item)
141
+ }
142
+
143
+ const timeout = setTimeout(async () => {
144
+ const queued = queue.get(sessionID)
145
+ if (!queued) return
146
+ queue.delete(sessionID)
147
+ const share = await get(sessionID)
148
+ if (!share) return
149
+
150
+ const headers: Record<string, string> = {
151
+ "Content-Type": "application/json",
152
+ }
153
+ if (process.env.RIRD_API_KEY) {
154
+ headers["Authorization"] = `Bearer ${process.env.RIRD_API_KEY}`
155
+ }
156
+
157
+ await fetch(`${await url()}/api/share/${share.id}/sync`, {
158
+ method: "POST",
159
+ headers,
160
+ body: JSON.stringify({
161
+ secret: share.secret,
162
+ data: Array.from(queued.data.values()),
163
+ }),
164
+ })
165
+ }, 1000)
166
+ queue.set(sessionID, { timeout, data: dataMap })
167
+ }
168
+
169
+ export async function remove(sessionID: string) {
170
+ log.info("removing share", { sessionID })
171
+ const share = await get(sessionID)
172
+ if (!share) return
173
+
174
+ const headers: Record<string, string> = {
175
+ "Content-Type": "application/json",
176
+ }
177
+ if (process.env.RIRD_API_KEY) {
178
+ headers["Authorization"] = `Bearer ${process.env.RIRD_API_KEY}`
179
+ }
180
+
181
+ await fetch(`${await url()}/api/share/${share.id}`, {
182
+ method: "DELETE",
183
+ headers,
184
+ body: JSON.stringify({
185
+ secret: share.secret,
186
+ }),
187
+ })
188
+ await Storage.remove(["session_share", sessionID])
189
+ }
190
+
191
+ async function fullSync(sessionID: string) {
192
+ log.info("full sync", { sessionID })
193
+ const session = await Session.get(sessionID)
194
+ const diffs = await Session.diff(sessionID)
195
+ const messages = await Array.fromAsync(MessageV2.stream(sessionID))
196
+ const models = await Promise.all(
197
+ messages
198
+ .filter((m) => m.info.role === "user")
199
+ .map((m) => (m.info as SDK.UserMessage).model)
200
+ .map((m) => Provider.getModel(m.providerID, m.modelID).then((m) => m)),
201
+ )
202
+ await sync(sessionID, [
203
+ {
204
+ type: "session",
205
+ data: session,
206
+ },
207
+ ...messages.map((x) => ({
208
+ type: "message" as const,
209
+ data: x.info,
210
+ })),
211
+ ...messages.flatMap((x) => x.parts.map((y) => ({ type: "part" as const, data: y }))),
212
+ {
213
+ type: "session_diff",
214
+ data: diffs,
215
+ },
216
+ {
217
+ type: "model",
218
+ data: models,
219
+ },
220
+ ])
221
+ }
222
+ }
@@ -0,0 +1,87 @@
1
+ import { Bus } from "../bus"
2
+ import { Installation } from "../installation"
3
+ import { Session } from "../session"
4
+ import { MessageV2 } from "../session/message-v2"
5
+ import { Log } from "../util/log"
6
+
7
+ export namespace Share {
8
+ const log = Log.create({ service: "share" })
9
+
10
+ let queue: Promise<void> = Promise.resolve()
11
+ const pending = new Map<string, any>()
12
+
13
+ export async function sync(key: string, content: any) {
14
+ const [root, ...splits] = key.split("/")
15
+ if (root !== "session") return
16
+ const [sub, sessionID] = splits
17
+ if (sub === "share") return
18
+ const share = await Session.getShare(sessionID).catch(() => {})
19
+ if (!share) return
20
+ const { secret } = share
21
+ pending.set(key, content)
22
+ queue = queue
23
+ .then(async () => {
24
+ const content = pending.get(key)
25
+ if (content === undefined) return
26
+ pending.delete(key)
27
+
28
+ return fetch(`${URL}/share_sync`, {
29
+ method: "POST",
30
+ body: JSON.stringify({
31
+ sessionID: sessionID,
32
+ secret,
33
+ key: key,
34
+ content,
35
+ }),
36
+ })
37
+ })
38
+ .then((x) => {
39
+ if (x) {
40
+ log.info("synced", {
41
+ key: key,
42
+ status: x.status,
43
+ })
44
+ }
45
+ })
46
+ }
47
+
48
+ export function init() {
49
+ Bus.subscribe(Session.Event.Updated, async (evt) => {
50
+ await sync("session/info/" + evt.properties.info.id, evt.properties.info)
51
+ })
52
+ Bus.subscribe(MessageV2.Event.Updated, async (evt) => {
53
+ await sync("session/message/" + evt.properties.info.sessionID + "/" + evt.properties.info.id, evt.properties.info)
54
+ })
55
+ Bus.subscribe(MessageV2.Event.PartUpdated, async (evt) => {
56
+ await sync(
57
+ "session/part/" +
58
+ evt.properties.part.sessionID +
59
+ "/" +
60
+ evt.properties.part.messageID +
61
+ "/" +
62
+ evt.properties.part.id,
63
+ evt.properties.part,
64
+ )
65
+ })
66
+ }
67
+
68
+ export const URL =
69
+ process.env["RIRD_API"] ??
70
+ (Installation.isPreview() || Installation.isLocal() ? "https://api.dev.rird.ai" : "https://api.rird.ai")
71
+
72
+ export async function create(sessionID: string) {
73
+ return fetch(`${URL}/share_create`, {
74
+ method: "POST",
75
+ body: JSON.stringify({ sessionID: sessionID }),
76
+ })
77
+ .then((x) => x.json())
78
+ .then((x) => x as { url: string; secret: string })
79
+ }
80
+
81
+ export async function remove(sessionID: string, secret: string) {
82
+ return fetch(`${URL}/share_delete`, {
83
+ method: "POST",
84
+ body: JSON.stringify({ sessionID, secret }),
85
+ }).then((x) => x.json())
86
+ }
87
+ }