pi-ui-extend 0.1.1

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 (420) hide show
  1. package/README.md +307 -0
  2. package/bin/pix.mjs +219 -0
  3. package/dist/app/app.d.ts +96 -0
  4. package/dist/app/app.js +871 -0
  5. package/dist/app/blink-controller.d.ts +23 -0
  6. package/dist/app/blink-controller.js +82 -0
  7. package/dist/app/cli.d.ts +8 -0
  8. package/dist/app/cli.js +83 -0
  9. package/dist/app/clipboard.d.ts +1 -0
  10. package/dist/app/clipboard.js +24 -0
  11. package/dist/app/command-controller.d.ts +18 -0
  12. package/dist/app/command-controller.js +58 -0
  13. package/dist/app/command-host.d.ts +44 -0
  14. package/dist/app/command-host.js +1 -0
  15. package/dist/app/command-model-actions.d.ts +12 -0
  16. package/dist/app/command-model-actions.js +176 -0
  17. package/dist/app/command-navigation-actions.d.ts +19 -0
  18. package/dist/app/command-navigation-actions.js +267 -0
  19. package/dist/app/command-registry.d.ts +32 -0
  20. package/dist/app/command-registry.js +225 -0
  21. package/dist/app/command-runtime.d.ts +5 -0
  22. package/dist/app/command-runtime.js +32 -0
  23. package/dist/app/command-session-actions.d.ts +20 -0
  24. package/dist/app/command-session-actions.js +295 -0
  25. package/dist/app/constants.d.ts +52 -0
  26. package/dist/app/constants.js +103 -0
  27. package/dist/app/conversation-entry-renderer.d.ts +21 -0
  28. package/dist/app/conversation-entry-renderer.js +81 -0
  29. package/dist/app/conversation-shell-renderer.d.ts +5 -0
  30. package/dist/app/conversation-shell-renderer.js +43 -0
  31. package/dist/app/conversation-tool-renderer.d.ts +16 -0
  32. package/dist/app/conversation-tool-renderer.js +216 -0
  33. package/dist/app/conversation-viewport.d.ts +55 -0
  34. package/dist/app/conversation-viewport.js +252 -0
  35. package/dist/app/dcp-stats.d.ts +2 -0
  36. package/dist/app/dcp-stats.js +116 -0
  37. package/dist/app/editor-layout-renderer.d.ts +31 -0
  38. package/dist/app/editor-layout-renderer.js +211 -0
  39. package/dist/app/editor-panels.d.ts +4 -0
  40. package/dist/app/editor-panels.js +130 -0
  41. package/dist/app/extension-actions-controller.d.ts +22 -0
  42. package/dist/app/extension-actions-controller.js +60 -0
  43. package/dist/app/extension-event-bus.d.ts +3 -0
  44. package/dist/app/extension-event-bus.js +23 -0
  45. package/dist/app/extension-ui-controller.d.ts +57 -0
  46. package/dist/app/extension-ui-controller.js +532 -0
  47. package/dist/app/file-link-opener.d.ts +2 -0
  48. package/dist/app/file-link-opener.js +66 -0
  49. package/dist/app/file-links.d.ts +10 -0
  50. package/dist/app/file-links.js +117 -0
  51. package/dist/app/guards.d.ts +3 -0
  52. package/dist/app/guards.js +9 -0
  53. package/dist/app/icons.d.ts +37 -0
  54. package/dist/app/icons.js +97 -0
  55. package/dist/app/id.d.ts +1 -0
  56. package/dist/app/id.js +4 -0
  57. package/dist/app/image-click-targets.d.ts +5 -0
  58. package/dist/app/image-click-targets.js +32 -0
  59. package/dist/app/image-opener.d.ts +2 -0
  60. package/dist/app/image-opener.js +64 -0
  61. package/dist/app/input-action-controller.d.ts +47 -0
  62. package/dist/app/input-action-controller.js +209 -0
  63. package/dist/app/input-controller.d.ts +60 -0
  64. package/dist/app/input-controller.js +425 -0
  65. package/dist/app/input-paste-handler.d.ts +27 -0
  66. package/dist/app/input-paste-handler.js +146 -0
  67. package/dist/app/menu-items-controller.d.ts +32 -0
  68. package/dist/app/menu-items-controller.js +182 -0
  69. package/dist/app/message-content.d.ts +8 -0
  70. package/dist/app/message-content.js +115 -0
  71. package/dist/app/model-ref.d.ts +13 -0
  72. package/dist/app/model-ref.js +50 -0
  73. package/dist/app/model-usage-controller.d.ts +35 -0
  74. package/dist/app/model-usage-controller.js +99 -0
  75. package/dist/app/model-usage-status.d.ts +125 -0
  76. package/dist/app/model-usage-status.js +749 -0
  77. package/dist/app/mouse-controller.d.ts +182 -0
  78. package/dist/app/mouse-controller.js +968 -0
  79. package/dist/app/native-modifiers.d.ts +3 -0
  80. package/dist/app/native-modifiers.js +60 -0
  81. package/dist/app/nerd-font-controller.d.ts +11 -0
  82. package/dist/app/nerd-font-controller.js +90 -0
  83. package/dist/app/popup-action-controller.d.ts +44 -0
  84. package/dist/app/popup-action-controller.js +278 -0
  85. package/dist/app/popup-menu-controller.d.ts +183 -0
  86. package/dist/app/popup-menu-controller.js +1070 -0
  87. package/dist/app/prompt-enhancer-controller.d.ts +40 -0
  88. package/dist/app/prompt-enhancer-controller.js +215 -0
  89. package/dist/app/queued-message-controller.d.ts +62 -0
  90. package/dist/app/queued-message-controller.js +377 -0
  91. package/dist/app/render-controller.d.ts +41 -0
  92. package/dist/app/render-controller.js +378 -0
  93. package/dist/app/render-text.d.ts +19 -0
  94. package/dist/app/render-text.js +67 -0
  95. package/dist/app/request-history.d.ts +23 -0
  96. package/dist/app/request-history.js +131 -0
  97. package/dist/app/runtime.d.ts +39 -0
  98. package/dist/app/runtime.js +195 -0
  99. package/dist/app/screen-selection.d.ts +6 -0
  100. package/dist/app/screen-selection.js +9 -0
  101. package/dist/app/screen-styler.d.ts +34 -0
  102. package/dist/app/screen-styler.js +168 -0
  103. package/dist/app/scroll-controller.d.ts +51 -0
  104. package/dist/app/scroll-controller.js +207 -0
  105. package/dist/app/session-event-controller.d.ts +69 -0
  106. package/dist/app/session-event-controller.js +338 -0
  107. package/dist/app/session-history.d.ts +23 -0
  108. package/dist/app/session-history.js +164 -0
  109. package/dist/app/session-lifecycle-controller.d.ts +55 -0
  110. package/dist/app/session-lifecycle-controller.js +116 -0
  111. package/dist/app/session-search.d.ts +24 -0
  112. package/dist/app/session-search.js +215 -0
  113. package/dist/app/shell-command.d.ts +27 -0
  114. package/dist/app/shell-command.js +176 -0
  115. package/dist/app/shell-controller.d.ts +28 -0
  116. package/dist/app/shell-controller.js +124 -0
  117. package/dist/app/slash-commands.d.ts +6 -0
  118. package/dist/app/slash-commands.js +75 -0
  119. package/dist/app/startup-checks.d.ts +8 -0
  120. package/dist/app/startup-checks.js +59 -0
  121. package/dist/app/startup-info.d.ts +3 -0
  122. package/dist/app/startup-info.js +176 -0
  123. package/dist/app/status-controller.d.ts +35 -0
  124. package/dist/app/status-controller.js +105 -0
  125. package/dist/app/status-line-renderer.d.ts +68 -0
  126. package/dist/app/status-line-renderer.js +453 -0
  127. package/dist/app/subagents-files.d.ts +10 -0
  128. package/dist/app/subagents-files.js +193 -0
  129. package/dist/app/subagents-model.d.ts +23 -0
  130. package/dist/app/subagents-model.js +224 -0
  131. package/dist/app/subagents-widget-controller.d.ts +43 -0
  132. package/dist/app/subagents-widget-controller.js +311 -0
  133. package/dist/app/tab-line-renderer.d.ts +26 -0
  134. package/dist/app/tab-line-renderer.js +222 -0
  135. package/dist/app/tabs-controller.d.ts +100 -0
  136. package/dist/app/tabs-controller.js +885 -0
  137. package/dist/app/terminal-controller.d.ts +40 -0
  138. package/dist/app/terminal-controller.js +135 -0
  139. package/dist/app/terminal-edit-shortcuts.d.ts +23 -0
  140. package/dist/app/terminal-edit-shortcuts.js +138 -0
  141. package/dist/app/terminal-output-buffer.d.ts +20 -0
  142. package/dist/app/terminal-output-buffer.js +52 -0
  143. package/dist/app/toast-controller.d.ts +13 -0
  144. package/dist/app/toast-controller.js +40 -0
  145. package/dist/app/toast-renderer.d.ts +9 -0
  146. package/dist/app/toast-renderer.js +79 -0
  147. package/dist/app/todo-model.d.ts +22 -0
  148. package/dist/app/todo-model.js +179 -0
  149. package/dist/app/todo-widget-controller.d.ts +21 -0
  150. package/dist/app/todo-widget-controller.js +59 -0
  151. package/dist/app/tool-block-renderer.d.ts +26 -0
  152. package/dist/app/tool-block-renderer.js +439 -0
  153. package/dist/app/types.d.ts +550 -0
  154. package/dist/app/types.js +1 -0
  155. package/dist/app/update.d.ts +36 -0
  156. package/dist/app/update.js +315 -0
  157. package/dist/app/voice-controller.d.ts +52 -0
  158. package/dist/app/voice-controller.js +600 -0
  159. package/dist/app/workspace-actions-controller.d.ts +40 -0
  160. package/dist/app/workspace-actions-controller.js +215 -0
  161. package/dist/app/workspace-undo.d.ts +44 -0
  162. package/dist/app/workspace-undo.js +215 -0
  163. package/dist/config.d.ts +62 -0
  164. package/dist/config.js +527 -0
  165. package/dist/context-progress-bar.d.ts +17 -0
  166. package/dist/context-progress-bar.js +48 -0
  167. package/dist/default-pix-config.d.ts +1 -0
  168. package/dist/default-pix-config.js +375 -0
  169. package/dist/fuzzy.d.ts +23 -0
  170. package/dist/fuzzy.js +165 -0
  171. package/dist/input-editor-files.d.ts +15 -0
  172. package/dist/input-editor-files.js +62 -0
  173. package/dist/input-editor.d.ts +186 -0
  174. package/dist/input-editor.js +835 -0
  175. package/dist/main.d.ts +1 -0
  176. package/dist/main.js +12 -0
  177. package/dist/markdown-format.d.ts +22 -0
  178. package/dist/markdown-format.js +542 -0
  179. package/dist/sdk.d.ts +3 -0
  180. package/dist/sdk.js +1 -0
  181. package/dist/syntax-highlight.d.ts +22 -0
  182. package/dist/syntax-highlight.js +528 -0
  183. package/dist/terminal-width.d.ts +6 -0
  184. package/dist/terminal-width.js +245 -0
  185. package/dist/theme.d.ts +56 -0
  186. package/dist/theme.js +118 -0
  187. package/dist/tool-renderers/apply-patch.d.ts +2 -0
  188. package/dist/tool-renderers/apply-patch.js +36 -0
  189. package/dist/tool-renderers/ast.d.ts +2 -0
  190. package/dist/tool-renderers/ast.js +5 -0
  191. package/dist/tool-renderers/compress.d.ts +2 -0
  192. package/dist/tool-renderers/compress.js +126 -0
  193. package/dist/tool-renderers/index.d.ts +3 -0
  194. package/dist/tool-renderers/index.js +44 -0
  195. package/dist/tool-renderers/question.d.ts +2 -0
  196. package/dist/tool-renderers/question.js +88 -0
  197. package/dist/tool-renderers/read.d.ts +2 -0
  198. package/dist/tool-renderers/read.js +36 -0
  199. package/dist/tool-renderers/repo.d.ts +2 -0
  200. package/dist/tool-renderers/repo.js +13 -0
  201. package/dist/tool-renderers/shell.d.ts +3 -0
  202. package/dist/tool-renderers/shell.js +27 -0
  203. package/dist/tool-renderers/skill.d.ts +2 -0
  204. package/dist/tool-renderers/skill.js +132 -0
  205. package/dist/tool-renderers/subagents.d.ts +2 -0
  206. package/dist/tool-renderers/subagents.js +51 -0
  207. package/dist/tool-renderers/todo.d.ts +2 -0
  208. package/dist/tool-renderers/todo.js +9 -0
  209. package/dist/tool-renderers/types.d.ts +42 -0
  210. package/dist/tool-renderers/types.js +1 -0
  211. package/dist/tool-renderers/utils.d.ts +31 -0
  212. package/dist/tool-renderers/utils.js +230 -0
  213. package/dist/tool-renderers/web.d.ts +3 -0
  214. package/dist/tool-renderers/web.js +9 -0
  215. package/dist/tool-renderers/write.d.ts +2 -0
  216. package/dist/tool-renderers/write.js +42 -0
  217. package/dist/ui.d.ts +56 -0
  218. package/dist/ui.js +94 -0
  219. package/docs/release.md +81 -0
  220. package/extensions/question/contract.ts +100 -0
  221. package/extensions/question/index.ts +34 -0
  222. package/extensions/question/render.ts +28 -0
  223. package/extensions/question/result.ts +86 -0
  224. package/extensions/question/tool-description.ts +11 -0
  225. package/extensions/question/tui.ts +629 -0
  226. package/extensions/question/types.ts +123 -0
  227. package/extensions/session-title/config.ts +169 -0
  228. package/extensions/session-title/index.ts +459 -0
  229. package/extensions/terminal-bell/index.ts +315 -0
  230. package/external/pi-tools-suite/README.md +242 -0
  231. package/external/pi-tools-suite/index.ts +1 -0
  232. package/external/pi-tools-suite/licenses/ollama-pi-web-search.MIT +21 -0
  233. package/external/pi-tools-suite/licenses/opencode-mystatus-MIT.txt +21 -0
  234. package/external/pi-tools-suite/package.json +53 -0
  235. package/external/pi-tools-suite/src/antigravity-auth/auth-store.ts +194 -0
  236. package/external/pi-tools-suite/src/antigravity-auth/commands.ts +80 -0
  237. package/external/pi-tools-suite/src/antigravity-auth/constants.ts +26 -0
  238. package/external/pi-tools-suite/src/antigravity-auth/headers.ts +20 -0
  239. package/external/pi-tools-suite/src/antigravity-auth/index.ts +104 -0
  240. package/external/pi-tools-suite/src/antigravity-auth/models.ts +86 -0
  241. package/external/pi-tools-suite/src/antigravity-auth/oauth.ts +305 -0
  242. package/external/pi-tools-suite/src/antigravity-auth/payload.ts +423 -0
  243. package/external/pi-tools-suite/src/antigravity-auth/status.ts +78 -0
  244. package/external/pi-tools-suite/src/antigravity-auth/stream.ts +302 -0
  245. package/external/pi-tools-suite/src/antigravity-auth/types.ts +113 -0
  246. package/external/pi-tools-suite/src/ast-grep/index.ts +6 -0
  247. package/external/pi-tools-suite/src/ast-grep/render.ts +70 -0
  248. package/external/pi-tools-suite/src/ast-grep/schema.ts +109 -0
  249. package/external/pi-tools-suite/src/ast-grep/tool.ts +345 -0
  250. package/external/pi-tools-suite/src/ast-grep/types.ts +55 -0
  251. package/external/pi-tools-suite/src/ast-grep/utils.ts +65 -0
  252. package/external/pi-tools-suite/src/async-subagents/async-subagents.sample.jsonc +222 -0
  253. package/external/pi-tools-suite/src/async-subagents/commands.ts +518 -0
  254. package/external/pi-tools-suite/src/async-subagents/constants.ts +11 -0
  255. package/external/pi-tools-suite/src/async-subagents/core/agent-strategy.ts +74 -0
  256. package/external/pi-tools-suite/src/async-subagents/core/attachment-bridge.ts +133 -0
  257. package/external/pi-tools-suite/src/async-subagents/core/cleanup.ts +66 -0
  258. package/external/pi-tools-suite/src/async-subagents/core/concurrency.ts +90 -0
  259. package/external/pi-tools-suite/src/async-subagents/core/config.ts +819 -0
  260. package/external/pi-tools-suite/src/async-subagents/core/log-limits.ts +166 -0
  261. package/external/pi-tools-suite/src/async-subagents/core/model-fallback.ts +133 -0
  262. package/external/pi-tools-suite/src/async-subagents/core/paths.ts +47 -0
  263. package/external/pi-tools-suite/src/async-subagents/core/pi-invocation.ts +35 -0
  264. package/external/pi-tools-suite/src/async-subagents/core/presets.ts +67 -0
  265. package/external/pi-tools-suite/src/async-subagents/core/process.ts +15 -0
  266. package/external/pi-tools-suite/src/async-subagents/core/prompt.ts +66 -0
  267. package/external/pi-tools-suite/src/async-subagents/core/registry.ts +224 -0
  268. package/external/pi-tools-suite/src/async-subagents/core/retry.ts +191 -0
  269. package/external/pi-tools-suite/src/async-subagents/core/routing.ts +259 -0
  270. package/external/pi-tools-suite/src/async-subagents/core/sessions.ts +138 -0
  271. package/external/pi-tools-suite/src/async-subagents/core/spawn.ts +688 -0
  272. package/external/pi-tools-suite/src/async-subagents/core/state.ts +281 -0
  273. package/external/pi-tools-suite/src/async-subagents/core/stop.ts +131 -0
  274. package/external/pi-tools-suite/src/async-subagents/core/structured-result.ts +237 -0
  275. package/external/pi-tools-suite/src/async-subagents/core/tool-guard.ts +34 -0
  276. package/external/pi-tools-suite/src/async-subagents/core/types.ts +150 -0
  277. package/external/pi-tools-suite/src/async-subagents/core/ultrawork-auto.ts +184 -0
  278. package/external/pi-tools-suite/src/async-subagents/core/utils.ts +11 -0
  279. package/external/pi-tools-suite/src/async-subagents/format.ts +41 -0
  280. package/external/pi-tools-suite/src/async-subagents/index.ts +422 -0
  281. package/external/pi-tools-suite/src/async-subagents/lib.ts +88 -0
  282. package/external/pi-tools-suite/src/async-subagents/live.ts +10 -0
  283. package/external/pi-tools-suite/src/async-subagents/polling.ts +83 -0
  284. package/external/pi-tools-suite/src/async-subagents/render.ts +230 -0
  285. package/external/pi-tools-suite/src/async-subagents/subagent-overlay.ts +77 -0
  286. package/external/pi-tools-suite/src/async-subagents/tasks.ts +120 -0
  287. package/external/pi-tools-suite/src/async-subagents/tools/cleanup.ts +99 -0
  288. package/external/pi-tools-suite/src/async-subagents/tools/result.ts +179 -0
  289. package/external/pi-tools-suite/src/async-subagents/tools/spawn.ts +372 -0
  290. package/external/pi-tools-suite/src/async-subagents/tools/status.ts +60 -0
  291. package/external/pi-tools-suite/src/async-subagents/tools/stop.ts +79 -0
  292. package/external/pi-tools-suite/src/async-subagents/tools/subagents.ts +152 -0
  293. package/external/pi-tools-suite/src/async-subagents/tools/wait.ts +67 -0
  294. package/external/pi-tools-suite/src/async-subagents/types.ts +45 -0
  295. package/external/pi-tools-suite/src/async-subagents/ui.ts +5 -0
  296. package/external/pi-tools-suite/src/compress/commands.ts +440 -0
  297. package/external/pi-tools-suite/src/compress/compress-tool.ts +368 -0
  298. package/external/pi-tools-suite/src/compress/compression-blocks.ts +524 -0
  299. package/external/pi-tools-suite/src/compress/config.ts +310 -0
  300. package/external/pi-tools-suite/src/compress/dcp-tui-filter.ts +498 -0
  301. package/external/pi-tools-suite/src/compress/index.ts +397 -0
  302. package/external/pi-tools-suite/src/compress/prompts.ts +269 -0
  303. package/external/pi-tools-suite/src/compress/pruner-candidates.ts +176 -0
  304. package/external/pi-tools-suite/src/compress/pruner-compression-blocks.ts +260 -0
  305. package/external/pi-tools-suite/src/compress/pruner-message-ids.ts +147 -0
  306. package/external/pi-tools-suite/src/compress/pruner-metadata.ts +268 -0
  307. package/external/pi-tools-suite/src/compress/pruner-nudge.ts +315 -0
  308. package/external/pi-tools-suite/src/compress/pruner-tools.ts +263 -0
  309. package/external/pi-tools-suite/src/compress/pruner-types.ts +25 -0
  310. package/external/pi-tools-suite/src/compress/pruner.ts +92 -0
  311. package/external/pi-tools-suite/src/compress/state.ts +486 -0
  312. package/external/pi-tools-suite/src/compress/ui.ts +308 -0
  313. package/external/pi-tools-suite/src/config.ts +176 -0
  314. package/external/pi-tools-suite/src/context-usage.ts +31 -0
  315. package/external/pi-tools-suite/src/default-pi-tools-suite-config.ts +355 -0
  316. package/external/pi-tools-suite/src/index.ts +46 -0
  317. package/external/pi-tools-suite/src/lib/lsp.ts +62 -0
  318. package/external/pi-tools-suite/src/lib/project.ts +42 -0
  319. package/external/pi-tools-suite/src/lib/tool-args.ts +137 -0
  320. package/external/pi-tools-suite/src/lsp/_shared/config.ts +156 -0
  321. package/external/pi-tools-suite/src/lsp/_shared/glob.ts +60 -0
  322. package/external/pi-tools-suite/src/lsp/_shared/output.ts +102 -0
  323. package/external/pi-tools-suite/src/lsp/_shared/paths.ts +138 -0
  324. package/external/pi-tools-suite/src/lsp/_shared/runner.ts +64 -0
  325. package/external/pi-tools-suite/src/lsp/_shared/template.ts +23 -0
  326. package/external/pi-tools-suite/src/lsp/_shared/trust.ts +116 -0
  327. package/external/pi-tools-suite/src/lsp/_shared/types.ts +98 -0
  328. package/external/pi-tools-suite/src/lsp/async.ts +29 -0
  329. package/external/pi-tools-suite/src/lsp/child-process.ts +81 -0
  330. package/external/pi-tools-suite/src/lsp/client.ts +340 -0
  331. package/external/pi-tools-suite/src/lsp/constants.ts +9 -0
  332. package/external/pi-tools-suite/src/lsp/diagnostics-store.ts +64 -0
  333. package/external/pi-tools-suite/src/lsp/documents.ts +24 -0
  334. package/external/pi-tools-suite/src/lsp/index.ts +31 -0
  335. package/external/pi-tools-suite/src/lsp/lsp-utils.ts +37 -0
  336. package/external/pi-tools-suite/src/lsp/manager.ts +190 -0
  337. package/external/pi-tools-suite/src/lsp/mutation-events.ts +78 -0
  338. package/external/pi-tools-suite/src/lsp/renderer.ts +1 -0
  339. package/external/pi-tools-suite/src/lsp/tool-result.ts +6 -0
  340. package/external/pi-tools-suite/src/lsp/tsserver.ts +107 -0
  341. package/external/pi-tools-suite/src/lsp/types.ts +15 -0
  342. package/external/pi-tools-suite/src/model-tools/apply-patch.ts +590 -0
  343. package/external/pi-tools-suite/src/model-tools/index.ts +430 -0
  344. package/external/pi-tools-suite/src/model-tools/path-utils.ts +6 -0
  345. package/external/pi-tools-suite/src/model-tools/tool-args.ts +11 -0
  346. package/external/pi-tools-suite/src/prompt-commands/index.ts +349 -0
  347. package/external/pi-tools-suite/src/repo-discovery/index.ts +384 -0
  348. package/external/pi-tools-suite/src/repo-discovery/project.ts +7 -0
  349. package/external/pi-tools-suite/src/startup-section.ts +13 -0
  350. package/external/pi-tools-suite/src/terminal-bell/index.ts +309 -0
  351. package/external/pi-tools-suite/src/todo/index.ts +201 -0
  352. package/external/pi-tools-suite/src/todo/state/auto-clear.ts +13 -0
  353. package/external/pi-tools-suite/src/todo/state/invariants.ts +21 -0
  354. package/external/pi-tools-suite/src/todo/state/persistence.ts +94 -0
  355. package/external/pi-tools-suite/src/todo/state/replay.ts +38 -0
  356. package/external/pi-tools-suite/src/todo/state/selectors.ts +49 -0
  357. package/external/pi-tools-suite/src/todo/state/state-reducer.ts +362 -0
  358. package/external/pi-tools-suite/src/todo/state/state.ts +18 -0
  359. package/external/pi-tools-suite/src/todo/state/store.ts +52 -0
  360. package/external/pi-tools-suite/src/todo/state/task-graph.ts +57 -0
  361. package/external/pi-tools-suite/src/todo/todo.ts +487 -0
  362. package/external/pi-tools-suite/src/todo/tool/response-envelope.ts +143 -0
  363. package/external/pi-tools-suite/src/todo/tool/types.ts +188 -0
  364. package/external/pi-tools-suite/src/todo/view/format.ts +18 -0
  365. package/external/pi-tools-suite/src/todo/view/labels.ts +13 -0
  366. package/external/pi-tools-suite/src/tool-descriptions.ts +369 -0
  367. package/external/pi-tools-suite/src/usage/index.ts +152 -0
  368. package/external/pi-tools-suite/src/usage/lib/copilot.ts +535 -0
  369. package/external/pi-tools-suite/src/usage/lib/google.ts +478 -0
  370. package/external/pi-tools-suite/src/usage/lib/openai.ts +268 -0
  371. package/external/pi-tools-suite/src/usage/lib/types.ts +114 -0
  372. package/external/pi-tools-suite/src/usage/lib/utils.ts +134 -0
  373. package/external/pi-tools-suite/src/usage/lib/zhipu.ts +228 -0
  374. package/external/pi-tools-suite/src/web-search/index.ts +397 -0
  375. package/package.json +89 -0
  376. package/skills/context7/SKILL.md +69 -0
  377. package/skills/context7/scripts/context7.sh +73 -0
  378. package/skills/handoff/SKILL.md +15 -0
  379. package/skills/pdf/LICENSE.txt +30 -0
  380. package/skills/pdf/SKILL.md +314 -0
  381. package/skills/pdf/forms.md +294 -0
  382. package/skills/pdf/reference.md +612 -0
  383. package/skills/pdf/scripts/check_bounding_boxes.py +65 -0
  384. package/skills/pdf/scripts/check_fillable_fields.py +11 -0
  385. package/skills/pdf/scripts/convert_pdf_to_images.py +33 -0
  386. package/skills/pdf/scripts/create_validation_image.py +37 -0
  387. package/skills/pdf/scripts/extract_form_field_info.py +122 -0
  388. package/skills/pdf/scripts/extract_form_structure.py +115 -0
  389. package/skills/pdf/scripts/fill_fillable_fields.py +98 -0
  390. package/skills/pdf/scripts/fill_pdf_form_with_annotations.py +107 -0
  391. package/skills/playwright-cli/SKILL.md +388 -0
  392. package/skills/playwright-cli/references/element-attributes.md +23 -0
  393. package/skills/playwright-cli/references/playwright-tests.md +39 -0
  394. package/skills/playwright-cli/references/request-mocking.md +87 -0
  395. package/skills/playwright-cli/references/running-code.md +241 -0
  396. package/skills/playwright-cli/references/session-management.md +225 -0
  397. package/skills/playwright-cli/references/spec-driven-testing.md +305 -0
  398. package/skills/playwright-cli/references/storage-state.md +275 -0
  399. package/skills/playwright-cli/references/test-generation.md +134 -0
  400. package/skills/playwright-cli/references/tracing.md +139 -0
  401. package/skills/playwright-cli/references/video-recording.md +143 -0
  402. package/skills/simplify/SKILL.md +51 -0
  403. package/skills/skill-creator/LICENSE.txt +202 -0
  404. package/skills/skill-creator/SKILL.md +485 -0
  405. package/skills/skill-creator/agents/analyzer.md +274 -0
  406. package/skills/skill-creator/agents/comparator.md +202 -0
  407. package/skills/skill-creator/agents/grader.md +223 -0
  408. package/skills/skill-creator/assets/eval_review.html +146 -0
  409. package/skills/skill-creator/eval-viewer/generate_review.py +471 -0
  410. package/skills/skill-creator/eval-viewer/viewer.html +1325 -0
  411. package/skills/skill-creator/references/schemas.md +430 -0
  412. package/skills/skill-creator/scripts/__init__.py +0 -0
  413. package/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
  414. package/skills/skill-creator/scripts/generate_report.py +326 -0
  415. package/skills/skill-creator/scripts/improve_description.py +247 -0
  416. package/skills/skill-creator/scripts/package_skill.py +136 -0
  417. package/skills/skill-creator/scripts/quick_validate.py +103 -0
  418. package/skills/skill-creator/scripts/run_eval.py +310 -0
  419. package/skills/skill-creator/scripts/run_loop.py +328 -0
  420. package/skills/skill-creator/scripts/utils.py +47 -0
