rimuru-ai 1.19.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 (470) hide show
  1. package/.rimuru/AGENTS.md +30 -0
  2. package/.rimuru/agents/backend.md +27 -0
  3. package/.rimuru/agents/database.md +31 -0
  4. package/.rimuru/agents/devops.md +30 -0
  5. package/.rimuru/agents/document-prep.md +49 -0
  6. package/.rimuru/agents/erp-architect.md +41 -0
  7. package/.rimuru/agents/ethical-hacking.md +49 -0
  8. package/.rimuru/agents/frontend.md +31 -0
  9. package/.rimuru/agents/fullstack.md +24 -0
  10. package/.rimuru/agents/system-engineer.md +31 -0
  11. package/.rimuru/agents/veldora-agent-tool-dev.md +30 -0
  12. package/.rimuru/agents/veldora-backend-dev.md +32 -0
  13. package/.rimuru/agents/veldora-cicd.md +32 -0
  14. package/.rimuru/agents/veldora-database.md +32 -0
  15. package/.rimuru/agents/veldora-doc.md +87 -0
  16. package/.rimuru/agents/veldora-frontend-dev.md +32 -0
  17. package/.rimuru/agents/veldora-great-sage.md +33 -0
  18. package/.rimuru/agents/veldora-mcp-creator.md +30 -0
  19. package/.rimuru/agents/veldora-pro.md +224 -0
  20. package/.rimuru/agents/veldora-prompt-enhancer.md +27 -0
  21. package/.rimuru/agents/veldora-skill-creator.md +28 -0
  22. package/.rimuru/agents/veldora.md +225 -0
  23. package/.rimuru/agents/veldorapro-agent-tool-dev.md +29 -0
  24. package/.rimuru/agents/veldorapro-backend-dev.md +29 -0
  25. package/.rimuru/agents/veldorapro-cicd.md +29 -0
  26. package/.rimuru/agents/veldorapro-database.md +29 -0
  27. package/.rimuru/agents/veldorapro-frontend-dev.md +29 -0
  28. package/.rimuru/agents/veldorapro-great-sage.md +33 -0
  29. package/.rimuru/agents/veldorapro-mcp-creator.md +27 -0
  30. package/.rimuru/agents/veldorapro-prompt-enhancer.md +25 -0
  31. package/.rimuru/agents/veldorapro-skill-creator.md +27 -0
  32. package/.rimuru/command/ai-deps.md +24 -0
  33. package/.rimuru/command/changelog.md +49 -0
  34. package/.rimuru/command/commit.md +37 -0
  35. package/.rimuru/command/issues.md +23 -0
  36. package/.rimuru/command/learn.md +42 -0
  37. package/.rimuru/command/rmslop.md +15 -0
  38. package/.rimuru/command/spellcheck.md +5 -0
  39. package/.rimuru/command/translate.md +14 -0
  40. package/.rimuru/glossary/README.md +63 -0
  41. package/.rimuru/glossary/ar.md +28 -0
  42. package/.rimuru/glossary/br.md +34 -0
  43. package/.rimuru/glossary/bs.md +33 -0
  44. package/.rimuru/glossary/da.md +27 -0
  45. package/.rimuru/glossary/de.md +27 -0
  46. package/.rimuru/glossary/es.md +27 -0
  47. package/.rimuru/glossary/fr.md +27 -0
  48. package/.rimuru/glossary/ja.md +33 -0
  49. package/.rimuru/glossary/ko.md +27 -0
  50. package/.rimuru/glossary/no.md +38 -0
  51. package/.rimuru/glossary/pl.md +27 -0
  52. package/.rimuru/glossary/ru.md +27 -0
  53. package/.rimuru/glossary/th.md +34 -0
  54. package/.rimuru/glossary/tr.md +38 -0
  55. package/.rimuru/glossary/zh-cn.md +42 -0
  56. package/.rimuru/glossary/zh-tw.md +42 -0
  57. package/.rimuru/improver/changelog.md +250 -0
  58. package/.rimuru/improver/knowledge.md +172 -0
  59. package/.rimuru/improver/plugins.md +21 -0
  60. package/.rimuru/improver/skills.md +60 -0
  61. package/.rimuru/improver/token-audit.md +21 -0
  62. package/.rimuru/opencode.jsonc +140 -0
  63. package/.rimuru/plugins/smoke-theme.json +223 -0
  64. package/.rimuru/plugins/tui-smoke.tsx +1019 -0
  65. package/.rimuru/skills/effect/SKILL.md +38 -0
  66. package/.rimuru/themes/mytheme.json +223 -0
  67. package/.rimuru/tool/github-pr-search.ts +64 -0
  68. package/.rimuru/tool/github-triage.ts +60 -0
  69. package/README.md +31 -0
  70. package/package.json +167 -0
  71. package/src/account/account.ts +463 -0
  72. package/src/account/repo.ts +173 -0
  73. package/src/account/schema.ts +99 -0
  74. package/src/account/url.ts +8 -0
  75. package/src/acp/agent.ts +95 -0
  76. package/src/acp/config-option.ts +203 -0
  77. package/src/acp/content.ts +250 -0
  78. package/src/acp/directory.ts +210 -0
  79. package/src/acp/error.ts +90 -0
  80. package/src/acp/event.ts +342 -0
  81. package/src/acp/permission.ts +124 -0
  82. package/src/acp/profile.ts +42 -0
  83. package/src/acp/service.ts +1048 -0
  84. package/src/acp/session.ts +231 -0
  85. package/src/acp/tool.ts +367 -0
  86. package/src/acp/usage.ts +232 -0
  87. package/src/agent/agent.ts +459 -0
  88. package/src/agent/generate.txt +75 -0
  89. package/src/agent/prompt/compaction.txt +9 -0
  90. package/src/agent/prompt/explore.txt +18 -0
  91. package/src/agent/prompt/summary.txt +11 -0
  92. package/src/agent/prompt/title.txt +44 -0
  93. package/src/agent/subagent-permissions.ts +27 -0
  94. package/src/audio.d.ts +14 -0
  95. package/src/auth/index.ts +99 -0
  96. package/src/background/job.ts +39 -0
  97. package/src/bus/global.ts +22 -0
  98. package/src/cli/bootstrap.ts +11 -0
  99. package/src/cli/cmd/account.ts +264 -0
  100. package/src/cli/cmd/acp.ts +73 -0
  101. package/src/cli/cmd/agent.ts +259 -0
  102. package/src/cli/cmd/attach.ts +97 -0
  103. package/src/cli/cmd/cmd.ts +7 -0
  104. package/src/cli/cmd/db.ts +62 -0
  105. package/src/cli/cmd/debug/agent.handler.ts +193 -0
  106. package/src/cli/cmd/debug/agent.ts +27 -0
  107. package/src/cli/cmd/debug/config.ts +14 -0
  108. package/src/cli/cmd/debug/file.ts +73 -0
  109. package/src/cli/cmd/debug/index.ts +87 -0
  110. package/src/cli/cmd/debug/lsp.ts +50 -0
  111. package/src/cli/cmd/debug/ripgrep.ts +79 -0
  112. package/src/cli/cmd/debug/scrap.ts +15 -0
  113. package/src/cli/cmd/debug/skill.ts +15 -0
  114. package/src/cli/cmd/debug/snapshot.ts +50 -0
  115. package/src/cli/cmd/debug/startup.ts +11 -0
  116. package/src/cli/cmd/debug/v2.ts +49 -0
  117. package/src/cli/cmd/export.ts +292 -0
  118. package/src/cli/cmd/generate.ts +54 -0
  119. package/src/cli/cmd/github.handler.ts +1593 -0
  120. package/src/cli/cmd/github.shared.ts +30 -0
  121. package/src/cli/cmd/github.ts +42 -0
  122. package/src/cli/cmd/import.ts +224 -0
  123. package/src/cli/cmd/mcp.ts +850 -0
  124. package/src/cli/cmd/models.ts +66 -0
  125. package/src/cli/cmd/plug.ts +230 -0
  126. package/src/cli/cmd/pr.ts +115 -0
  127. package/src/cli/cmd/prompt-display.ts +1 -0
  128. package/src/cli/cmd/providers.ts +534 -0
  129. package/src/cli/cmd/run/demo.ts +1274 -0
  130. package/src/cli/cmd/run/entry.body.ts +205 -0
  131. package/src/cli/cmd/run/footer.command.tsx +1064 -0
  132. package/src/cli/cmd/run/footer.menu.tsx +351 -0
  133. package/src/cli/cmd/run/footer.permission.tsx +472 -0
  134. package/src/cli/cmd/run/footer.prompt.tsx +1306 -0
  135. package/src/cli/cmd/run/footer.question.tsx +573 -0
  136. package/src/cli/cmd/run/footer.subagent.tsx +173 -0
  137. package/src/cli/cmd/run/footer.ts +1129 -0
  138. package/src/cli/cmd/run/footer.view.tsx +943 -0
  139. package/src/cli/cmd/run/footer.width.ts +27 -0
  140. package/src/cli/cmd/run/permission.shared.ts +256 -0
  141. package/src/cli/cmd/run/prompt.editor.ts +157 -0
  142. package/src/cli/cmd/run/prompt.shared.ts +153 -0
  143. package/src/cli/cmd/run/question.shared.ts +340 -0
  144. package/src/cli/cmd/run/runtime.boot.ts +202 -0
  145. package/src/cli/cmd/run/runtime.lifecycle.ts +406 -0
  146. package/src/cli/cmd/run/runtime.queue.ts +349 -0
  147. package/src/cli/cmd/run/runtime.shared.ts +17 -0
  148. package/src/cli/cmd/run/runtime.stdin.ts +37 -0
  149. package/src/cli/cmd/run/runtime.ts +814 -0
  150. package/src/cli/cmd/run/scrollback.shared.ts +92 -0
  151. package/src/cli/cmd/run/scrollback.surface.ts +431 -0
  152. package/src/cli/cmd/run/scrollback.writer.tsx +352 -0
  153. package/src/cli/cmd/run/session-data.ts +1113 -0
  154. package/src/cli/cmd/run/session-replay.ts +374 -0
  155. package/src/cli/cmd/run/session.shared.ts +196 -0
  156. package/src/cli/cmd/run/splash.ts +280 -0
  157. package/src/cli/cmd/run/stream.transport.ts +1462 -0
  158. package/src/cli/cmd/run/stream.ts +175 -0
  159. package/src/cli/cmd/run/subagent-data.ts +876 -0
  160. package/src/cli/cmd/run/theme.ts +690 -0
  161. package/src/cli/cmd/run/tool.ts +1489 -0
  162. package/src/cli/cmd/run/trace.ts +94 -0
  163. package/src/cli/cmd/run/turn-summary.ts +47 -0
  164. package/src/cli/cmd/run/types.ts +350 -0
  165. package/src/cli/cmd/run/variant.shared.ts +215 -0
  166. package/src/cli/cmd/run.ts +894 -0
  167. package/src/cli/cmd/serve.ts +24 -0
  168. package/src/cli/cmd/session.ts +147 -0
  169. package/src/cli/cmd/stats.ts +393 -0
  170. package/src/cli/cmd/tui.ts +224 -0
  171. package/src/cli/cmd/uninstall.ts +353 -0
  172. package/src/cli/cmd/upgrade.ts +74 -0
  173. package/src/cli/cmd/web.ts +84 -0
  174. package/src/cli/effect/prompt.ts +37 -0
  175. package/src/cli/effect-cmd.ts +96 -0
  176. package/src/cli/error.ts +130 -0
  177. package/src/cli/heap.ts +45 -0
  178. package/src/cli/logo.ts +1 -0
  179. package/src/cli/network.ts +64 -0
  180. package/src/cli/tui/layer.ts +7 -0
  181. package/src/cli/tui/validate-session.ts +29 -0
  182. package/src/cli/tui/worker.ts +71 -0
  183. package/src/cli/ui.ts +98 -0
  184. package/src/cli/upgrade.ts +53 -0
  185. package/src/command/index.ts +184 -0
  186. package/src/command/template/initialize.txt +66 -0
  187. package/src/command/template/review.txt +101 -0
  188. package/src/config/agent.ts +59 -0
  189. package/src/config/command.ts +39 -0
  190. package/src/config/config.ts +686 -0
  191. package/src/config/entry-name.ts +19 -0
  192. package/src/config/managed.ts +69 -0
  193. package/src/config/markdown.ts +36 -0
  194. package/src/config/parse.ts +79 -0
  195. package/src/config/paths.ts +45 -0
  196. package/src/config/plugin.ts +79 -0
  197. package/src/config/tui-cwd.ts +5 -0
  198. package/src/config/tui-host-attention.ts +21 -0
  199. package/src/config/tui-migrate.ts +132 -0
  200. package/src/config/tui.ts +274 -0
  201. package/src/config/variable.ts +91 -0
  202. package/src/control-plane/adapters/index.ts +41 -0
  203. package/src/control-plane/adapters/worktree.ts +96 -0
  204. package/src/control-plane/dev/README.md +19 -0
  205. package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
  206. package/src/control-plane/types.ts +59 -0
  207. package/src/control-plane/util.ts +39 -0
  208. package/src/control-plane/workspace-adapter-runtime.ts +51 -0
  209. package/src/control-plane/workspace-context.ts +26 -0
  210. package/src/control-plane/workspace.ts +989 -0
  211. package/src/effect/app-runtime.ts +132 -0
  212. package/src/effect/bootstrap-runtime.ts +23 -0
  213. package/src/effect/bridge.ts +84 -0
  214. package/src/effect/config-service.ts +67 -0
  215. package/src/effect/instance-ref.ts +11 -0
  216. package/src/effect/instance-registry.ts +12 -0
  217. package/src/effect/instance-state.ts +69 -0
  218. package/src/effect/promise.ts +17 -0
  219. package/src/effect/run-service.ts +47 -0
  220. package/src/effect/runner.ts +217 -0
  221. package/src/effect/runtime-flags.ts +79 -0
  222. package/src/env/index.ts +43 -0
  223. package/src/event-v2-bridge.ts +79 -0
  224. package/src/format/formatter.ts +404 -0
  225. package/src/format/index.ts +205 -0
  226. package/src/git/index.ts +350 -0
  227. package/src/id/id.ts +80 -0
  228. package/src/ide/index.ts +61 -0
  229. package/src/image/image.ts +174 -0
  230. package/src/index.ts +142 -0
  231. package/src/installation/index.ts +350 -0
  232. package/src/lsp/client.ts +650 -0
  233. package/src/lsp/diagnostic.ts +29 -0
  234. package/src/lsp/language.ts +121 -0
  235. package/src/lsp/launch.ts +21 -0
  236. package/src/lsp/lsp.ts +511 -0
  237. package/src/lsp/server.ts +1983 -0
  238. package/src/markdown.d.ts +4 -0
  239. package/src/mcp/auth.ts +174 -0
  240. package/src/mcp/catalog.ts +153 -0
  241. package/src/mcp/index.ts +946 -0
  242. package/src/mcp/oauth-callback.ts +233 -0
  243. package/src/mcp/oauth-provider.ts +206 -0
  244. package/src/node.ts +4 -0
  245. package/src/patch/index.ts +686 -0
  246. package/src/permission/arity.ts +163 -0
  247. package/src/permission/evaluate.ts +1 -0
  248. package/src/permission/index.ts +230 -0
  249. package/src/plugin/azure.ts +26 -0
  250. package/src/plugin/cloudflare.ts +76 -0
  251. package/src/plugin/digitalocean.ts +383 -0
  252. package/src/plugin/github-copilot/copilot.ts +413 -0
  253. package/src/plugin/github-copilot/models.ts +246 -0
  254. package/src/plugin/index.ts +316 -0
  255. package/src/plugin/install.ts +439 -0
  256. package/src/plugin/loader.ts +237 -0
  257. package/src/plugin/meta.ts +188 -0
  258. package/src/plugin/openai/README.md +31 -0
  259. package/src/plugin/openai/codex.ts +641 -0
  260. package/src/plugin/openai/ws-pool.ts +270 -0
  261. package/src/plugin/openai/ws.ts +381 -0
  262. package/src/plugin/pty-environment.ts +24 -0
  263. package/src/plugin/shared.ts +323 -0
  264. package/src/plugin/snowflake-cortex.ts +529 -0
  265. package/src/plugin/tui/internal.ts +10 -0
  266. package/src/plugin/tui/runtime.ts +1130 -0
  267. package/src/plugin/xai.ts +716 -0
  268. package/src/project/bootstrap-service.ts +9 -0
  269. package/src/project/bootstrap.ts +76 -0
  270. package/src/project/instance-context.ts +24 -0
  271. package/src/project/instance-layer.ts +11 -0
  272. package/src/project/instance-runtime.ts +16 -0
  273. package/src/project/instance-store.ts +209 -0
  274. package/src/project/project.ts +519 -0
  275. package/src/project/vcs.ts +431 -0
  276. package/src/provider/auth.ts +233 -0
  277. package/src/provider/error.ts +188 -0
  278. package/src/provider/model-status.ts +8 -0
  279. package/src/provider/provider.ts +1979 -0
  280. package/src/provider/transform.ts +1426 -0
  281. package/src/question/index.ts +229 -0
  282. package/src/question/schema.ts +10 -0
  283. package/src/server/auth.ts +48 -0
  284. package/src/server/event.ts +13 -0
  285. package/src/server/global-lifecycle.ts +28 -0
  286. package/src/server/init-projectors.ts +3 -0
  287. package/src/server/mdns.ts +47 -0
  288. package/src/server/projectors.ts +1 -0
  289. package/src/server/proxy-util.ts +48 -0
  290. package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
  291. package/src/server/routes/instance/httpapi/api.ts +78 -0
  292. package/src/server/routes/instance/httpapi/errors.ts +193 -0
  293. package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
  294. package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
  295. package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
  296. package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
  297. package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
  298. package/src/server/routes/instance/httpapi/groups/file.ts +185 -0
  299. package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
  300. package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
  301. package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
  302. package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
  303. package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
  304. package/src/server/routes/instance/httpapi/groups/project-copy.ts +32 -0
  305. package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
  306. package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
  307. package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
  308. package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
  309. package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
  310. package/src/server/routes/instance/httpapi/groups/session.ts +462 -0
  311. package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
  312. package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
  313. package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
  314. package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
  315. package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
  316. package/src/server/routes/instance/httpapi/handlers/control.ts +43 -0
  317. package/src/server/routes/instance/httpapi/handlers/event.ts +99 -0
  318. package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
  319. package/src/server/routes/instance/httpapi/handlers/file.ts +139 -0
  320. package/src/server/routes/instance/httpapi/handlers/global.ts +156 -0
  321. package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
  322. package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
  323. package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
  324. package/src/server/routes/instance/httpapi/handlers/project-copy.ts +83 -0
  325. package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
  326. package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
  327. package/src/server/routes/instance/httpapi/handlers/pty.ts +273 -0
  328. package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
  329. package/src/server/routes/instance/httpapi/handlers/session-errors.ts +21 -0
  330. package/src/server/routes/instance/httpapi/handlers/session.ts +440 -0
  331. package/src/server/routes/instance/httpapi/handlers/sync.ts +89 -0
  332. package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
  333. package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
  334. package/src/server/routes/instance/httpapi/lifecycle.ts +54 -0
  335. package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
  336. package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
  337. package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
  338. package/src/server/routes/instance/httpapi/middleware/error.ts +43 -0
  339. package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
  340. package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
  341. package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
  342. package/src/server/routes/instance/httpapi/middleware/schema-error.ts +41 -0
  343. package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
  344. package/src/server/routes/instance/httpapi/public.ts +535 -0
  345. package/src/server/routes/instance/httpapi/server.ts +298 -0
  346. package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
  347. package/src/server/server.ts +225 -0
  348. package/src/server/shared/fence.ts +60 -0
  349. package/src/server/shared/pty-ticket.ts +15 -0
  350. package/src/server/shared/public-ui.ts +12 -0
  351. package/src/server/shared/tui-control.ts +28 -0
  352. package/src/server/shared/ui.ts +122 -0
  353. package/src/server/shared/workspace-routing.ts +38 -0
  354. package/src/server/tui-event.ts +53 -0
  355. package/src/session/compaction.ts +620 -0
  356. package/src/session/instruction.ts +241 -0
  357. package/src/session/llm/AGENTS.md +90 -0
  358. package/src/session/llm/ai-sdk.ts +288 -0
  359. package/src/session/llm/native-request.ts +196 -0
  360. package/src/session/llm/native-runtime.ts +196 -0
  361. package/src/session/llm/request.ts +218 -0
  362. package/src/session/llm.ts +415 -0
  363. package/src/session/message-error.ts +14 -0
  364. package/src/session/message-v2.ts +744 -0
  365. package/src/session/message.ts +148 -0
  366. package/src/session/overflow.ts +34 -0
  367. package/src/session/processor.ts +1084 -0
  368. package/src/session/prompt/anthropic.txt +105 -0
  369. package/src/session/prompt/beast.txt +147 -0
  370. package/src/session/prompt/build-switch.txt +5 -0
  371. package/src/session/prompt/codex.txt +79 -0
  372. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  373. package/src/session/prompt/default.txt +95 -0
  374. package/src/session/prompt/gemini.txt +155 -0
  375. package/src/session/prompt/gpt.txt +107 -0
  376. package/src/session/prompt/kimi.txt +95 -0
  377. package/src/session/prompt/max-steps.txt +16 -0
  378. package/src/session/prompt/plan-mode.txt +70 -0
  379. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  380. package/src/session/prompt/plan.txt +26 -0
  381. package/src/session/prompt/trinity.txt +97 -0
  382. package/src/session/prompt.ts +1722 -0
  383. package/src/session/reminders.ts +92 -0
  384. package/src/session/retry.ts +201 -0
  385. package/src/session/revert.ts +160 -0
  386. package/src/session/run-state.ts +156 -0
  387. package/src/session/schema.ts +26 -0
  388. package/src/session/session.ts +1119 -0
  389. package/src/session/status.ts +97 -0
  390. package/src/session/summary.ts +165 -0
  391. package/src/session/system.ts +117 -0
  392. package/src/session/todo.ts +90 -0
  393. package/src/session/tools.ts +207 -0
  394. package/src/share/session.ts +61 -0
  395. package/src/share/share-next.ts +385 -0
  396. package/src/skill/discovery.ts +109 -0
  397. package/src/skill/index.ts +366 -0
  398. package/src/snapshot/index.ts +808 -0
  399. package/src/sql.d.ts +4 -0
  400. package/src/storage/schema.ts +5 -0
  401. package/src/storage/storage.ts +329 -0
  402. package/src/sync/README.md +179 -0
  403. package/src/sync/schema.ts +11 -0
  404. package/src/temporary.ts +31 -0
  405. package/src/tool/apply_patch.ts +313 -0
  406. package/src/tool/apply_patch.txt +33 -0
  407. package/src/tool/edit.ts +737 -0
  408. package/src/tool/edit.txt +10 -0
  409. package/src/tool/external-directory.ts +49 -0
  410. package/src/tool/glob.ts +76 -0
  411. package/src/tool/glob.txt +6 -0
  412. package/src/tool/grep.ts +112 -0
  413. package/src/tool/grep.txt +8 -0
  414. package/src/tool/invalid.ts +21 -0
  415. package/src/tool/json-schema.ts +164 -0
  416. package/src/tool/lsp.ts +113 -0
  417. package/src/tool/lsp.txt +24 -0
  418. package/src/tool/mcp-websearch.ts +96 -0
  419. package/src/tool/plan-enter.txt +14 -0
  420. package/src/tool/plan-exit.txt +13 -0
  421. package/src/tool/plan.ts +79 -0
  422. package/src/tool/question.ts +44 -0
  423. package/src/tool/question.txt +10 -0
  424. package/src/tool/read.ts +386 -0
  425. package/src/tool/read.txt +14 -0
  426. package/src/tool/registry.ts +440 -0
  427. package/src/tool/schema.ts +14 -0
  428. package/src/tool/shell/id.ts +19 -0
  429. package/src/tool/shell/prompt.ts +307 -0
  430. package/src/tool/shell/shell.txt +21 -0
  431. package/src/tool/shell.ts +657 -0
  432. package/src/tool/skill.ts +71 -0
  433. package/src/tool/skill.txt +5 -0
  434. package/src/tool/task.ts +346 -0
  435. package/src/tool/task.txt +19 -0
  436. package/src/tool/todo.ts +57 -0
  437. package/src/tool/todowrite.txt +44 -0
  438. package/src/tool/tool.ts +183 -0
  439. package/src/tool/truncate.ts +158 -0
  440. package/src/tool/truncation-dir.ts +4 -0
  441. package/src/tool/webfetch.ts +192 -0
  442. package/src/tool/webfetch.txt +13 -0
  443. package/src/tool/websearch.ts +143 -0
  444. package/src/tool/websearch.txt +14 -0
  445. package/src/tool/write.ts +104 -0
  446. package/src/tool/write.txt +8 -0
  447. package/src/util/archive.ts +17 -0
  448. package/src/util/bom.ts +27 -0
  449. package/src/util/data-url.ts +9 -0
  450. package/src/util/defer.ts +10 -0
  451. package/src/util/effect-http-client.ts +11 -0
  452. package/src/util/error.ts +1 -0
  453. package/src/util/filesystem.ts +251 -0
  454. package/src/util/html.ts +8 -0
  455. package/src/util/iife.ts +3 -0
  456. package/src/util/lazy.ts +20 -0
  457. package/src/util/local-context.ts +25 -0
  458. package/src/util/locale.ts +2 -0
  459. package/src/util/media.ts +26 -0
  460. package/src/util/process.ts +177 -0
  461. package/src/util/proxy-env.ts +72 -0
  462. package/src/util/queue.ts +32 -0
  463. package/src/util/record.ts +1 -0
  464. package/src/util/repository.ts +232 -0
  465. package/src/util/rpc.ts +66 -0
  466. package/src/util/signal.ts +12 -0
  467. package/src/util/timeout.ts +13 -0
  468. package/src/util/token.ts +1 -0
  469. package/src/util/wildcard.ts +59 -0
  470. package/src/worktree/index.ts +654 -0
