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,650 @@
1
+ import path from "path"
2
+ import { pathToFileURL, fileURLToPath } from "url"
3
+ import { createMessageConnection, StreamMessageReader, StreamMessageWriter } from "vscode-jsonrpc/node"
4
+ import type { Diagnostic as VSCodeDiagnostic } from "vscode-languageserver-types"
5
+ import { Process } from "@/util/process"
6
+ import { LANGUAGE_EXTENSIONS } from "./language"
7
+ import { Effect, Schema } from "effect"
8
+ import type * as LSPServer from "./server"
9
+ import { withTimeout } from "../util/timeout"
10
+ import { Filesystem } from "@/util/filesystem"
11
+ import type { InstanceContext } from "@/project/instance-context"
12
+
13
+ const DIAGNOSTICS_DEBOUNCE_MS = 150
14
+ const DIAGNOSTICS_DOCUMENT_WAIT_TIMEOUT_MS = 5_000
15
+ const DIAGNOSTICS_FULL_WAIT_TIMEOUT_MS = 10_000
16
+ const DIAGNOSTICS_REQUEST_TIMEOUT_MS = 3_000
17
+
18
+ const INITIALIZE_TIMEOUT_MS = 45_000
19
+
20
+ // LSP spec constants
21
+ const FILE_CHANGE_CREATED = 1
22
+ const FILE_CHANGE_CHANGED = 2
23
+ const TEXT_DOCUMENT_SYNC_INCREMENTAL = 2
24
+
25
+ export type Info = NonNullable<Awaited<ReturnType<typeof create>>>
26
+
27
+ export type Diagnostic = VSCodeDiagnostic
28
+
29
+ export class InitializeError extends Schema.TaggedErrorClass<InitializeError>()("LSPInitializeError", {
30
+ serverID: Schema.String,
31
+ cause: Schema.optional(Schema.Defect),
32
+ }) {}
33
+
34
+ type DocumentDiagnosticReport = {
35
+ items?: Diagnostic[]
36
+ relatedDocuments?: Record<string, DocumentDiagnosticReport>
37
+ }
38
+
39
+ type WorkspaceDiagnosticReport = {
40
+ items?: {
41
+ uri?: string
42
+ items?: Diagnostic[]
43
+ }[]
44
+ }
45
+
46
+ type DiagnosticRequestResult = {
47
+ handled: boolean
48
+ matched: boolean
49
+ byFile: Map<string, Diagnostic[]>
50
+ }
51
+
52
+ type CapabilityRegistration = {
53
+ id: string
54
+ method: string
55
+ registerOptions?: {
56
+ identifier?: string
57
+ workspaceDiagnostics?: boolean
58
+ }
59
+ }
60
+
61
+ type ServerCapabilities = {
62
+ textDocumentSync?:
63
+ | number
64
+ | {
65
+ change?: number
66
+ }
67
+ diagnosticProvider?: unknown
68
+ [key: string]: unknown
69
+ }
70
+
71
+ function getFilePath(uri: string) {
72
+ if (!uri.startsWith("file://")) return
73
+ return Filesystem.normalizePath(fileURLToPath(uri))
74
+ }
75
+
76
+ function getSyncKind(capabilities?: ServerCapabilities) {
77
+ if (!capabilities) return
78
+ const sync = capabilities.textDocumentSync
79
+ if (typeof sync === "number") return sync
80
+ return sync?.change
81
+ }
82
+
83
+ function endPosition(text: string) {
84
+ const lines = text.split(/\r\n|\r|\n/)
85
+ return {
86
+ line: lines.length - 1,
87
+ character: lines.at(-1)?.length ?? 0,
88
+ }
89
+ }
90
+
91
+ function dedupeDiagnostics(items: Diagnostic[]) {
92
+ const seen = new Set<string>()
93
+ return items.filter((item) => {
94
+ const key = JSON.stringify({
95
+ code: item.code,
96
+ severity: item.severity,
97
+ message: item.message,
98
+ source: item.source,
99
+ range: item.range,
100
+ })
101
+ if (seen.has(key)) return false
102
+ seen.add(key)
103
+ return true
104
+ })
105
+ }
106
+
107
+ function configurationValue(settings: unknown, section?: string) {
108
+ if (!section) return settings ?? null
109
+ const result = section.split(".").reduce<unknown>((acc, key) => {
110
+ if (!acc || typeof acc !== "object" || !(key in acc)) return undefined
111
+ return (acc as Record<string, unknown>)[key]
112
+ }, settings)
113
+ return result ?? null
114
+ }
115
+
116
+ // TypeScript's built-in LSP pushes diagnostics aggressively on first open.
117
+ // We seed the push cache on the very first publish so waitForFreshPush can
118
+ // resolve immediately instead of waiting for a second debounced push.
119
+ function shouldSeedDiagnosticsOnFirstPush(serverID: string) {
120
+ return serverID === "typescript"
121
+ }
122
+
123
+ export async function create(input: {
124
+ serverID: string
125
+ server: LSPServer.Handle
126
+ root: string
127
+ directory: string
128
+ instance: InstanceContext
129
+ }) {
130
+ const instance = input.instance
131
+
132
+ const connection = createMessageConnection(
133
+ new StreamMessageReader(input.server.process.stdout as any),
134
+ new StreamMessageWriter(input.server.process.stdin as any),
135
+ )
136
+ input.server.process.stderr?.resume()
137
+ // --- Connection state ---
138
+
139
+ const pushDiagnostics = new Map<string, Diagnostic[]>()
140
+ const pullDiagnostics = new Map<string, Diagnostic[]>()
141
+ const published = new Map<string, { at: number; version?: number }>()
142
+ const diagnosticRegistrations = new Map<string, CapabilityRegistration>()
143
+ const registrationListeners = new Set<() => void>()
144
+ const diagnosticListeners = new Set<(input: { path: string; serverID: string }) => void>()
145
+ const mergedDiagnostics = (filePath: string) =>
146
+ dedupeDiagnostics([...(pushDiagnostics.get(filePath) ?? []), ...(pullDiagnostics.get(filePath) ?? [])])
147
+ const updatePushDiagnostics = (filePath: string, next: Diagnostic[]) => {
148
+ pushDiagnostics.set(filePath, next)
149
+ for (const listener of diagnosticListeners) listener({ path: filePath, serverID: input.serverID })
150
+ }
151
+ const updatePullDiagnostics = (filePath: string, next: Diagnostic[]) => {
152
+ pullDiagnostics.set(filePath, next)
153
+ }
154
+ const emitRegistrationChange = () => {
155
+ for (const listener of [...registrationListeners]) listener()
156
+ }
157
+
158
+ // --- LSP connection handlers ---
159
+
160
+ connection.onNotification("textDocument/publishDiagnostics", (params) => {
161
+ const filePath = getFilePath(params.uri)
162
+ if (!filePath) return
163
+ published.set(filePath, {
164
+ at: Date.now(),
165
+ version: typeof params.version === "number" ? params.version : undefined,
166
+ })
167
+ if (shouldSeedDiagnosticsOnFirstPush(input.serverID) && !pushDiagnostics.has(filePath)) {
168
+ pushDiagnostics.set(filePath, params.diagnostics)
169
+ return
170
+ }
171
+ updatePushDiagnostics(filePath, params.diagnostics)
172
+ })
173
+ connection.onRequest("window/workDoneProgress/create", (params) => {
174
+ return null
175
+ })
176
+ connection.onRequest("workspace/configuration", async (params) => {
177
+ const items = (params as { items?: { section?: string }[] }).items ?? []
178
+ return items.map((item) => configurationValue(input.server.initialization, item.section))
179
+ })
180
+ connection.onRequest("client/registerCapability", async (params) => {
181
+ const registrations = (params as { registrations?: CapabilityRegistration[] }).registrations ?? []
182
+ let changed = false
183
+ for (const registration of registrations) {
184
+ if (registration.method !== "textDocument/diagnostic") continue
185
+ diagnosticRegistrations.set(registration.id, registration)
186
+ changed = true
187
+ }
188
+ if (changed) emitRegistrationChange()
189
+ })
190
+ connection.onRequest("client/unregisterCapability", async (params) => {
191
+ const registrations = (params as { unregisterations?: { id: string; method: string }[] }).unregisterations ?? []
192
+ let changed = false
193
+ for (const registration of registrations) {
194
+ if (registration.method !== "textDocument/diagnostic") continue
195
+ diagnosticRegistrations.delete(registration.id)
196
+ changed = true
197
+ }
198
+ if (changed) emitRegistrationChange()
199
+ })
200
+ connection.onRequest("workspace/workspaceFolders", async () => [
201
+ {
202
+ name: "workspace",
203
+ uri: pathToFileURL(input.root).href,
204
+ },
205
+ ])
206
+ connection.onRequest("workspace/diagnostic/refresh", async () => null)
207
+ connection.listen()
208
+
209
+ // --- Initialize handshake ---
210
+
211
+ const initialized = await withTimeout(
212
+ connection.sendRequest<{ capabilities?: ServerCapabilities }>("initialize", {
213
+ rootUri: pathToFileURL(input.root).href,
214
+ processId: input.server.process.pid,
215
+ workspaceFolders: [
216
+ {
217
+ name: "workspace",
218
+ uri: pathToFileURL(input.root).href,
219
+ },
220
+ ],
221
+ initializationOptions: {
222
+ ...input.server.initialization,
223
+ },
224
+ capabilities: {
225
+ window: {
226
+ workDoneProgress: true,
227
+ },
228
+ workspace: {
229
+ configuration: true,
230
+ didChangeWatchedFiles: {
231
+ dynamicRegistration: true,
232
+ },
233
+ diagnostics: {
234
+ refreshSupport: false,
235
+ },
236
+ },
237
+ textDocument: {
238
+ synchronization: {
239
+ didOpen: true,
240
+ didChange: true,
241
+ },
242
+ diagnostic: {
243
+ dynamicRegistration: true,
244
+ relatedDocumentSupport: true,
245
+ },
246
+ publishDiagnostics: {
247
+ versionSupport: false,
248
+ },
249
+ },
250
+ },
251
+ }),
252
+ INITIALIZE_TIMEOUT_MS,
253
+ ).catch((err) => {
254
+ throw new InitializeError({ serverID: input.serverID, cause: err })
255
+ })
256
+
257
+ const syncKind = getSyncKind(initialized.capabilities)
258
+ const hasStaticPullDiagnostics = Boolean(initialized.capabilities?.diagnosticProvider)
259
+
260
+ await connection.sendNotification("initialized", {})
261
+
262
+ if (input.server.initialization) {
263
+ await connection.sendNotification("workspace/didChangeConfiguration", {
264
+ settings: input.server.initialization,
265
+ })
266
+ }
267
+
268
+ const files: Record<string, { version: number; text: string }> = {}
269
+
270
+ // --- Diagnostic helpers ---
271
+
272
+ const mergeResults = (filePath: string, results: DiagnosticRequestResult[]) => {
273
+ const handled = results.some((result) => result.handled)
274
+ const matched = results.some((result) => result.matched)
275
+ if (!handled) return { handled: false, matched: false }
276
+
277
+ const merged = new Map<string, Diagnostic[]>()
278
+ for (const result of results) {
279
+ for (const [target, items] of result.byFile.entries()) {
280
+ const existing = merged.get(target) ?? []
281
+ merged.set(target, existing.concat(items))
282
+ }
283
+ }
284
+
285
+ if (matched && !merged.has(filePath)) merged.set(filePath, [])
286
+ for (const [target, items] of merged.entries()) {
287
+ updatePullDiagnostics(target, dedupeDiagnostics(items))
288
+ }
289
+
290
+ return { handled, matched }
291
+ }
292
+
293
+ async function requestDiagnosticReport(filePath: string, identifier?: string): Promise<DiagnosticRequestResult> {
294
+ const report = await withTimeout(
295
+ connection.sendRequest<DocumentDiagnosticReport | null>("textDocument/diagnostic", {
296
+ ...(identifier ? { identifier } : {}),
297
+ textDocument: {
298
+ uri: pathToFileURL(filePath).href,
299
+ },
300
+ }),
301
+ DIAGNOSTICS_REQUEST_TIMEOUT_MS,
302
+ ).catch(() => null)
303
+ if (!report) return { handled: false, matched: false, byFile: new Map<string, Diagnostic[]>() }
304
+
305
+ const byFile = new Map<string, Diagnostic[]>()
306
+ const push = (target: string, items: Diagnostic[]) => {
307
+ const existing = byFile.get(target) ?? []
308
+ byFile.set(target, existing.concat(items))
309
+ }
310
+
311
+ let handled = false
312
+ let matched = false
313
+ if (Array.isArray(report.items)) {
314
+ push(filePath, report.items)
315
+ handled = true
316
+ matched = true
317
+ }
318
+ for (const [uri, related] of Object.entries(report.relatedDocuments ?? {})) {
319
+ const relatedPath = getFilePath(uri)
320
+ if (!relatedPath || !Array.isArray(related.items)) continue
321
+ push(relatedPath, related.items)
322
+ handled = true
323
+ matched = matched || relatedPath === filePath
324
+ }
325
+
326
+ return { handled, matched, byFile }
327
+ }
328
+
329
+ async function requestWorkspaceDiagnosticReport(
330
+ filePath: string,
331
+ identifier?: string,
332
+ ): Promise<DiagnosticRequestResult> {
333
+ const report = await withTimeout(
334
+ connection.sendRequest<WorkspaceDiagnosticReport | null>("workspace/diagnostic", {
335
+ ...(identifier ? { identifier } : {}),
336
+ previousResultIds: [],
337
+ }),
338
+ DIAGNOSTICS_REQUEST_TIMEOUT_MS,
339
+ ).catch(() => null)
340
+ if (!report) return { handled: false, matched: false, byFile: new Map<string, Diagnostic[]>() }
341
+
342
+ const byFile = new Map<string, Diagnostic[]>()
343
+ let matched = false
344
+ for (const item of report.items ?? []) {
345
+ const relatedPath = item.uri ? getFilePath(item.uri) : undefined
346
+ if (!relatedPath || !Array.isArray(item.items)) continue
347
+ const existing = byFile.get(relatedPath) ?? []
348
+ byFile.set(relatedPath, existing.concat(item.items))
349
+ matched = matched || relatedPath === filePath
350
+ }
351
+
352
+ return { handled: true, matched, byFile }
353
+ }
354
+
355
+ function documentPullState() {
356
+ const documentRegistrations = [...diagnosticRegistrations.values()].filter(
357
+ (registration) => registration.registerOptions?.workspaceDiagnostics !== true,
358
+ )
359
+ return {
360
+ documentIdentifiers: [
361
+ ...new Set(documentRegistrations.flatMap((registration) => registration.registerOptions?.identifier ?? [])),
362
+ ],
363
+ supported: hasStaticPullDiagnostics || documentRegistrations.length > 0,
364
+ }
365
+ }
366
+
367
+ function workspacePullState() {
368
+ const workspaceRegistrations = [...diagnosticRegistrations.values()].filter(
369
+ (registration) => registration.registerOptions?.workspaceDiagnostics === true,
370
+ )
371
+ return {
372
+ workspaceIdentifiers: [
373
+ ...new Set(workspaceRegistrations.flatMap((registration) => registration.registerOptions?.identifier ?? [])),
374
+ ],
375
+ supported: workspaceRegistrations.length > 0,
376
+ }
377
+ }
378
+
379
+ const hasCurrentFileDiagnostics = (filePath: string, results: DiagnosticRequestResult[]) =>
380
+ results.some((result) => (result.byFile.get(filePath)?.length ?? 0) > 0)
381
+
382
+ async function requestDiagnostics(
383
+ filePath: string,
384
+ requests: Promise<DiagnosticRequestResult>[],
385
+ done: (results: DiagnosticRequestResult[]) => boolean,
386
+ ) {
387
+ if (!requests.length) return { handled: false, matched: false }
388
+
389
+ const results: DiagnosticRequestResult[] = []
390
+ return new Promise<{ handled: boolean; matched: boolean }>((resolve) => {
391
+ let pending = requests.length
392
+ let resolved = false
393
+ const finish = (merged: { handled: boolean; matched: boolean }, force = false) => {
394
+ if (resolved) return
395
+ if (!force && !done(results)) return
396
+ resolved = true
397
+ resolve(merged)
398
+ }
399
+
400
+ for (const request of requests) {
401
+ request.then((result) => {
402
+ results.push(result)
403
+ pending -= 1
404
+ const merged = mergeResults(filePath, results)
405
+ finish(merged)
406
+ if (pending === 0) finish(merged, true)
407
+ })
408
+ }
409
+ })
410
+ }
411
+
412
+ // LATENCY-CRITICAL: dispatch identifier pulls in parallel and unblock once one
413
+ // batch already produced diagnostics for the current file. Let slower pulls keep
414
+ // merging in the background; do not sequence identifier-by-identifier, and do
415
+ // not add a post-match settle/debounce delay. See PR #23771.
416
+ async function requestDocumentDiagnostics(filePath: string) {
417
+ const state = documentPullState()
418
+ if (!state.supported) return { handled: false, matched: false }
419
+ return requestDiagnostics(
420
+ filePath,
421
+ [
422
+ requestDiagnosticReport(filePath),
423
+ ...state.documentIdentifiers.map((identifier) => requestDiagnosticReport(filePath, identifier)),
424
+ ],
425
+ (results) => hasCurrentFileDiagnostics(filePath, results),
426
+ )
427
+ }
428
+
429
+ async function requestFullDiagnostics(filePath: string) {
430
+ const documentState = documentPullState()
431
+ const workspaceState = workspacePullState()
432
+ if (!documentState.supported && !workspaceState.supported) return { handled: false, matched: false }
433
+ return mergeResults(
434
+ filePath,
435
+ await Promise.all([
436
+ ...(documentState.supported ? [requestDiagnosticReport(filePath)] : []),
437
+ ...documentState.documentIdentifiers.map((identifier) => requestDiagnosticReport(filePath, identifier)),
438
+ ...(workspaceState.supported ? [requestWorkspaceDiagnosticReport(filePath)] : []),
439
+ ...workspaceState.workspaceIdentifiers.map((identifier) =>
440
+ requestWorkspaceDiagnosticReport(filePath, identifier),
441
+ ),
442
+ ]),
443
+ )
444
+ }
445
+
446
+ function waitForRegistrationChange(timeout: number) {
447
+ if (timeout <= 0) return Promise.resolve(false)
448
+ return new Promise<boolean>((resolve) => {
449
+ let finished = false
450
+ let timer: ReturnType<typeof setTimeout> | undefined
451
+ const finish = (result: boolean) => {
452
+ if (finished) return
453
+ finished = true
454
+ if (timer) clearTimeout(timer)
455
+ registrationListeners.delete(listener)
456
+ resolve(result)
457
+ }
458
+ const listener = () => finish(true)
459
+ registrationListeners.add(listener)
460
+ timer = setTimeout(() => finish(false), timeout)
461
+ })
462
+ }
463
+
464
+ function waitForFreshPush(request: { path: string; version: number; after: number; timeout: number }) {
465
+ if (request.timeout <= 0) return Promise.resolve(false)
466
+ return new Promise<boolean>((resolve) => {
467
+ let finished = false
468
+ let debounceTimer: ReturnType<typeof setTimeout> | undefined
469
+ let timeoutTimer: ReturnType<typeof setTimeout> | undefined
470
+ let unsub: (() => void) | undefined
471
+ const finish = (result: boolean) => {
472
+ if (finished) return
473
+ finished = true
474
+ if (debounceTimer) clearTimeout(debounceTimer)
475
+ if (timeoutTimer) clearTimeout(timeoutTimer)
476
+ unsub?.()
477
+ resolve(result)
478
+ }
479
+ const schedule = () => {
480
+ const hit = published.get(request.path)
481
+ if (!hit) return
482
+ if (typeof hit.version === "number" && hit.version !== request.version) return
483
+ if (hit.at < request.after && hit.version !== request.version) return
484
+ if (debounceTimer) clearTimeout(debounceTimer)
485
+ debounceTimer = setTimeout(() => finish(true), Math.max(0, DIAGNOSTICS_DEBOUNCE_MS - (Date.now() - hit.at)))
486
+ }
487
+
488
+ timeoutTimer = setTimeout(() => finish(false), request.timeout)
489
+ const listener = (event: { path: string; serverID: string }) => {
490
+ if (event.path !== request.path || event.serverID !== input.serverID) return
491
+ schedule()
492
+ }
493
+ diagnosticListeners.add(listener)
494
+ unsub = () => diagnosticListeners.delete(listener)
495
+ schedule()
496
+ })
497
+ }
498
+
499
+ async function waitForDocumentDiagnostics(request: { path: string; version: number; after?: number }) {
500
+ const startedAt = request.after ?? Date.now()
501
+ const pushWait = waitForFreshPush({
502
+ path: request.path,
503
+ version: request.version,
504
+ after: startedAt,
505
+ timeout: DIAGNOSTICS_DOCUMENT_WAIT_TIMEOUT_MS,
506
+ })
507
+
508
+ while (Date.now() - startedAt < DIAGNOSTICS_DOCUMENT_WAIT_TIMEOUT_MS) {
509
+ const result = await requestDocumentDiagnostics(request.path)
510
+ if (result.matched) return
511
+ const remaining = DIAGNOSTICS_DOCUMENT_WAIT_TIMEOUT_MS - (Date.now() - startedAt)
512
+ if (remaining <= 0) return
513
+ const next = await Promise.race([
514
+ pushWait.then((ready) => (ready ? "push" : ("timeout" as const))),
515
+ waitForRegistrationChange(remaining).then((changed) => (changed ? "registration" : ("timeout" as const))),
516
+ ])
517
+ if (next !== "registration") return
518
+ }
519
+ }
520
+
521
+ async function waitForFullDiagnostics(request: { path: string; version: number; after?: number }) {
522
+ const startedAt = request.after ?? Date.now()
523
+ const pushWait = waitForFreshPush({
524
+ path: request.path,
525
+ version: request.version,
526
+ after: startedAt,
527
+ timeout: DIAGNOSTICS_FULL_WAIT_TIMEOUT_MS,
528
+ })
529
+
530
+ while (Date.now() - startedAt < DIAGNOSTICS_FULL_WAIT_TIMEOUT_MS) {
531
+ const result = await requestFullDiagnostics(request.path)
532
+ if (result.handled || result.matched) return
533
+ const remaining = DIAGNOSTICS_FULL_WAIT_TIMEOUT_MS - (Date.now() - startedAt)
534
+ if (remaining <= 0) return
535
+ const next = await Promise.race([
536
+ pushWait.then((ready) => (ready ? "push" : ("timeout" as const))),
537
+ waitForRegistrationChange(remaining).then((changed) => (changed ? "registration" : ("timeout" as const))),
538
+ ])
539
+ if (next !== "registration") return
540
+ }
541
+ }
542
+
543
+ // --- Public API ---
544
+
545
+ const result = {
546
+ root: input.root,
547
+ get serverID() {
548
+ return input.serverID
549
+ },
550
+ get connection() {
551
+ return connection
552
+ },
553
+ notify: {
554
+ async open(request: { path: string }) {
555
+ request.path = Filesystem.normalizePath(
556
+ path.isAbsolute(request.path) ? request.path : path.resolve(input.directory, request.path),
557
+ )
558
+ const text = await Filesystem.readText(request.path)
559
+ const extension = path.extname(request.path)
560
+ const languageId = LANGUAGE_EXTENSIONS[extension] ?? "plaintext"
561
+
562
+ const document = files[request.path]
563
+ if (document !== undefined) {
564
+ // Do not wipe diagnostics on didChange. Some servers (e.g. clangd) only
565
+ // re-emit diagnostics when the content actually changes, so clearing
566
+ // here would lose errors for no-op touchFile calls. Let the server's
567
+ // next push/pull overwrite naturally.
568
+ await connection.sendNotification("workspace/didChangeWatchedFiles", {
569
+ changes: [
570
+ {
571
+ uri: pathToFileURL(request.path).href,
572
+ type: FILE_CHANGE_CHANGED,
573
+ },
574
+ ],
575
+ })
576
+
577
+ const next = document.version + 1
578
+ files[request.path] = { version: next, text }
579
+ await connection.sendNotification("textDocument/didChange", {
580
+ textDocument: {
581
+ uri: pathToFileURL(request.path).href,
582
+ version: next,
583
+ },
584
+ contentChanges:
585
+ syncKind === TEXT_DOCUMENT_SYNC_INCREMENTAL
586
+ ? [
587
+ {
588
+ range: {
589
+ start: { line: 0, character: 0 },
590
+ end: endPosition(document.text),
591
+ },
592
+ text,
593
+ },
594
+ ]
595
+ : [{ text }],
596
+ })
597
+ return next
598
+ }
599
+
600
+ await connection.sendNotification("workspace/didChangeWatchedFiles", {
601
+ changes: [
602
+ {
603
+ uri: pathToFileURL(request.path).href,
604
+ type: FILE_CHANGE_CREATED,
605
+ },
606
+ ],
607
+ })
608
+
609
+ pushDiagnostics.delete(request.path)
610
+ pullDiagnostics.delete(request.path)
611
+ await connection.sendNotification("textDocument/didOpen", {
612
+ textDocument: {
613
+ uri: pathToFileURL(request.path).href,
614
+ languageId,
615
+ version: 0,
616
+ text,
617
+ },
618
+ })
619
+ files[request.path] = { version: 0, text }
620
+ return 0
621
+ },
622
+ },
623
+ get diagnostics() {
624
+ const result = new Map<string, Diagnostic[]>()
625
+ for (const key of new Set([...pushDiagnostics.keys(), ...pullDiagnostics.keys()])) {
626
+ result.set(key, mergedDiagnostics(key))
627
+ }
628
+ return result
629
+ },
630
+ async waitForDiagnostics(request: { path: string; version: number; mode?: "document" | "full"; after?: number }) {
631
+ const normalizedPath = Filesystem.normalizePath(
632
+ path.isAbsolute(request.path) ? request.path : path.resolve(input.directory, request.path),
633
+ )
634
+ if (request.mode === "document") {
635
+ await waitForDocumentDiagnostics({ path: normalizedPath, version: request.version, after: request.after })
636
+ return
637
+ }
638
+ await waitForFullDiagnostics({ path: normalizedPath, version: request.version, after: request.after })
639
+ },
640
+ async shutdown() {
641
+ connection.end()
642
+ connection.dispose()
643
+ await Process.stop(input.server.process)
644
+ },
645
+ }
646
+
647
+ return result
648
+ }
649
+
650
+ export * as LSPClient from "./client"
@@ -0,0 +1,29 @@
1
+ import * as LSPClient from "./client"
2
+
3
+ const MAX_PER_FILE = 20
4
+
5
+ export function pretty(diagnostic: LSPClient.Diagnostic) {
6
+ const severityMap = {
7
+ 1: "ERROR",
8
+ 2: "WARN",
9
+ 3: "INFO",
10
+ 4: "HINT",
11
+ }
12
+
13
+ const severity = severityMap[diagnostic.severity || 1]
14
+ const line = diagnostic.range.start.line + 1
15
+ const col = diagnostic.range.start.character + 1
16
+
17
+ return `${severity} [${line}:${col}] ${diagnostic.message}`
18
+ }
19
+
20
+ export function report(file: string, issues: LSPClient.Diagnostic[]) {
21
+ const errors = issues.filter((item) => item.severity === 1)
22
+ if (errors.length === 0) return ""
23
+ const limited = errors.slice(0, MAX_PER_FILE)
24
+ const more = errors.length - MAX_PER_FILE
25
+ const suffix = more > 0 ? `\n... and ${more} more` : ""
26
+ return `<diagnostics file="${file}">\n${limited.map(pretty).join("\n")}${suffix}\n</diagnostics>`
27
+ }
28
+
29
+ export * as Diagnostic from "./diagnostic"