@@ -0,0 +1,98 @@
1
+ import type { Diagnostic } from "vscode-languageserver-protocol";
2
+
3
+ export type ConfigKind = "lsp";
4
+
5
+ export interface CommandConfig {
6
+ bin: string;
7
+ args?: string[];
8
+ cwd?: string;
9
+ env?: Record<string, string>;
10
+ config?: string;
11
+ timeoutMs?: number;
12
+ diagnosticExitCodes?: number[];
13
+ }
14
+
15
+ export interface MatchableConfig {
16
+ id: string;
17
+ enabled?: boolean;
18
+ include?: string[];
19
+ exclude?: string[];
20
+ rootMarkers?: string[];
21
+ maxFileSizeBytes?: number;
22
+ }
23
+
24
+ export interface LspServerConfig extends MatchableConfig {
25
+ bin: string;
26
+ args?: string[];
27
+ cwd?: string;
28
+ env?: Record<string, string>;
29
+ config?: string;
30
+ languageIdByExtension?: Record<string, string>;
31
+ startupTimeoutMs?: number;
32
+ diagnosticsWaitMs?: number;
33
+ initializationOptions?: unknown;
34
+ settings?: unknown;
35
+ }
36
+
37
+ export interface LspConfigFile {
38
+ version?: number;
39
+ servers?: LspServerConfig[];
40
+ }
41
+
42
+ export interface ConfigLayer<TItem extends MatchableConfig> {
43
+ scope: "global" | "project";
44
+ path: string;
45
+ dir: string;
46
+ raw: string;
47
+ hash: string;
48
+ items: TItem[];
49
+ }
50
+
51
+ export interface LoadedConfig<TItem extends MatchableConfig> {
52
+ items: TItem[];
53
+ layers: ConfigLayer<TItem>[];
54
+ warnings: string[];
55
+ }
56
+
57
+ export interface PathPlaceholders {
58
+ workspace: string;
59
+ root: string;
60
+ file: string;
61
+ relFile: string;
62
+ dir: string;
63
+ relDir: string;
64
+ config: string;
65
+ configDir: string;
66
+ }
67
+
68
+ export interface ResolvedCommand {
69
+ id: string;
70
+ bin: string;
71
+ args: string[];
72
+ cwd: string;
73
+ env?: Record<string, string>;
74
+ timeoutMs?: number;
75
+ configPath?: string;
76
+ placeholders: PathPlaceholders;
77
+ }
78
+
79
+ export interface CommandRunResult {
80
+ id: string;
81
+ bin: string;
82
+ args: string[];
83
+ cwd: string;
84
+ stdout: string;
85
+ stderr: string;
86
+ code: number;
87
+ killed: boolean;
88
+ durationMs: number;
89
+ }
90
+
91
+ export interface StoredDiagnostics {
92
+ serverId: string;
93
+ root: string;
94
+ file: string;
95
+ version?: number;
96
+ diagnostics: Diagnostic[];
97
+ updatedAt: number;
98
+ }
@@ -0,0 +1,29 @@
1
+ export function delay(ms: number, signal?: AbortSignal): Promise<void> {
2
+ if (ms <= 0) return Promise.resolve();
3
+ if (signal?.aborted) return Promise.reject(new Error("aborted"));
4
+ return new Promise((resolve, reject) => {
5
+ const cleanup = () => {
6
+ clearTimeout(timeout);
7
+ signal?.removeEventListener("abort", abort);
8
+ };
9
+ const timeout = setTimeout(() => {
10
+ cleanup();
11
+ resolve();
12
+ }, ms);
13
+ const abort = () => {
14
+ cleanup();
15
+ reject(new Error("aborted"));
16
+ };
17
+ signal?.addEventListener("abort", abort, { once: true });
18
+ });
19
+ }
20
+
21
+ export async function withTimeout<T>(promise: Promise<T>, timeoutMs: number, label: string): Promise<T> {
22
+ let timeout: NodeJS.Timeout | undefined;
23
+ const timeoutPromise = new Promise<never>((_resolve, reject) => {
24
+ timeout = setTimeout(() => reject(new Error(`${label} timed out after ${timeoutMs}ms`)), timeoutMs);
25
+ });
26
+ return Promise.race([promise, timeoutPromise]).finally(() => {
27
+ if (timeout) clearTimeout(timeout);
28
+ });
29
+ }
@@ -0,0 +1,81 @@
1
+ import { spawnSync, type ChildProcessWithoutNullStreams } from "node:child_process";
2
+ import { SHUTDOWN_KILL_TIMEOUT_MS, SHUTDOWN_TERM_TIMEOUT_MS, SHUTDOWN_WRITE_TIMEOUT_MS } from "./constants";
3
+
4
+ export function isChildRunning(child: ChildProcessWithoutNullStreams): boolean {
5
+ return child.exitCode === null && child.signalCode === null;
6
+ }
7
+
8
+ function canWriteToChild(child: ChildProcessWithoutNullStreams): boolean {
9
+ return isChildRunning(child) && child.stdin.writable && !child.stdin.destroyed && !child.stdin.writableEnded;
10
+ }
11
+
12
+ export function killChild(child: ChildProcessWithoutNullStreams, signal: NodeJS.Signals): boolean {
13
+ try {
14
+ if (process.platform === "win32" && child.pid) {
15
+ const result = spawnSync("taskkill", ["/pid", String(child.pid), "/T", "/F"], {
16
+ stdio: "ignore",
17
+ timeout: 1_000,
18
+ windowsHide: true,
19
+ });
20
+ if (!result.error && result.status === 0) return true;
21
+ }
22
+ return child.kill(signal);
23
+ } catch {
24
+ return false;
25
+ }
26
+ }
27
+
28
+ function waitForChildExit(child: ChildProcessWithoutNullStreams, timeoutMs: number): Promise<boolean> {
29
+ if (!isChildRunning(child)) return Promise.resolve(true);
30
+
31
+ return new Promise((resolve) => {
32
+ let done = false;
33
+ const finish = (exited: boolean) => {
34
+ if (done) return;
35
+ done = true;
36
+ clearTimeout(timeout);
37
+ child.off("exit", onExit);
38
+ resolve(exited);
39
+ };
40
+ const onExit = () => finish(true);
41
+ const timeout = setTimeout(() => finish(false), timeoutMs);
42
+ timeout.unref();
43
+ child.once("exit", onExit);
44
+ if (!isChildRunning(child)) finish(true);
45
+ });
46
+ }
47
+
48
+ export async function terminateChild(child: ChildProcessWithoutNullStreams): Promise<void> {
49
+ if (!isChildRunning(child)) return;
50
+
51
+ if (!killChild(child, "SIGTERM")) return;
52
+ const exited = await waitForChildExit(child, SHUTDOWN_TERM_TIMEOUT_MS);
53
+ if (exited || !isChildRunning(child)) return;
54
+
55
+ killChild(child, "SIGKILL");
56
+ await waitForChildExit(child, SHUTDOWN_KILL_TIMEOUT_MS);
57
+ }
58
+
59
+ export async function bestEffortWriteJsonRpc(child: ChildProcessWithoutNullStreams, message: Record<string, unknown>): Promise<void> {
60
+ if (!canWriteToChild(child)) return;
61
+
62
+ const json = JSON.stringify(message);
63
+ const payload = `Content-Length: ${Buffer.byteLength(json, "utf8")}\r\n\r\n${json}`;
64
+ await new Promise<void>((resolve) => {
65
+ let done = false;
66
+ const finish = () => {
67
+ if (done) return;
68
+ done = true;
69
+ clearTimeout(timeout);
70
+ resolve();
71
+ };
72
+ const timeout = setTimeout(finish, SHUTDOWN_WRITE_TIMEOUT_MS);
73
+ timeout.unref();
74
+
75
+ try {
76
+ child.stdin.write(payload, "utf8", finish);
77
+ } catch {
78
+ finish();
79
+ }
80
+ });
81
+ }
@@ -0,0 +1,340 @@
1
+ import path from "node:path";
2
+ import { spawn, type ChildProcessWithoutNullStreams } from "node:child_process";
3
+ import type { MessageConnection } from "vscode-jsonrpc";
4
+ import { createMessageConnection, StreamMessageReader, StreamMessageWriter } from "vscode-jsonrpc/node";
5
+ import {
6
+ DefinitionRequest,
7
+ DidChangeConfigurationNotification,
8
+ DidChangeTextDocumentNotification,
9
+ DidOpenTextDocumentNotification,
10
+ DidSaveTextDocumentNotification,
11
+ DocumentSymbolRequest,
12
+ ExecuteCommandRequest,
13
+ HoverRequest,
14
+ InitializeRequest,
15
+ InitializedNotification,
16
+ PublishDiagnosticsNotification,
17
+ ReferencesRequest,
18
+ type Diagnostic,
19
+ type InitializeResult,
20
+ type ServerCapabilities,
21
+ } from "vscode-languageserver-protocol";
22
+ import { withTimeout } from "./async";
23
+ import { bestEffortWriteJsonRpc, isChildRunning, killChild, terminateChild } from "./child-process";
24
+ import { DEFAULT_STARTUP_TIMEOUT_MS, REQUEST_TIMEOUT_MS } from "./constants";
25
+ import { DocumentStore } from "./documents";
26
+ import type { DiagnosticsStore } from "./diagnostics-store";
27
+ import { filePathToUri } from "./_shared/paths";
28
+ import { isExecutableAvailable } from "./_shared/runner";
29
+ import type { LspServerConfig, ResolvedCommand } from "./_shared/types";
30
+ import { supportsSave } from "./lsp-utils";
31
+ import type { OpenDocument } from "./types";
32
+ import { tsserverDiagnosticToLsp, tsserverDiagnosticsFromResponse } from "./tsserver";
33
+
34
+ export class LspClient {
35
+ private process: ChildProcessWithoutNullStreams | undefined;
36
+ private connection: MessageConnection | undefined;
37
+ private capabilities: ServerCapabilities | undefined;
38
+ private readonly documents = new DocumentStore();
39
+ private startPromise: Promise<void> | undefined;
40
+ private initialized = false;
41
+ private unavailableReason: string | undefined;
42
+ private stderrTail = "";
43
+
44
+ constructor(
45
+ private readonly server: LspServerConfig,
46
+ private readonly root: string,
47
+ private readonly command: ResolvedCommand,
48
+ private readonly diagnostics: DiagnosticsStore,
49
+ ) {}
50
+
51
+ get isUnavailable(): boolean {
52
+ return !!this.unavailableReason;
53
+ }
54
+
55
+ get reason(): string | undefined {
56
+ return this.unavailableReason;
57
+ }
58
+
59
+ async ensureStarted(signal?: AbortSignal): Promise<void> {
60
+ if (this.initialized && this.connection && !this.unavailableReason) return;
61
+ if (this.unavailableReason) throw new Error(this.unavailableReason);
62
+ this.startPromise ??= this.start(signal).catch(async (error) => {
63
+ this.startPromise = undefined;
64
+ await this.shutdown();
65
+ throw error;
66
+ });
67
+ await this.startPromise;
68
+ }
69
+
70
+ private async start(signal?: AbortSignal): Promise<void> {
71
+ if (!isExecutableAvailable(this.command.bin)) {
72
+ this.unavailableReason = `${this.server.id}: LSP binary not found: ${this.command.bin}`;
73
+ throw new Error(this.unavailableReason);
74
+ }
75
+
76
+ const child = spawn(this.command.bin, this.command.args, {
77
+ cwd: this.command.cwd,
78
+ env: this.command.env ? { ...process.env, ...this.command.env } : process.env,
79
+ shell: false,
80
+ stdio: ["pipe", "pipe", "pipe"],
81
+ });
82
+
83
+ this.process = child;
84
+ let failStartup: ((error: Error) => void) | undefined;
85
+ const startupFailure = new Promise<never>((_resolve, reject) => {
86
+ failStartup = reject;
87
+ });
88
+ const markUnavailable = (reason: string) => {
89
+ this.unavailableReason = reason;
90
+ this.initialized = false;
91
+ this.startPromise = undefined;
92
+ this.connection?.dispose();
93
+ this.connection = undefined;
94
+ failStartup?.(new Error(reason));
95
+ };
96
+
97
+ child.stderr.on("data", (chunk: Buffer) => {
98
+ this.stderrTail = `${this.stderrTail}${chunk.toString()}`.slice(-4000);
99
+ });
100
+ child.on("error", (error) => {
101
+ markUnavailable(`${this.server.id}: LSP failed to start: ${error.message}`);
102
+ });
103
+ child.on("exit", (code, sig) => {
104
+ markUnavailable(`${this.server.id}: LSP exited (${code ?? sig ?? "unknown"})${this.stderrTail ? `: ${this.stderrTail.trim()}` : ""}`);
105
+ });
106
+
107
+ const connection = createMessageConnection(new StreamMessageReader(child.stdout), new StreamMessageWriter(child.stdin));
108
+ this.connection = connection;
109
+ this.registerHandlers(connection);
110
+ connection.listen();
111
+
112
+ const initializeResult = (await withTimeout(
113
+ Promise.race([connection.sendRequest(InitializeRequest.method, this.initializeParams()), startupFailure]),
114
+ this.server.startupTimeoutMs ?? DEFAULT_STARTUP_TIMEOUT_MS,
115
+ `${this.server.id} initialize`,
116
+ )) as InitializeResult;
117
+ this.capabilities = initializeResult.capabilities;
118
+ failStartup = undefined;
119
+
120
+ await connection.sendNotification(InitializedNotification.method, {});
121
+ if (this.server.settings !== undefined) {
122
+ await connection.sendNotification(DidChangeConfigurationNotification.method, { settings: this.server.settings });
123
+ }
124
+
125
+ if (signal?.aborted) throw new Error("aborted");
126
+ this.initialized = true;
127
+ }
128
+
129
+ private initializeParams() {
130
+ const rootUri = filePathToUri(this.root);
131
+ return {
132
+ processId: process.pid,
133
+ rootUri,
134
+ workspaceFolders: [{ name: path.basename(this.root), uri: rootUri }],
135
+ capabilities: {
136
+ window: { workDoneProgress: true },
137
+ workspace: {
138
+ configuration: true,
139
+ workspaceFolders: true,
140
+ didChangeWatchedFiles: { dynamicRegistration: true },
141
+ },
142
+ textDocument: {
143
+ synchronization: {
144
+ didOpen: true,
145
+ didChange: true,
146
+ didSave: true,
147
+ },
148
+ publishDiagnostics: {
149
+ relatedInformation: true,
150
+ versionSupport: true,
151
+ },
152
+ hover: {},
153
+ definition: {},
154
+ references: {},
155
+ documentSymbol: {},
156
+ },
157
+ },
158
+ initializationOptions: this.server.initializationOptions ?? {},
159
+ };
160
+ }
161
+
162
+ private registerHandlers(connection: MessageConnection): void {
163
+ connection.onNotification(
164
+ PublishDiagnosticsNotification.method,
165
+ (params: { uri: string; diagnostics: Diagnostic[]; version?: number }) => {
166
+ this.diagnostics.set(this.server.id, this.root, params.uri, params.diagnostics, params.version);
167
+ },
168
+ );
169
+
170
+ const anyConnection = connection as unknown as {
171
+ onRequest(method: string, handler: (params: unknown) => unknown): void;
172
+ onNotification(method: string, handler: (params: unknown) => void): void;
173
+ };
174
+
175
+ anyConnection.onRequest("workspace/configuration", (params: unknown) => {
176
+ const items = (params as { items?: unknown[] } | undefined)?.items;
177
+ if (!Array.isArray(items)) return [this.server.settings ?? {}];
178
+ return items.map(() => this.server.settings ?? {});
179
+ });
180
+ anyConnection.onRequest("workspace/workspaceFolders", () => [{ name: path.basename(this.root), uri: filePathToUri(this.root) }]);
181
+ anyConnection.onRequest("client/registerCapability", () => null);
182
+ anyConnection.onRequest("client/unregisterCapability", () => null);
183
+ anyConnection.onRequest("window/workDoneProgress/create", () => null);
184
+ anyConnection.onNotification("window/logMessage", () => undefined);
185
+ anyConnection.onNotification("telemetry/event", () => undefined);
186
+ }
187
+
188
+ async openOrChange(file: string, languageId: string, text: string, signal?: AbortSignal): Promise<OpenDocument> {
189
+ await this.ensureStarted(signal);
190
+ if (!this.connection) throw new Error(`${this.server.id}: LSP connection unavailable`);
191
+
192
+ const existing = this.documents.get(file);
193
+ if (!existing) {
194
+ const doc = this.documents.open(file, languageId, text);
195
+ await this.connection.sendNotification(DidOpenTextDocumentNotification.method, {
196
+ textDocument: {
197
+ uri: doc.uri,
198
+ languageId: doc.languageId,
199
+ version: doc.version,
200
+ text: doc.text,
201
+ },
202
+ });
203
+ return doc;
204
+ }
205
+
206
+ const doc = this.documents.change(file, text);
207
+ await this.connection.sendNotification(DidChangeTextDocumentNotification.method, {
208
+ textDocument: { uri: doc.uri, version: doc.version },
209
+ contentChanges: [{ text: doc.text }],
210
+ });
211
+ return doc;
212
+ }
213
+
214
+ async didSave(file: string): Promise<void> {
215
+ if (!this.connection || !supportsSave(this.capabilities)) return;
216
+ const doc = this.documents.get(file);
217
+ if (!doc) return;
218
+ await this.connection.sendNotification(DidSaveTextDocumentNotification.method, {
219
+ textDocument: { uri: doc.uri },
220
+ text: doc.text,
221
+ });
222
+ }
223
+
224
+ private supportsTsserverDiagnostics(): boolean {
225
+ const commands = this.capabilities?.executeCommandProvider?.commands;
226
+ return Array.isArray(commands) && commands.includes("typescript.tsserverRequest");
227
+ }
228
+
229
+ async tsserverDiagnostics(file: string, text: string, timeoutMs: number): Promise<Diagnostic[] | undefined> {
230
+ const connection = this.connection;
231
+ if (!connection || !this.supportsTsserverDiagnostics()) return undefined;
232
+
233
+ const requests = [
234
+ { command: "syntacticDiagnosticsSync", executionTarget: 1 },
235
+ { command: "semanticDiagnosticsSync", executionTarget: 0 },
236
+ { command: "suggestionDiagnosticsSync", executionTarget: 0 },
237
+ ];
238
+
239
+ const responses = await Promise.all(requests.map((request) => withTimeout(
240
+ connection.sendRequest(ExecuteCommandRequest.method, {
241
+ command: "typescript.tsserverRequest",
242
+ arguments: [
243
+ request.command,
244
+ { file, includeLinePosition: true },
245
+ {
246
+ executionTarget: request.executionTarget,
247
+ expectsResult: true,
248
+ isAsync: false,
249
+ lowPriority: false,
250
+ },
251
+ ],
252
+ }),
253
+ timeoutMs,
254
+ `${this.server.id} ${request.command}`,
255
+ )));
256
+
257
+ return responses.flatMap((response) => tsserverDiagnosticsFromResponse(response).map((diagnostic) => tsserverDiagnosticToLsp(diagnostic, text)));
258
+ }
259
+
260
+ async hover(file: string, line: number, character: number): Promise<unknown> {
261
+ if (!this.connection) throw new Error(`${this.server.id}: LSP connection unavailable`);
262
+ return withTimeout(
263
+ this.connection.sendRequest(HoverRequest.method, {
264
+ textDocument: { uri: filePathToUri(file) },
265
+ position: { line, character },
266
+ }),
267
+ REQUEST_TIMEOUT_MS,
268
+ `${this.server.id} hover`,
269
+ );
270
+ }
271
+
272
+ async definition(file: string, line: number, character: number): Promise<unknown> {
273
+ if (!this.connection) throw new Error(`${this.server.id}: LSP connection unavailable`);
274
+ return withTimeout(
275
+ this.connection.sendRequest(DefinitionRequest.method, {
276
+ textDocument: { uri: filePathToUri(file) },
277
+ position: { line, character },
278
+ }),
279
+ REQUEST_TIMEOUT_MS,
280
+ `${this.server.id} definition`,
281
+ );
282
+ }
283
+
284
+ async references(file: string, line: number, character: number, includeDeclaration: boolean): Promise<unknown> {
285
+ if (!this.connection) throw new Error(`${this.server.id}: LSP connection unavailable`);
286
+ return withTimeout(
287
+ this.connection.sendRequest(ReferencesRequest.method, {
288
+ textDocument: { uri: filePathToUri(file) },
289
+ position: { line, character },
290
+ context: { includeDeclaration },
291
+ }),
292
+ REQUEST_TIMEOUT_MS,
293
+ `${this.server.id} references`,
294
+ );
295
+ }
296
+
297
+ async symbols(file: string): Promise<unknown> {
298
+ if (!this.connection) throw new Error(`${this.server.id}: LSP connection unavailable`);
299
+ return withTimeout(
300
+ this.connection.sendRequest(DocumentSymbolRequest.method, {
301
+ textDocument: { uri: filePathToUri(file) },
302
+ }),
303
+ REQUEST_TIMEOUT_MS,
304
+ `${this.server.id} document symbols`,
305
+ );
306
+ }
307
+
308
+ async shutdown(): Promise<void> {
309
+ const connection = this.connection;
310
+ const child = this.process;
311
+ this.connection = undefined;
312
+ this.process = undefined;
313
+ this.initialized = false;
314
+ this.startPromise = undefined;
315
+
316
+ if (child) {
317
+ await bestEffortWriteJsonRpc(child, { jsonrpc: "2.0", id: "pi-lsp-shutdown", method: "shutdown" });
318
+ await bestEffortWriteJsonRpc(child, { jsonrpc: "2.0", method: "exit" });
319
+ }
320
+
321
+ connection?.dispose();
322
+
323
+ if (child) {
324
+ await terminateChild(child);
325
+ }
326
+ }
327
+
328
+ shutdownSync(): void {
329
+ const connection = this.connection;
330
+ const child = this.process;
331
+ this.connection = undefined;
332
+ this.process = undefined;
333
+ this.initialized = false;
334
+ this.startPromise = undefined;
335
+
336
+ connection?.dispose();
337
+
338
+ if (child && isChildRunning(child)) killChild(child, "SIGKILL");
339
+ }
340
+ }
@@ -0,0 +1,9 @@
1
+ export const DEFAULT_STARTUP_TIMEOUT_MS = 45_000;
2
+ export const DEFAULT_DIAGNOSTICS_WAIT_MS = 10_000;
3
+ export const DEFAULT_MAX_FILE_SIZE_BYTES = 2 * 1024 * 1024;
4
+ export const REQUEST_TIMEOUT_MS = 30_000;
5
+ export const SHUTDOWN_WRITE_TIMEOUT_MS = 100;
6
+ export const SHUTDOWN_TERM_TIMEOUT_MS = 2_000;
7
+ export const SHUTDOWN_KILL_TIMEOUT_MS = 500;
8
+
9
+ export const LSP_MANAGER_GLOBAL_KEY = "__piToolsSuiteLspManager";
@@ -0,0 +1,64 @@
1
+ import type { Diagnostic } from "vscode-languageserver-protocol";
2
+ import { delay } from "./async";
3
+ import { uriToFilePath } from "./_shared/paths";
4
+ import type { StoredDiagnostics } from "./_shared/types";
5
+
6
+ export class DiagnosticsStore {
7
+ private readonly diagnostics = new Map<string, StoredDiagnostics>();
8
+
9
+ private key(serverId: string, root: string, file: string): string {
10
+ return `${serverId}\u0000${root}\u0000${file}`;
11
+ }
12
+
13
+ set(serverId: string, root: string, uri: string, diagnostics: Diagnostic[], version?: number): void {
14
+ const file = uriToFilePath(uri);
15
+ const key = this.key(serverId, root, file);
16
+ this.diagnostics.set(key, {
17
+ serverId,
18
+ root,
19
+ file,
20
+ version,
21
+ diagnostics,
22
+ updatedAt: Date.now(),
23
+ });
24
+ }
25
+
26
+ clear(serverId: string, root: string, uri: string): void {
27
+ this.diagnostics.delete(this.key(serverId, root, uriToFilePath(uri)));
28
+ }
29
+
30
+ get(serverId: string, root: string, file: string): StoredDiagnostics | undefined {
31
+ return this.diagnostics.get(this.key(serverId, root, file));
32
+ }
33
+
34
+ async waitForFile(
35
+ serverId: string,
36
+ root: string,
37
+ file: string,
38
+ since: number,
39
+ version: number | undefined,
40
+ timeoutMs: number,
41
+ signal?: AbortSignal,
42
+ ): Promise<StoredDiagnostics | undefined> {
43
+ const deadline = Date.now() + Math.max(0, timeoutMs);
44
+
45
+ while (true) {
46
+ const entry = this.get(serverId, root, file);
47
+ if (entry && entry.updatedAt >= since) {
48
+ // Some language servers omit diagnostic versions. If present, require
49
+ // diagnostics for the document version we just sent or newer.
50
+ if (entry.version === undefined || version === undefined || entry.version >= version) {
51
+ return entry;
52
+ }
53
+ }
54
+
55
+ const remaining = deadline - Date.now();
56
+ if (remaining <= 0) return this.get(serverId, root, file);
57
+ await delay(Math.min(50, remaining), signal);
58
+ }
59
+ }
60
+
61
+ getAllForFile(file: string): StoredDiagnostics[] {
62
+ return [...this.diagnostics.values()].filter((entry) => entry.file === file);
63
+ }
64
+ }
@@ -0,0 +1,24 @@
1
+ import { filePathToUri } from "./_shared/paths";
2
+ import type { OpenDocument } from "./types";
3
+
4
+ export class DocumentStore {
5
+ private readonly documents = new Map<string, OpenDocument>();
6
+
7
+ get(file: string): OpenDocument | undefined {
8
+ return this.documents.get(file);
9
+ }
10
+
11
+ open(file: string, languageId: string, text: string): OpenDocument {
12
+ const doc: OpenDocument = { file, uri: filePathToUri(file), languageId, version: 1, text };
13
+ this.documents.set(file, doc);
14
+ return doc;
15
+ }
16
+
17
+ change(file: string, text: string): OpenDocument {
18
+ const existing = this.documents.get(file);
19
+ if (!existing) throw new Error(`document not opened: ${file}`);
20
+ const updated: OpenDocument = { ...existing, version: existing.version + 1, text };
21
+ this.documents.set(file, updated);
22
+ return updated;
23
+ }
24
+ }
@@ -0,0 +1,31 @@
1
+ import type { ExtensionAPI } from "@earendil-works/pi-coding-agent";
2
+ import { appendLspDiagnosticsToMutationResult, shutdownGlobalLspManager } from "./tool-result.js";
3
+
4
+ // renderer.ts is intentionally empty — TUI rendering removed.
5
+ export { LspManager, getGlobalLspManager } from "./manager";
6
+ export { getEventPaths, isMutationToolResult } from "./mutation-events";
7
+ export {
8
+ appendLspDiagnosticsToMutationResult,
9
+ shutdownGlobalLspManager,
10
+ type LspEnrichableToolResult,
11
+ type LspMutationToolResultInput,
12
+ } from "./tool-result";
13
+
14
+ export default function piLspExtension(pi: ExtensionAPI) {
15
+ pi.on("tool_result", async (event, ctx) => {
16
+ const result = await appendLspDiagnosticsToMutationResult({
17
+ toolName: event.toolName,
18
+ input: event.input,
19
+ result: { content: event.content, details: event.details },
20
+ ctx,
21
+ isError: event.isError,
22
+ });
23
+
24
+ if (result.content === event.content && result.details === event.details) return undefined;
25
+ return { content: result.content, details: result.details };
26
+ });
27
+
28
+ pi.on("session_shutdown", async () => {
29
+ await shutdownGlobalLspManager();
30
+ });
31
+ }