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,230 @@
1
+ import { truncateToWidth, visibleWidth } from "@mariozechner/pi-tui";
2
+ import type { AgentState } from "./lib.js";
3
+ import { modelName, plural, statusGlyph, statusLabel } from "./format.js";
4
+ import type { AgentTaskPreview, SubagentRunRenderDetails } from "./types.js";
5
+
6
+ interface Component {
7
+ invalidate(): void;
8
+ render(width: number): string[];
9
+ }
10
+
11
+ function statusColor(status: AgentState["status"]): string {
12
+ if (status === "done") return "success";
13
+ if (status === "failed" || status === "stopped") return "error";
14
+ if (status === "running") return "warning";
15
+ return "dim";
16
+ }
17
+
18
+ function statusVerb(status: AgentState["status"]): string {
19
+ if (status === "done") return "Completed";
20
+ if (status === "failed") return "Failed";
21
+ if (status === "stopped") return "Stopped";
22
+ if (status === "running") return "Started";
23
+ return "Planned";
24
+ }
25
+
26
+ function statusSummary(agents: AgentState[], theme: any): string {
27
+ const running = agents.filter((agent) => agent.status === "running").length;
28
+ const done = agents.filter((agent) => agent.status === "done").length;
29
+ const failed = agents.filter((agent) => agent.status === "failed" || agent.status === "stopped").length;
30
+ const planned = agents.filter((agent) => agent.status === "planned").length;
31
+ const summary: string[] = [];
32
+ if (running) summary.push(theme.fg("warning", `${running} running`));
33
+ if (done) summary.push(theme.fg("success", `${done} done`));
34
+ if (failed) summary.push(theme.fg("error", `${failed} failed`));
35
+ if (planned) summary.push(theme.fg("dim", `${planned} planned`));
36
+ return summary.length ? theme.fg("dim", " · ") + summary.join(theme.fg("dim", ", ")) : "";
37
+ }
38
+
39
+ function appendDetailLine(text: string, prefix: string, line: string, theme: any): string {
40
+ return text + `\n${theme.fg("dim", prefix + line)}`;
41
+ }
42
+
43
+ function outputLineSummary(agent: AgentState): string | undefined {
44
+ const outputLines = (agent.resultLines ?? 0) + (agent.stderrLines ?? 0);
45
+ if (outputLines > 0) return `${plural(outputLines, "output line")}`;
46
+ if (agent.eventLines !== undefined) return `${plural(agent.eventLines, "event line")}`;
47
+ return undefined;
48
+ }
49
+
50
+ function elapsedSummary(agent: AgentState): string | undefined {
51
+ if (!agent.startedAt) return undefined;
52
+ const start = Date.parse(agent.startedAt);
53
+ if (!Number.isFinite(start)) return undefined;
54
+ const end = agent.finishedAt ? Date.parse(agent.finishedAt) : Date.now();
55
+ if (!Number.isFinite(end) || end < start) return undefined;
56
+ const totalSeconds = Math.floor((end - start) / 1000);
57
+ const minutes = Math.floor(totalSeconds / 60);
58
+ const seconds = totalSeconds % 60;
59
+ return minutes > 0 ? `${minutes}m ${seconds}s` : `${seconds}s`;
60
+ }
61
+
62
+ function commandVerbForMode(mode: SubagentRunRenderDetails["mode"]): string {
63
+ if (mode === "spawn") return "Started";
64
+ if (mode === "status") return "Checked";
65
+ if (mode === "completion") return "Completed";
66
+ return "Ran";
67
+ }
68
+
69
+ const SPAWN_PROMPT_COMPACT_TOTAL_LINE_LIMIT = 6;
70
+
71
+ function findTaskPreview(details: SubagentRunRenderDetails, agentId: string): AgentTaskPreview | undefined {
72
+ return details.tasks?.find((task) => task.id === agentId);
73
+ }
74
+
75
+ class SubagentRunComponent implements Component {
76
+ constructor(
77
+ private details: SubagentRunRenderDetails,
78
+ private options: { expanded?: boolean; isPartial?: boolean },
79
+ private theme: any,
80
+ ) {}
81
+
82
+ invalidate(): void {
83
+ // Stateless; rebuilt on each render width.
84
+ }
85
+
86
+ render(width: number): string[] {
87
+ return renderSubagentRunText(this.details, this.options, this.theme, width)
88
+ .split("\n")
89
+ .map((line) => truncateToWidth(line, width));
90
+ }
91
+ }
92
+
93
+ class SubagentSpawnPromptComponent implements Component {
94
+ constructor(
95
+ private details: SubagentRunRenderDetails,
96
+ private options: { expanded?: boolean; isPartial?: boolean },
97
+ private theme: any,
98
+ ) {}
99
+
100
+ invalidate(): void {
101
+ // Stateless; rebuilt on each render width.
102
+ }
103
+
104
+ render(width: number): string[] {
105
+ return renderSubagentSpawnPromptsText(this.details, this.options, this.theme, width)
106
+ .split("\n")
107
+ .map((line) => truncateToWidth(line, width));
108
+ }
109
+ }
110
+
111
+ function renderSubagentRunText(details: SubagentRunRenderDetails, options: { expanded?: boolean; isPartial?: boolean }, theme: any, width: number): string {
112
+ const agents = details.agents ?? [];
113
+ const verb = commandVerbForMode(details.mode);
114
+ let text = theme.fg("toolTitle", theme.bold(`${verb} ${plural(agents.length, "subagent")}, tracked 1 run`));
115
+ text += statusSummary(agents, theme);
116
+
117
+ const visibleAgents = options.expanded ? agents : agents.slice(0, 6);
118
+ for (const [index, agent] of visibleAgents.entries()) {
119
+ const isLastVisible = index === visibleAgents.length - 1 && (options.expanded || visibleAgents.length === agents.length);
120
+ const branch = isLastVisible ? "└" : "├";
121
+ const detailPrefix = isLastVisible ? " " : "│ ";
122
+ const preview = findTaskPreview(details, agent.id);
123
+ const outputLines = outputLineSummary(agent);
124
+ const elapsed = elapsedSummary(agent);
125
+ const model = preview?.model ? modelName(preview.model) : undefined;
126
+ const showElapsed = Boolean(elapsed && (options.expanded || agent.status === "running"));
127
+ const prefixParts: string[] = [];
128
+ prefixParts.push(theme.fg(statusColor(agent.status), statusGlyph(agent.status)));
129
+ prefixParts.push(theme.fg("accent", agent.id));
130
+ if (model) prefixParts.push(theme.fg("success", model));
131
+ const suffixParts: string[] = [];
132
+ if (outputLines) suffixParts.push(theme.fg("dim", `· ${outputLines}`));
133
+ if (showElapsed) suffixParts.push(theme.fg("dim", `· ${elapsed}`));
134
+
135
+ const parts: string[] = [...prefixParts];
136
+ if (preview?.task) {
137
+ const prefix = `${theme.fg("dim", branch)} ${prefixParts.join(" ")} `;
138
+ const suffix = suffixParts.length ? ` ${suffixParts.join(" ")}` : "";
139
+ const taskWidth = Math.max(1, width - visibleWidth(prefix) - visibleWidth(suffix));
140
+ const task = truncateToWidth(preview.task, taskWidth);
141
+ parts.push(theme.fg("dim", task));
142
+ }
143
+ parts.push(...suffixParts);
144
+ text += `\n${theme.fg("dim", branch)} ${parts.join(" ")}`;
145
+
146
+ if (options.expanded) {
147
+ if (agent.pid) text = appendDetailLine(text, detailPrefix, `pid ${agent.pid}`, theme);
148
+ if (agent.exitCode !== undefined) text = appendDetailLine(text, detailPrefix, `exit ${agent.exitCode}`, theme);
149
+ if (preview?.scope) text = appendDetailLine(text, detailPrefix, `scope ${preview.scope}`, theme);
150
+ if (agent.startedAt) text = appendDetailLine(text, detailPrefix, `started ${agent.startedAt}`, theme);
151
+ if (agent.finishedAt) text = appendDetailLine(text, detailPrefix, `finished ${agent.finishedAt}`, theme);
152
+ }
153
+ }
154
+
155
+ if (!options.expanded && agents.length > visibleAgents.length) {
156
+ text += `\n${theme.fg("dim", "└")} ${theme.fg("muted", `… +${agents.length - visibleAgents.length} more`)}`;
157
+ }
158
+
159
+ if (options.expanded) {
160
+ text += `\n${theme.fg("dim", `run ${details.runDir}`)}`;
161
+ } else {
162
+ text += `\n${theme.fg("dim", "ctrl+o to expand")}`;
163
+ }
164
+
165
+ return text;
166
+ }
167
+
168
+ export function renderSubagentRun(details: SubagentRunRenderDetails, options: { expanded?: boolean; isPartial?: boolean }, theme: any): Component {
169
+ return new SubagentRunComponent(details, options, theme);
170
+ }
171
+
172
+ function spawnPromptCount(details: SubagentRunRenderDetails): number {
173
+ return details.tasks?.length ?? details.agents?.length ?? 0;
174
+ }
175
+
176
+ function renderSubagentSpawnPromptsText(details: SubagentRunRenderDetails, options: { expanded?: boolean; isPartial?: boolean }, theme: any, width: number): string {
177
+ const tasks = details.tasks ?? [];
178
+ const count = spawnPromptCount(details);
179
+ let text = theme.fg("toolTitle", theme.bold(renderSpawnChatSummary(details)));
180
+
181
+ const compactPromptLineLimit = Math.max(0, SPAWN_PROMPT_COMPACT_TOTAL_LINE_LIMIT - 1);
182
+ const promptLineLimit = options.expanded ? tasks.length : Math.min(tasks.length, compactPromptLineLimit);
183
+ const visibleTasks = tasks.slice(0, promptLineLimit);
184
+ const hiddenTasks = tasks.length - visibleTasks.length;
185
+ const lines: string[] = [];
186
+ for (const [index, task] of visibleTasks.entries()) {
187
+ const isOverflowSummary = !options.expanded && hiddenTasks > 0 && index === compactPromptLineLimit - 1;
188
+ const isLastVisible = index === visibleTasks.length - 1;
189
+ const branch = isLastVisible ? "└" : "├";
190
+ if (isOverflowSummary) {
191
+ lines.push(`${theme.fg("dim", branch)} ${theme.fg("muted", `… +${hiddenTasks + 1} more prompts`)}`);
192
+ break;
193
+ }
194
+
195
+ const prefix = `${theme.fg("dim", branch)} ${theme.fg("accent", `${task.id}:`)} `;
196
+ const suffix = task.model ? ` ${theme.fg("success", modelName(task.model))}` : "";
197
+ const promptWidth = Math.max(1, width - visibleWidth(prefix) - visibleWidth(suffix));
198
+ const prompt = truncateToWidth(task.task ?? "(prompt unavailable)", promptWidth);
199
+ lines.push(`${prefix}${theme.fg("dim", prompt)}${suffix}`);
200
+ }
201
+
202
+ if (lines.length === 0 && count > 0) {
203
+ text += `\n${theme.fg("dim", `Prompts unavailable for ${plural(count, "subagent")}.`)}`;
204
+ } else if (lines.length > 0) {
205
+ text += `\n${lines.join("\n")}`;
206
+ }
207
+
208
+ return text;
209
+ }
210
+
211
+ export function renderSubagentSpawnPrompts(details: SubagentRunRenderDetails, options: { expanded?: boolean; isPartial?: boolean }, theme: any): Component {
212
+ return new SubagentSpawnPromptComponent(details, options, theme);
213
+ }
214
+
215
+ export function renderPlainRunSummary(details: SubagentRunRenderDetails): string {
216
+ const agents = details.agents ?? [];
217
+ const verb = commandVerbForMode(details.mode);
218
+ const lines = [`${verb} ${plural(agents.length, "subagent")}, tracked 1 run`];
219
+ for (const agent of agents) {
220
+ const outputLines = outputLineSummary(agent);
221
+ lines.push(
222
+ `${statusGlyph(agent.status)} ${statusVerb(agent.status)} ${agent.id} -> ${statusLabel(agent.status)}${outputLines ? ` · ${outputLines}` : ""}`,
223
+ );
224
+ }
225
+ return lines.join("\n");
226
+ }
227
+
228
+ export function renderSpawnChatSummary(details: SubagentRunRenderDetails): string {
229
+ return `Started ${plural(spawnPromptCount(details), "subagent")}.`;
230
+ }
@@ -0,0 +1,77 @@
1
+ import * as fs from "node:fs";
2
+ import * as path from "node:path";
3
+ import { getRunState, listRunDirs, readParentSessionLink, type AgentState } from "./lib.js";
4
+ import { getLiveRun } from "./live.js";
5
+ import type { LiveAgent } from "./types.js";
6
+
7
+ /**
8
+ * Tracks live sub-agents for the event system. TUI widget rendering has been
9
+ * removed — extensions should not render custom TUI components.
10
+ */
11
+ export class SubagentOverlay {
12
+ constructor(private liveAgents: Map<string, Map<string, LiveAgent>>) {}
13
+
14
+ restoreRunningAgents(cwd: string, parentSession?: string): void {
15
+ for (const runDir of listRunDirs(cwd)) {
16
+ const running = getRunState(runDir, undefined, {
17
+ includeLineCounts: false,
18
+ checkRpcPromptFailure: false,
19
+ }).agents.filter((agent) => agent.status === "running");
20
+ if (running.length === 0) continue;
21
+ const liveRun = getLiveRun(this.liveAgents, runDir);
22
+ for (const agent of running) {
23
+ if (liveRun.has(agent.id)) continue;
24
+ const agentDir = path.join(runDir, agent.id);
25
+ const agentParentSession = readParentSessionLink(agentDir);
26
+ if (parentSession && agentParentSession && !pathsEqual(parentSession, agentParentSession)) continue;
27
+ liveRun.set(agent.id, { runDir, agentId: agent.id, parentSession: agentParentSession, completed: Promise.resolve() });
28
+ }
29
+ if (liveRun.size === 0) this.liveAgents.delete(runDir);
30
+ }
31
+ }
32
+
33
+ update(): void {
34
+ // Prune completed agents from liveAgents, but do not render any UI.
35
+ for (const [runDir, liveRun] of [...this.liveAgents.entries()]) {
36
+ const ids = [...liveRun.keys()];
37
+ const states = new Map(
38
+ getRunState(runDir, ids, {
39
+ includeLineCounts: false,
40
+ checkRpcPromptFailure: false,
41
+ }).agents.map((agent) => [agent.id, agent]),
42
+ );
43
+
44
+ for (const [agentId] of [...liveRun.entries()]) {
45
+ const agent = states.get(agentId);
46
+ if (!agent || isTerminalStatus(agent.status)) {
47
+ liveRun.delete(agentId);
48
+ continue;
49
+ }
50
+ }
51
+
52
+ if (liveRun.size === 0) this.liveAgents.delete(runDir);
53
+ }
54
+ }
55
+
56
+ dispose(): void {
57
+ // No UI state to clean up.
58
+ }
59
+ }
60
+
61
+ function isTerminalStatus(status: AgentState["status"]): boolean {
62
+ return status === "done" || status === "failed" || status === "stopped";
63
+ }
64
+
65
+
66
+ function pathsEqual(left: string, right: string): boolean {
67
+ return normalizePath(left) === normalizePath(right);
68
+ }
69
+
70
+ function normalizePath(filePath: string): string {
71
+ const resolved = path.resolve(filePath);
72
+ try {
73
+ return fs.realpathSync.native(resolved);
74
+ } catch {
75
+ return resolved;
76
+ }
77
+ }
@@ -0,0 +1,120 @@
1
+ import type { AgentTask } from "./lib.js";
2
+ import { validateBasename } from "./lib.js";
3
+ import { truncate } from "./format.js";
4
+ import type { AgentTaskPreview } from "./types.js";
5
+
6
+ function isRecord(value: unknown): value is Record<string, unknown> {
7
+ return typeof value === "object" && value !== null;
8
+ }
9
+
10
+ function optionalString(value: unknown): string | undefined {
11
+ return typeof value === "string" && value.trim().length > 0 ? value.trim() : undefined;
12
+ }
13
+
14
+ function optionalPositiveNumber(value: unknown): number | undefined {
15
+ return typeof value === "number" && Number.isFinite(value) && value > 0 ? value : undefined;
16
+ }
17
+
18
+ function optionalTimeoutMs(item: Record<string, unknown>): number | undefined {
19
+ const timeoutMs = optionalPositiveNumber(item.timeoutMs);
20
+ if (timeoutMs !== undefined) return Math.max(1, Math.round(timeoutMs));
21
+ const timeoutSeconds = optionalPositiveNumber(item.timeoutSeconds);
22
+ if (timeoutSeconds !== undefined) return Math.max(1, Math.round(timeoutSeconds * 1000));
23
+ return undefined;
24
+ }
25
+
26
+ function nextGeneratedAgentId(seenIds: Set<string>, reservedIds: Set<string>): string {
27
+ let index = 1;
28
+ while (true) {
29
+ const id = `agent-${index}`;
30
+ if (!seenIds.has(id) && !reservedIds.has(id)) return id;
31
+ index++;
32
+ }
33
+ }
34
+
35
+ export function normalizeAgentTasks(input: unknown): { tasks: AgentTask[]; error?: undefined } | { tasks?: undefined; error: string } {
36
+ if (!Array.isArray(input) || input.length === 0) {
37
+ return { error: "spawn requires at least one task in the tasks array." };
38
+ }
39
+
40
+ const reservedIds = new Set<string>();
41
+ for (const item of input) {
42
+ if (!isRecord(item)) continue;
43
+ const id = optionalString(item.id);
44
+ if (id) reservedIds.add(id);
45
+ }
46
+
47
+ const tasks: AgentTask[] = [];
48
+ const seenIds = new Set<string>();
49
+ for (let index = 0; index < input.length; index++) {
50
+ const item = input[index];
51
+ if (!isRecord(item)) {
52
+ return { error: `Task ${index + 1} must be an object.` };
53
+ }
54
+
55
+ const taskText = optionalString(item.task);
56
+ if (!taskText) {
57
+ return { error: `Task ${index + 1} is missing a non-empty task description.` };
58
+ }
59
+
60
+ const id = optionalString(item.id) ?? nextGeneratedAgentId(seenIds, reservedIds);
61
+ if (seenIds.has(id)) {
62
+ return { error: `Duplicate agent ID: "${id}". Each agent must have a unique ID.` };
63
+ }
64
+
65
+ try {
66
+ validateBasename(id, `tasks[${index}].id`);
67
+ } catch (error) {
68
+ return { error: error instanceof Error ? error.message : String(error) };
69
+ }
70
+
71
+ const tools = Array.isArray(item.tools)
72
+ ? item.tools.filter((tool): tool is string => typeof tool === "string" && tool.trim().length > 0)
73
+ : undefined;
74
+ const extraArgs = Array.isArray(item.extraArgs)
75
+ ? item.extraArgs.filter((arg): arg is string => typeof arg === "string" && arg.trim().length > 0)
76
+ : undefined;
77
+ const scope = optionalString(item.scope);
78
+ const subagentType = optionalString(item.subagentType) ?? optionalString(item.type);
79
+ const model = optionalString(item.model);
80
+ const thinking = optionalString(item.thinking);
81
+ const promptAppend = optionalString(item.promptAppend);
82
+ const promptOverride = optionalString(item.promptOverride);
83
+ const focus = optionalString(item.focus) ?? optionalString(item.attention);
84
+ const imagePaths = Array.isArray(item.imagePaths)
85
+ ? item.imagePaths.filter((imagePath): imagePath is string => typeof imagePath === "string" && imagePath.trim().length > 0).map((imagePath) => imagePath.trim())
86
+ : undefined;
87
+ const timeoutMs = optionalTimeoutMs(item);
88
+ const parentObjective = optionalString(item.parentObjective);
89
+
90
+ seenIds.add(id);
91
+ const normalizedTask: AgentTask = {
92
+ id,
93
+ task: taskText,
94
+ };
95
+ if (scope) normalizedTask.scope = scope;
96
+ if (subagentType) normalizedTask.subagentType = subagentType;
97
+ if (model) normalizedTask.model = model;
98
+ if (thinking) normalizedTask.thinking = thinking;
99
+ if (promptAppend) normalizedTask.promptAppend = promptAppend;
100
+ if (promptOverride) normalizedTask.promptOverride = promptOverride;
101
+ if (focus) normalizedTask.focus = focus;
102
+ if (imagePaths && imagePaths.length > 0) normalizedTask.imagePaths = imagePaths;
103
+ if (tools && tools.length > 0) normalizedTask.tools = tools;
104
+ if (extraArgs && extraArgs.length > 0) normalizedTask.extraArgs = extraArgs;
105
+ if (timeoutMs !== undefined) normalizedTask.timeoutMs = timeoutMs;
106
+ if (parentObjective) normalizedTask.parentObjective = parentObjective;
107
+ tasks.push(normalizedTask);
108
+ }
109
+
110
+ return { tasks };
111
+ }
112
+
113
+ export function toTaskPreviews(tasks: AgentTask[]): AgentTaskPreview[] {
114
+ return tasks.map((task) => ({
115
+ id: task.id,
116
+ task: task.task,
117
+ scope: task.scope ? truncate(task.scope, 80) : undefined,
118
+ model: task.model,
119
+ }));
120
+ }
@@ -0,0 +1,99 @@
1
+ import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
2
+ import { Type } from "@mariozechner/pi-ai";
3
+ import * as fs from "node:fs";
4
+ import * as path from "node:path";
5
+ import { ASYNC_SUBAGENT_TOOL_DESCRIPTIONS } from "../../tool-descriptions.js";
6
+ import { deleteCleanupCandidates, findCleanupCandidates, getRunRoot, removeSubagentRunsFromRegistry, resolveRunDir } from "../lib.js";
7
+
8
+ export function registerCleanupTool(pi: ExtensionAPI): void {
9
+ pi.registerTool({
10
+ ...ASYNC_SUBAGENT_TOOL_DESCRIPTIONS.cleanupAction,
11
+ parameters: Type.Object({
12
+ runRoot: Type.Optional(Type.String({ description: "Root directory (default: .pi/subagents)" })),
13
+ days: Type.Optional(Type.Number({ description: "Remove runs older than N days (default 7)", default: 7 })),
14
+ keep: Type.Optional(Type.Number({ description: "Always keep newest N runs (default 20)", default: 20 })),
15
+ delete: Type.Optional(Type.Boolean({ description: "Actually delete (default: dry-run)", default: false })),
16
+ }),
17
+
18
+ async execute(_toolCallId, params, _signal, _onUpdate, ctx) {
19
+ const expectedRoot = getRunRoot(ctx.cwd);
20
+ const runRoot = params.runRoot
21
+ ? resolveRunDir(ctx.cwd, params.runRoot)
22
+ : expectedRoot;
23
+
24
+ let canonicalExpected: string;
25
+ try {
26
+ fs.mkdirSync(expectedRoot, { recursive: true });
27
+ canonicalExpected = fs.realpathSync(expectedRoot);
28
+ } catch {
29
+ return {
30
+ content: [{ type: "text", text: `Could not create or resolve run root: ${expectedRoot}` }],
31
+ details: {},
32
+ isError: true,
33
+ };
34
+ }
35
+
36
+ let canonicalRunRoot: string;
37
+ try {
38
+ canonicalRunRoot = fs.realpathSync(runRoot);
39
+ } catch {
40
+ return {
41
+ content: [{ type: "text", text: `Run root does not exist: ${runRoot}` }],
42
+ details: {},
43
+ };
44
+ }
45
+
46
+ if (!canonicalRunRoot.startsWith(canonicalExpected + path.sep) && canonicalRunRoot !== canonicalExpected) {
47
+ return {
48
+ content: [{ type: "text", text: `Refusing to delete outside ${canonicalExpected}. Got: ${canonicalRunRoot}` }],
49
+ details: {},
50
+ isError: true,
51
+ };
52
+ }
53
+
54
+ const days = params.days ?? 7;
55
+ const keep = params.keep ?? 20;
56
+ const doDelete = params.delete ?? false;
57
+
58
+ const candidates = findCleanupCandidates(runRoot, days, keep);
59
+
60
+ if (candidates.length === 0) {
61
+ return {
62
+ content: [{ type: "text", text: `No cleanup candidates in ${runRoot} (days>${days}, keep newest ${keep}).` }],
63
+ details: { candidates, deleted: false },
64
+ };
65
+ }
66
+
67
+ if (!doDelete) {
68
+ return {
69
+ content: [
70
+ {
71
+ type: "text",
72
+ text: [
73
+ `Dry run. Would delete ${candidates.length} completed run(s):`,
74
+ ...candidates.map((c) => ` ${c}`),
75
+ "",
76
+ "Pass delete=true to actually remove.",
77
+ ].join("\n"),
78
+ },
79
+ ],
80
+ details: { candidates, deleted: false },
81
+ };
82
+ }
83
+
84
+
85
+ deleteCleanupCandidates(candidates);
86
+ removeSubagentRunsFromRegistry(ctx.cwd, candidates);
87
+
88
+ return {
89
+ content: [
90
+ {
91
+ type: "text",
92
+ text: `Deleted ${candidates.length} completed run(s):\n${candidates.map((c) => ` ${c}`).join("\n")}`,
93
+ },
94
+ ],
95
+ details: { candidates, deleted: true },
96
+ };
97
+ },
98
+ });
99
+ }