@@ -0,0 +1,270 @@
1
+ import WebSocket from "ws"
2
+ import { ProviderError } from "@/provider/error"
3
+ import { isRecord } from "@/util/record"
4
+ import { OpenAIWebSocket } from "./ws"
5
+
6
+ export const TITLE_HEADER = "x-rimuru-title"
7
+
8
+ export interface CreateWebSocketFetchOptions {
9
+ httpFetch?: typeof globalThis.fetch
10
+ url?: string
11
+ connectTimeout?: number
12
+ idleTimeout?: number
13
+ maxConnectionAge?: number
14
+ streamRetries?: number
15
+ }
16
+
17
+ interface PoolEntry {
18
+ socket?: WebSocket
19
+ connectedAt?: number
20
+ lastUsedAt: number
21
+ busy: boolean
22
+ fallback: boolean
23
+ streamFailures: number
24
+ }
25
+
26
+ const DEFAULT_CONNECT_TIMEOUT = 15_000
27
+ const DEFAULT_IDLE_TIMEOUT = 5 * 60 * 1000
28
+ const DEFAULT_MAX_CONNECTION_AGE = 55 * 60 * 1000
29
+ const CONNECTION_LIMIT_REACHED_CODE = "websocket_connection_limit_reached"
30
+
31
+ export function createWebSocketFetch(options?: CreateWebSocketFetchOptions) {
32
+ const httpFetch = options?.httpFetch ?? globalThis.fetch
33
+ const pool = new Map<string, PoolEntry>()
34
+ const connectTimeout = options?.connectTimeout ?? DEFAULT_CONNECT_TIMEOUT
35
+ const idleTimeout = options?.idleTimeout ?? DEFAULT_IDLE_TIMEOUT
36
+ const maxConnectionAge = options?.maxConnectionAge ?? DEFAULT_MAX_CONNECTION_AGE
37
+ const streamRetries = options?.streamRetries ?? 5
38
+ const pruneTimer = setInterval(() => prune(), Math.min(idleTimeout, 60_000))
39
+ if (typeof pruneTimer === "object" && "unref" in pruneTimer && typeof pruneTimer.unref === "function") {
40
+ pruneTimer.unref()
41
+ }
42
+
43
+ async function websocketFetch(input: RequestInfo | URL, init?: RequestInit): Promise<Response> {
44
+ const url = input instanceof URL ? input.toString() : typeof input === "string" ? input : input.url
45
+ const internalHeaders = OpenAIWebSocket.normalizeHeaders(init?.headers)
46
+ const httpInit = withoutInternalHeaders(init)
47
+
48
+ if (init?.method !== "POST" || !new URL(url).pathname.endsWith("/responses")) {
49
+ return httpFetch(input, httpInit)
50
+ }
51
+
52
+ const body = (() => {
53
+ try {
54
+ if (typeof init?.body !== "string") return undefined
55
+ const parsed = JSON.parse(init.body)
56
+ return typeof parsed === "object" && parsed !== null ? parsed : undefined
57
+ } catch {
58
+ return undefined
59
+ }
60
+ })()
61
+ if (!body?.stream) return httpFetch(input, httpInit)
62
+ if (internalHeaders[TITLE_HEADER] === "true") {
63
+ return httpFetch(input, httpInit)
64
+ }
65
+
66
+ const sessionID = internalHeaders["x-session-affinity"] ?? internalHeaders["session-id"]
67
+ if (!sessionID) {
68
+ return httpFetch(input, httpInit)
69
+ }
70
+ const key = `${sessionID}:conversation`
71
+
72
+ const entry = pool.get(key) ?? { lastUsedAt: Date.now(), busy: false, fallback: false, streamFailures: 0 }
73
+ pool.set(key, entry)
74
+
75
+ if (entry.fallback) {
76
+ return httpFetch(input, httpInit)
77
+ }
78
+ if (entry.busy) {
79
+ return httpFetch(input, httpInit)
80
+ }
81
+
82
+ entry.busy = true
83
+ entry.lastUsedAt = Date.now()
84
+ try {
85
+ entry.socket = await socket(
86
+ entry,
87
+ options?.url ?? url,
88
+ OpenAIWebSocket.normalizeHeaders(httpInit?.headers),
89
+ connectTimeout,
90
+ maxConnectionAge,
91
+ init?.signal,
92
+ )
93
+ let resolveFirstEvent: (event: boolean | OpenAIWebSocket.WrappedError) => void = () => {}
94
+ let rejectFirstEvent: (error: Error) => void = () => {}
95
+ const firstEvent = new Promise<boolean | OpenAIWebSocket.WrappedError>((resolve, reject) => {
96
+ resolveFirstEvent = resolve
97
+ rejectFirstEvent = reject
98
+ })
99
+ const response = OpenAIWebSocket.streamResponsesWebSocket({
100
+ socket: entry.socket,
101
+ body,
102
+ idleTimeout,
103
+ signal: init?.signal ?? undefined,
104
+ onFirstEvent: (error) => resolveFirstEvent(error ?? true),
105
+ onTerminal: (event) => {
106
+ entry.busy = false
107
+ entry.lastUsedAt = Date.now()
108
+ entry.streamFailures = 0
109
+ if (event.type !== "response.completed" && event.type !== "response.done") {
110
+ invalidate(entry)
111
+ }
112
+ },
113
+ onConnectionInvalid: (error) => {
114
+ entry.busy = false
115
+ entry.lastUsedAt = Date.now()
116
+ if (!entry.fallback) recordStreamFailure(entry)
117
+ invalidate(entry)
118
+ resolveFirstEvent(false)
119
+ },
120
+ onAbort: (error) => {
121
+ entry.busy = false
122
+ entry.lastUsedAt = Date.now()
123
+ entry.streamFailures = 0
124
+ invalidate(entry)
125
+ rejectFirstEvent(error)
126
+ },
127
+ onRetryableTerminal: async (event) => {
128
+ const error = connectionLimitError(event)
129
+ if (!error) return undefined
130
+ throw error
131
+ },
132
+ })
133
+ const first = await firstEvent
134
+ if (first !== false) {
135
+ if (first === true || first.status < 200 || first.status > 599) return response
136
+ return new Response(first.body, {
137
+ status: first.status,
138
+ headers: { "content-type": "application/json", ...first.headers },
139
+ })
140
+ }
141
+ if (!entry.fallback) return response
142
+ return httpFetch(input, httpInit)
143
+ } catch (error) {
144
+ entry.busy = false
145
+ entry.lastUsedAt = Date.now()
146
+ if (OpenAIWebSocket.isAbortError(error)) {
147
+ entry.streamFailures = 0
148
+ invalidate(entry)
149
+ throw error
150
+ }
151
+
152
+ recordStreamFailure(entry)
153
+ invalidate(entry)
154
+ if (entry.fallback) return httpFetch(input, httpInit)
155
+ return failedResponse(
156
+ new ProviderError.ResponseStreamError(error instanceof Error ? error.message : String(error), {
157
+ cause: error,
158
+ }),
159
+ )
160
+ }
161
+ }
162
+
163
+ function recordStreamFailure(entry: PoolEntry) {
164
+ entry.streamFailures++
165
+ // Codex counts retries after the initial failed WebSocket attempt.
166
+ if (entry.streamFailures > streamRetries) entry.fallback = true
167
+ }
168
+
169
+ function prune() {
170
+ const now = Date.now()
171
+ for (const [key, entry] of pool) {
172
+ if (entry.busy) continue
173
+ if (entry.fallback) continue
174
+ if (now - entry.lastUsedAt < idleTimeout) continue
175
+ invalidate(entry)
176
+ pool.delete(key)
177
+ }
178
+ }
179
+
180
+ function close() {
181
+ clearInterval(pruneTimer)
182
+ for (const entry of pool.values()) invalidate(entry)
183
+ pool.clear()
184
+ }
185
+
186
+ function remove(sessionID: string) {
187
+ const key = `${sessionID}:conversation`
188
+ const entry = pool.get(key)
189
+ if (!entry) return
190
+ invalidate(entry)
191
+ pool.delete(key)
192
+ }
193
+
194
+ return Object.assign(websocketFetch, { close, remove })
195
+ }
196
+
197
+ function connectionLimitError(event: Record<string, unknown>) {
198
+ if (event.type !== "error" || !isRecord(event.error) || event.error.code !== CONNECTION_LIMIT_REACHED_CODE) return
199
+ return new Error(typeof event.error.message === "string" ? event.error.message : CONNECTION_LIMIT_REACHED_CODE)
200
+ }
201
+
202
+ function failedResponse(error: ProviderError.ResponseStreamError) {
203
+ return new Response(
204
+ new ReadableStream({
205
+ start(controller) {
206
+ controller.error(error)
207
+ },
208
+ }),
209
+ {
210
+ status: 200,
211
+ headers: { "content-type": "text/event-stream" },
212
+ },
213
+ )
214
+ }
215
+
216
+ async function socket(
217
+ entry: PoolEntry,
218
+ url: string,
219
+ headers: Record<string, string>,
220
+ connectTimeout: number,
221
+ maxConnectionAge: number,
222
+ signal?: AbortSignal | null,
223
+ ) {
224
+ if (
225
+ entry.socket?.readyState === WebSocket.OPEN &&
226
+ entry.connectedAt &&
227
+ Date.now() - entry.connectedAt < maxConnectionAge
228
+ ) {
229
+ return entry.socket
230
+ }
231
+
232
+ invalidate(entry)
233
+ const next = await OpenAIWebSocket.connectResponsesWebSocket({
234
+ url: OpenAIWebSocket.toWebSocketUrl(url),
235
+ headers,
236
+ timeout: connectTimeout,
237
+ signal: signal ?? undefined,
238
+ })
239
+ entry.connectedAt = Date.now()
240
+ return next
241
+ }
242
+
243
+ function invalidate(entry: PoolEntry) {
244
+ if (entry.socket) {
245
+ entry.socket.on("error", () => {})
246
+ entry.socket.terminate()
247
+ entry.socket = undefined
248
+ }
249
+ entry.connectedAt = undefined
250
+ }
251
+
252
+ export function withoutInternalHeaders<T extends { headers?: HeadersInit }>(init: T | undefined): T | undefined {
253
+ if (!init?.headers) return init
254
+ if (init.headers instanceof Headers) {
255
+ const headers = new Headers(init.headers)
256
+ headers.delete(TITLE_HEADER)
257
+ return { ...init, headers }
258
+ }
259
+
260
+ if (Array.isArray(init.headers)) {
261
+ return { ...init, headers: init.headers.filter((item) => item[0].toLowerCase() !== TITLE_HEADER) }
262
+ }
263
+
264
+ return {
265
+ ...init,
266
+ headers: Object.fromEntries(Object.entries(init.headers).filter(([key]) => key.toLowerCase() !== TITLE_HEADER)),
267
+ }
268
+ }
269
+
270
+ export * as OpenAIWebSocketPool from "./ws-pool"
@@ -0,0 +1,381 @@
1
+ // Low-level OpenAI Responses WebSocket protocol helpers. Session pooling,
2
+ // fallback, and continuation state intentionally live above this file.
3
+
4
+ import WebSocket from "ws"
5
+ import { APICallError } from "ai"
6
+ import { ProviderError } from "@/provider/error"
7
+ import { errorMessage } from "@/util/error"
8
+ import { ProxyEnv } from "@/util/proxy-env"
9
+ import { isRecord } from "@/util/record"
10
+
11
+ export const PROTOCOL_HEADER = "responses_websockets=2026-02-06"
12
+
13
+ export interface ConnectResponsesWebSocketOptions {
14
+ url: string
15
+ headers: Record<string, string>
16
+ timeout?: number
17
+ signal?: AbortSignal
18
+ }
19
+
20
+ export interface StreamResponsesWebSocketOptions {
21
+ socket: WebSocket
22
+ body: Record<string, unknown>
23
+ idleTimeout?: number
24
+ signal?: AbortSignal
25
+ onFirstEvent?: (error?: WrappedError) => void
26
+ onComplete?: (event: Record<string, unknown>) => void
27
+ onTerminal?: (event: Record<string, unknown>) => void
28
+ onRetryableTerminal?: (event: Record<string, unknown>) => Promise<WebSocket | undefined>
29
+ onConnectionInvalid?: (error: ProviderError.ResponseStreamError) => void
30
+ onAbort?: (error: Error) => void
31
+ }
32
+
33
+ export interface WrappedError {
34
+ status: number
35
+ headers?: Record<string, string>
36
+ body: string
37
+ }
38
+
39
+ export function toWebSocketUrl(url: string) {
40
+ return url.replace(/^http/, "ws")
41
+ }
42
+
43
+ export function normalizeHeaders(headers: HeadersInit | undefined): Record<string, string> {
44
+ const result: Record<string, string> = {}
45
+ if (!headers) return result
46
+
47
+ if (headers instanceof Headers) {
48
+ headers.forEach((value, key) => {
49
+ result[key.toLowerCase()] = value
50
+ })
51
+ return result
52
+ }
53
+
54
+ if (Array.isArray(headers)) {
55
+ for (const [key, value] of headers) {
56
+ result[key.toLowerCase()] = value
57
+ }
58
+ return result
59
+ }
60
+
61
+ for (const [key, value] of Object.entries(headers)) {
62
+ if (value != null) result[key.toLowerCase()] = value
63
+ }
64
+ return result
65
+ }
66
+
67
+ export function isAbortError(error: unknown): error is DOMException {
68
+ return error instanceof DOMException && error.name === "AbortError"
69
+ }
70
+
71
+ export function connectResponsesWebSocket(options: ConnectResponsesWebSocketOptions) {
72
+ return new Promise<WebSocket>((resolve, reject) => {
73
+ if (options.signal?.aborted) {
74
+ reject(abortError(options.signal))
75
+ return
76
+ }
77
+
78
+ const headers: Record<string, string> = {
79
+ ...options.headers,
80
+ "openai-beta": options.headers["openai-beta"] ?? PROTOCOL_HEADER,
81
+ }
82
+ delete headers["content-length"]
83
+
84
+ // Bun does not apply HTTP(S)_PROXY to WebSockets unless the proxy is supplied explicitly.
85
+ const proxy =
86
+ typeof Bun === "undefined"
87
+ ? undefined
88
+ : ProxyEnv.getProxyForUrl(options.url.replace(/^wss:/, "https:").replace(/^ws:/, "http:"))
89
+ const connect = { headers, ...(proxy ? { proxy } : {}) }
90
+ const socket = new WebSocket(options.url, connect)
91
+ const timeout = options.timeout
92
+ ? setTimeout(() => {
93
+ cleanup()
94
+ socket.on("error", () => {})
95
+ socket.terminate()
96
+ reject(new Error("WebSocket connect timed out"))
97
+ }, options.timeout)
98
+ : undefined
99
+
100
+ function cleanup() {
101
+ if (timeout) clearTimeout(timeout)
102
+ socket.off("open", onOpen)
103
+ socket.off("error", onError)
104
+ socket.off("close", onClose)
105
+ options.signal?.removeEventListener("abort", onAbort)
106
+ }
107
+
108
+ function onOpen() {
109
+ cleanup()
110
+ resolve(socket)
111
+ }
112
+
113
+ function onError(error: unknown) {
114
+ socket.on("error", () => {})
115
+ cleanup()
116
+ reject(error instanceof Error ? error : new Error(errorMessage(error), { cause: error }))
117
+ }
118
+
119
+ function onClose(code: number, reason: Buffer) {
120
+ cleanup()
121
+ reject(new Error(closeMessage("WebSocket closed before open", code, reason)))
122
+ }
123
+
124
+ function onAbort() {
125
+ cleanup()
126
+ socket.on("error", () => {})
127
+ socket.terminate()
128
+ reject(abortError(options.signal))
129
+ }
130
+
131
+ socket.once("open", onOpen)
132
+ socket.once("error", onError)
133
+ socket.once("close", onClose)
134
+ options.signal?.addEventListener("abort", onAbort, { once: true })
135
+ })
136
+ }
137
+
138
+ export function streamResponsesWebSocket(options: StreamResponsesWebSocketOptions) {
139
+ const encoder = new TextEncoder()
140
+
141
+ let socket = options.socket
142
+ let controller: ReadableStreamDefaultController<Uint8Array> | undefined
143
+ let cleanupSocket = () => {}
144
+ let completed = false
145
+ let emitted = false
146
+ let idleTimer: ReturnType<typeof setTimeout> | undefined
147
+
148
+ function cleanup() {
149
+ if (idleTimer) clearTimeout(idleTimer)
150
+ cleanupSocket()
151
+ options.signal?.removeEventListener("abort", onAbort)
152
+ }
153
+
154
+ function terminateSocket(target = socket) {
155
+ target.on("error", () => {})
156
+ target.terminate()
157
+ }
158
+
159
+ function closeCompleted() {
160
+ cleanup()
161
+ controller?.enqueue(encoder.encode("data: [DONE]\n\n"))
162
+ controller?.close()
163
+ }
164
+
165
+ function invalidate(error: ProviderError.ResponseStreamError) {
166
+ if (completed) return
167
+ completed = true
168
+ cleanup()
169
+ options.onConnectionInvalid?.(error)
170
+ controller?.error(error)
171
+ }
172
+
173
+ function resetIdleTimeout(message: string) {
174
+ if (completed) return
175
+ if (!options.idleTimeout) return
176
+ if (idleTimer) clearTimeout(idleTimer)
177
+ idleTimer = setTimeout(() => invalidate(new ProviderError.ResponseStreamError(message)), options.idleTimeout)
178
+ }
179
+
180
+ async function onMessage(data: WebSocket.RawData, isBinary: boolean) {
181
+ if (completed) return
182
+ if (isBinary) {
183
+ invalidate(new ProviderError.ResponseStreamError("Unexpected binary WebSocket frame"))
184
+ return
185
+ }
186
+
187
+ const text = data.toString()
188
+ const event = (() => {
189
+ try {
190
+ const parsed = JSON.parse(text)
191
+ return typeof parsed === "object" && parsed !== null ? parsed : undefined
192
+ } catch {
193
+ return undefined
194
+ }
195
+ })()
196
+
197
+ if (event?.type === "error" && options.onRetryableTerminal) {
198
+ cleanupSocket()
199
+ if (idleTimer) clearTimeout(idleTimer)
200
+ idleTimer = undefined
201
+ try {
202
+ const next = await options.onRetryableTerminal(event)
203
+ if (completed) {
204
+ if (next) terminateSocket(next)
205
+ return
206
+ }
207
+ if (next) {
208
+ attach(next)
209
+ return
210
+ }
211
+ } catch (error) {
212
+ invalidate(
213
+ new ProviderError.ResponseStreamError(error instanceof Error ? error.message : String(error), {
214
+ cause: error,
215
+ }),
216
+ )
217
+ return
218
+ }
219
+ }
220
+
221
+ const wrappedError = parseWrappedError(event, text)
222
+ if (wrappedError && event) {
223
+ if (!emitted) options.onFirstEvent?.(wrappedError)
224
+ completed = true
225
+ cleanup()
226
+ options.onTerminal?.(event)
227
+ controller?.error(
228
+ new APICallError({
229
+ message: wrappedError.message,
230
+ url: socket.url,
231
+ requestBodyValues: options.body,
232
+ statusCode: wrappedError.status,
233
+ responseHeaders: wrappedError.headers,
234
+ responseBody: wrappedError.body,
235
+ }),
236
+ )
237
+ return
238
+ }
239
+
240
+ if (!emitted) options.onFirstEvent?.()
241
+ controller?.enqueue(
242
+ encoder.encode(
243
+ `${text
244
+ .split(/\r?\n/)
245
+ .map((line) => `data: ${line}`)
246
+ .join("\n")}\n\n`,
247
+ ),
248
+ )
249
+ emitted = true
250
+ resetIdleTimeout("idle timeout waiting for websocket")
251
+
252
+ if (!event) return
253
+
254
+ if (event.type === "response.completed" || event.type === "response.done") {
255
+ completed = true
256
+ options.onComplete?.(event)
257
+ options.onTerminal?.(event)
258
+ closeCompleted()
259
+ return
260
+ }
261
+
262
+ if (event.type === "response.failed" || event.type === "response.incomplete" || event.type === "error") {
263
+ completed = true
264
+ options.onTerminal?.(event)
265
+ closeCompleted()
266
+ }
267
+ }
268
+
269
+ function onError(error: Error) {
270
+ invalidate(new ProviderError.ResponseStreamError(error.message, { cause: error }))
271
+ }
272
+
273
+ function onClose(code: number, reason: Buffer) {
274
+ if (completed) return
275
+ invalidate(
276
+ new ProviderError.ResponseStreamError(closeMessage("WebSocket closed before response.completed", code, reason)),
277
+ )
278
+ }
279
+
280
+ function onAbort() {
281
+ const error = abortError(options.signal)
282
+ if (completed) return
283
+ completed = true
284
+ cleanup()
285
+ terminateSocket()
286
+ options.onAbort?.(error)
287
+ controller?.error(error)
288
+ }
289
+
290
+ function onCancel(reason: unknown) {
291
+ if (completed) return
292
+ completed = true
293
+ cleanup()
294
+ terminateSocket()
295
+ options.onAbort?.(cancelError(reason))
296
+ }
297
+
298
+ function attach(next: WebSocket) {
299
+ cleanupSocket()
300
+ socket = next
301
+ socket.on("message", onMessage)
302
+ socket.once("error", onError)
303
+ socket.once("close", onClose)
304
+ cleanupSocket = () => {
305
+ socket.off("message", onMessage)
306
+ socket.off("error", onError)
307
+ socket.off("close", onClose)
308
+ }
309
+ const { stream: _stream, background: _background, ...payload } = options.body
310
+ resetIdleTimeout("idle timeout sending websocket request")
311
+ socket.send(JSON.stringify({ type: "response.create", ...payload }), (error) => {
312
+ if (completed) return
313
+ resetIdleTimeout("idle timeout waiting for websocket")
314
+ if (error) invalidate(new ProviderError.ResponseStreamError(error.message, { cause: error }))
315
+ })
316
+ }
317
+
318
+ return new Response(
319
+ new ReadableStream<Uint8Array>({
320
+ start(next) {
321
+ controller = next
322
+ options.signal?.addEventListener("abort", onAbort, { once: true })
323
+
324
+ if (options.signal?.aborted) {
325
+ onAbort()
326
+ return
327
+ }
328
+
329
+ attach(socket)
330
+ },
331
+ cancel(reason) {
332
+ onCancel(reason)
333
+ },
334
+ }),
335
+ {
336
+ status: 200,
337
+ headers: { "content-type": "text/event-stream" },
338
+ },
339
+ )
340
+ }
341
+
342
+ function parseWrappedError(event: Record<string, unknown> | undefined, body: string) {
343
+ if (event?.type !== "error") return
344
+ const status = event.status ?? event.status_code
345
+ if (typeof status !== "number" || (status >= 200 && status < 300)) return
346
+ return {
347
+ status,
348
+ headers: isRecord(event.headers)
349
+ ? Object.fromEntries(
350
+ Object.entries(event.headers).flatMap(([key, value]) =>
351
+ typeof value === "string" || typeof value === "number" || typeof value === "boolean"
352
+ ? [[key, String(value)]]
353
+ : [],
354
+ ),
355
+ )
356
+ : undefined,
357
+ body,
358
+ message: isRecord(event.error) && typeof event.error.message === "string" ? event.error.message : `${status}`,
359
+ }
360
+ }
361
+
362
+ function cancelError(reason: unknown) {
363
+ if (isAbortError(reason)) return reason
364
+ if (reason instanceof Error) return reason
365
+ return new DOMException(typeof reason === "string" ? reason : "Aborted", "AbortError")
366
+ }
367
+
368
+ function abortError(signal: AbortSignal | undefined) {
369
+ const reason = signal?.reason
370
+ if (isAbortError(reason)) return reason
371
+ return new DOMException(reason instanceof Error ? reason.message : "Aborted", "AbortError")
372
+ }
373
+
374
+ function closeMessage(message: string, code: number, reason: Buffer) {
375
+ const details = [`code ${code}`]
376
+ if (code === 1009) details.push("message too big")
377
+ if (reason.length > 0) details.push(reason.toString())
378
+ return `${message} (${details.join(": ")})`
379
+ }
380
+
381
+ export * as OpenAIWebSocket from "./ws"
@@ -0,0 +1,24 @@
1
+ export * as PluginPtyEnvironment from "./pty-environment"
2
+
3
+ import { PtyEnvironment } from "@rimurucode-ai/server/pty-environment"
4
+ import { Effect, Layer } from "effect"
5
+ import { InstanceStore } from "@/project/instance-store"
6
+ import { Plugin } from "."
7
+
8
+ export const layer = Layer.effect(
9
+ PtyEnvironment.Service,
10
+ Effect.gen(function* () {
11
+ const plugin = yield* Plugin.Service
12
+ const instances = yield* InstanceStore.Service
13
+ return PtyEnvironment.Service.of({
14
+ get: Effect.fn("PtyEnvironment.get")(function* (input) {
15
+ return yield* instances.provide(
16
+ { directory: input.directory },
17
+ plugin
18
+ .trigger("shell.env", { cwd: input.cwd }, { env: {} as Record<string, string> })
19
+ .pipe(Effect.map((result) => result.env)),
20
+ )
21
+ }),
22
+ })
23
+ }),
24
+ )