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.
- package/README.md +307 -0
- package/bin/pix.mjs +219 -0
- package/dist/app/app.d.ts +96 -0
- package/dist/app/app.js +871 -0
- package/dist/app/blink-controller.d.ts +23 -0
- package/dist/app/blink-controller.js +82 -0
- package/dist/app/cli.d.ts +8 -0
- package/dist/app/cli.js +83 -0
- package/dist/app/clipboard.d.ts +1 -0
- package/dist/app/clipboard.js +24 -0
- package/dist/app/command-controller.d.ts +18 -0
- package/dist/app/command-controller.js +58 -0
- package/dist/app/command-host.d.ts +44 -0
- package/dist/app/command-host.js +1 -0
- package/dist/app/command-model-actions.d.ts +12 -0
- package/dist/app/command-model-actions.js +176 -0
- package/dist/app/command-navigation-actions.d.ts +19 -0
- package/dist/app/command-navigation-actions.js +267 -0
- package/dist/app/command-registry.d.ts +32 -0
- package/dist/app/command-registry.js +225 -0
- package/dist/app/command-runtime.d.ts +5 -0
- package/dist/app/command-runtime.js +32 -0
- package/dist/app/command-session-actions.d.ts +20 -0
- package/dist/app/command-session-actions.js +295 -0
- package/dist/app/constants.d.ts +52 -0
- package/dist/app/constants.js +103 -0
- package/dist/app/conversation-entry-renderer.d.ts +21 -0
- package/dist/app/conversation-entry-renderer.js +81 -0
- package/dist/app/conversation-shell-renderer.d.ts +5 -0
- package/dist/app/conversation-shell-renderer.js +43 -0
- package/dist/app/conversation-tool-renderer.d.ts +16 -0
- package/dist/app/conversation-tool-renderer.js +216 -0
- package/dist/app/conversation-viewport.d.ts +55 -0
- package/dist/app/conversation-viewport.js +252 -0
- package/dist/app/dcp-stats.d.ts +2 -0
- package/dist/app/dcp-stats.js +116 -0
- package/dist/app/editor-layout-renderer.d.ts +31 -0
- package/dist/app/editor-layout-renderer.js +211 -0
- package/dist/app/editor-panels.d.ts +4 -0
- package/dist/app/editor-panels.js +130 -0
- package/dist/app/extension-actions-controller.d.ts +22 -0
- package/dist/app/extension-actions-controller.js +60 -0
- package/dist/app/extension-event-bus.d.ts +3 -0
- package/dist/app/extension-event-bus.js +23 -0
- package/dist/app/extension-ui-controller.d.ts +57 -0
- package/dist/app/extension-ui-controller.js +532 -0
- package/dist/app/file-link-opener.d.ts +2 -0
- package/dist/app/file-link-opener.js +66 -0
- package/dist/app/file-links.d.ts +10 -0
- package/dist/app/file-links.js +117 -0
- package/dist/app/guards.d.ts +3 -0
- package/dist/app/guards.js +9 -0
- package/dist/app/icons.d.ts +37 -0
- package/dist/app/icons.js +97 -0
- package/dist/app/id.d.ts +1 -0
- package/dist/app/id.js +4 -0
- package/dist/app/image-click-targets.d.ts +5 -0
- package/dist/app/image-click-targets.js +32 -0
- package/dist/app/image-opener.d.ts +2 -0
- package/dist/app/image-opener.js +64 -0
- package/dist/app/input-action-controller.d.ts +47 -0
- package/dist/app/input-action-controller.js +209 -0
- package/dist/app/input-controller.d.ts +60 -0
- package/dist/app/input-controller.js +425 -0
- package/dist/app/input-paste-handler.d.ts +27 -0
- package/dist/app/input-paste-handler.js +146 -0
- package/dist/app/menu-items-controller.d.ts +32 -0
- package/dist/app/menu-items-controller.js +182 -0
- package/dist/app/message-content.d.ts +8 -0
- package/dist/app/message-content.js +115 -0
- package/dist/app/model-ref.d.ts +13 -0
- package/dist/app/model-ref.js +50 -0
- package/dist/app/model-usage-controller.d.ts +35 -0
- package/dist/app/model-usage-controller.js +99 -0
- package/dist/app/model-usage-status.d.ts +125 -0
- package/dist/app/model-usage-status.js +749 -0
- package/dist/app/mouse-controller.d.ts +182 -0
- package/dist/app/mouse-controller.js +968 -0
- package/dist/app/native-modifiers.d.ts +3 -0
- package/dist/app/native-modifiers.js +60 -0
- package/dist/app/nerd-font-controller.d.ts +11 -0
- package/dist/app/nerd-font-controller.js +90 -0
- package/dist/app/popup-action-controller.d.ts +44 -0
- package/dist/app/popup-action-controller.js +278 -0
- package/dist/app/popup-menu-controller.d.ts +183 -0
- package/dist/app/popup-menu-controller.js +1070 -0
- package/dist/app/prompt-enhancer-controller.d.ts +40 -0
- package/dist/app/prompt-enhancer-controller.js +215 -0
- package/dist/app/queued-message-controller.d.ts +62 -0
- package/dist/app/queued-message-controller.js +377 -0
- package/dist/app/render-controller.d.ts +41 -0
- package/dist/app/render-controller.js +378 -0
- package/dist/app/render-text.d.ts +19 -0
- package/dist/app/render-text.js +67 -0
- package/dist/app/request-history.d.ts +23 -0
- package/dist/app/request-history.js +131 -0
- package/dist/app/runtime.d.ts +39 -0
- package/dist/app/runtime.js +195 -0
- package/dist/app/screen-selection.d.ts +6 -0
- package/dist/app/screen-selection.js +9 -0
- package/dist/app/screen-styler.d.ts +34 -0
- package/dist/app/screen-styler.js +168 -0
- package/dist/app/scroll-controller.d.ts +51 -0
- package/dist/app/scroll-controller.js +207 -0
- package/dist/app/session-event-controller.d.ts +69 -0
- package/dist/app/session-event-controller.js +338 -0
- package/dist/app/session-history.d.ts +23 -0
- package/dist/app/session-history.js +164 -0
- package/dist/app/session-lifecycle-controller.d.ts +55 -0
- package/dist/app/session-lifecycle-controller.js +116 -0
- package/dist/app/session-search.d.ts +24 -0
- package/dist/app/session-search.js +215 -0
- package/dist/app/shell-command.d.ts +27 -0
- package/dist/app/shell-command.js +176 -0
- package/dist/app/shell-controller.d.ts +28 -0
- package/dist/app/shell-controller.js +124 -0
- package/dist/app/slash-commands.d.ts +6 -0
- package/dist/app/slash-commands.js +75 -0
- package/dist/app/startup-checks.d.ts +8 -0
- package/dist/app/startup-checks.js +59 -0
- package/dist/app/startup-info.d.ts +3 -0
- package/dist/app/startup-info.js +176 -0
- package/dist/app/status-controller.d.ts +35 -0
- package/dist/app/status-controller.js +105 -0
- package/dist/app/status-line-renderer.d.ts +68 -0
- package/dist/app/status-line-renderer.js +453 -0
- package/dist/app/subagents-files.d.ts +10 -0
- package/dist/app/subagents-files.js +193 -0
- package/dist/app/subagents-model.d.ts +23 -0
- package/dist/app/subagents-model.js +224 -0
- package/dist/app/subagents-widget-controller.d.ts +43 -0
- package/dist/app/subagents-widget-controller.js +311 -0
- package/dist/app/tab-line-renderer.d.ts +26 -0
- package/dist/app/tab-line-renderer.js +222 -0
- package/dist/app/tabs-controller.d.ts +100 -0
- package/dist/app/tabs-controller.js +885 -0
- package/dist/app/terminal-controller.d.ts +40 -0
- package/dist/app/terminal-controller.js +135 -0
- package/dist/app/terminal-edit-shortcuts.d.ts +23 -0
- package/dist/app/terminal-edit-shortcuts.js +138 -0
- package/dist/app/terminal-output-buffer.d.ts +20 -0
- package/dist/app/terminal-output-buffer.js +52 -0
- package/dist/app/toast-controller.d.ts +13 -0
- package/dist/app/toast-controller.js +40 -0
- package/dist/app/toast-renderer.d.ts +9 -0
- package/dist/app/toast-renderer.js +79 -0
- package/dist/app/todo-model.d.ts +22 -0
- package/dist/app/todo-model.js +179 -0
- package/dist/app/todo-widget-controller.d.ts +21 -0
- package/dist/app/todo-widget-controller.js +59 -0
- package/dist/app/tool-block-renderer.d.ts +26 -0
- package/dist/app/tool-block-renderer.js +439 -0
- package/dist/app/types.d.ts +550 -0
- package/dist/app/types.js +1 -0
- package/dist/app/update.d.ts +36 -0
- package/dist/app/update.js +315 -0
- package/dist/app/voice-controller.d.ts +52 -0
- package/dist/app/voice-controller.js +600 -0
- package/dist/app/workspace-actions-controller.d.ts +40 -0
- package/dist/app/workspace-actions-controller.js +215 -0
- package/dist/app/workspace-undo.d.ts +44 -0
- package/dist/app/workspace-undo.js +215 -0
- package/dist/config.d.ts +62 -0
- package/dist/config.js +527 -0
- package/dist/context-progress-bar.d.ts +17 -0
- package/dist/context-progress-bar.js +48 -0
- package/dist/default-pix-config.d.ts +1 -0
- package/dist/default-pix-config.js +375 -0
- package/dist/fuzzy.d.ts +23 -0
- package/dist/fuzzy.js +165 -0
- package/dist/input-editor-files.d.ts +15 -0
- package/dist/input-editor-files.js +62 -0
- package/dist/input-editor.d.ts +186 -0
- package/dist/input-editor.js +835 -0
- package/dist/main.d.ts +1 -0
- package/dist/main.js +12 -0
- package/dist/markdown-format.d.ts +22 -0
- package/dist/markdown-format.js +542 -0
- package/dist/sdk.d.ts +3 -0
- package/dist/sdk.js +1 -0
- package/dist/syntax-highlight.d.ts +22 -0
- package/dist/syntax-highlight.js +528 -0
- package/dist/terminal-width.d.ts +6 -0
- package/dist/terminal-width.js +245 -0
- package/dist/theme.d.ts +56 -0
- package/dist/theme.js +118 -0
- package/dist/tool-renderers/apply-patch.d.ts +2 -0
- package/dist/tool-renderers/apply-patch.js +36 -0
- package/dist/tool-renderers/ast.d.ts +2 -0
- package/dist/tool-renderers/ast.js +5 -0
- package/dist/tool-renderers/compress.d.ts +2 -0
- package/dist/tool-renderers/compress.js +126 -0
- package/dist/tool-renderers/index.d.ts +3 -0
- package/dist/tool-renderers/index.js +44 -0
- package/dist/tool-renderers/question.d.ts +2 -0
- package/dist/tool-renderers/question.js +88 -0
- package/dist/tool-renderers/read.d.ts +2 -0
- package/dist/tool-renderers/read.js +36 -0
- package/dist/tool-renderers/repo.d.ts +2 -0
- package/dist/tool-renderers/repo.js +13 -0
- package/dist/tool-renderers/shell.d.ts +3 -0
- package/dist/tool-renderers/shell.js +27 -0
- package/dist/tool-renderers/skill.d.ts +2 -0
- package/dist/tool-renderers/skill.js +132 -0
- package/dist/tool-renderers/subagents.d.ts +2 -0
- package/dist/tool-renderers/subagents.js +51 -0
- package/dist/tool-renderers/todo.d.ts +2 -0
- package/dist/tool-renderers/todo.js +9 -0
- package/dist/tool-renderers/types.d.ts +42 -0
- package/dist/tool-renderers/types.js +1 -0
- package/dist/tool-renderers/utils.d.ts +31 -0
- package/dist/tool-renderers/utils.js +230 -0
- package/dist/tool-renderers/web.d.ts +3 -0
- package/dist/tool-renderers/web.js +9 -0
- package/dist/tool-renderers/write.d.ts +2 -0
- package/dist/tool-renderers/write.js +42 -0
- package/dist/ui.d.ts +56 -0
- package/dist/ui.js +94 -0
- package/docs/release.md +81 -0
- package/extensions/question/contract.ts +100 -0
- package/extensions/question/index.ts +34 -0
- package/extensions/question/render.ts +28 -0
- package/extensions/question/result.ts +86 -0
- package/extensions/question/tool-description.ts +11 -0
- package/extensions/question/tui.ts +629 -0
- package/extensions/question/types.ts +123 -0
- package/extensions/session-title/config.ts +169 -0
- package/extensions/session-title/index.ts +459 -0
- package/extensions/terminal-bell/index.ts +315 -0
- package/external/pi-tools-suite/README.md +242 -0
- package/external/pi-tools-suite/index.ts +1 -0
- package/external/pi-tools-suite/licenses/ollama-pi-web-search.MIT +21 -0
- package/external/pi-tools-suite/licenses/opencode-mystatus-MIT.txt +21 -0
- package/external/pi-tools-suite/package.json +53 -0
- package/external/pi-tools-suite/src/antigravity-auth/auth-store.ts +194 -0
- package/external/pi-tools-suite/src/antigravity-auth/commands.ts +80 -0
- package/external/pi-tools-suite/src/antigravity-auth/constants.ts +26 -0
- package/external/pi-tools-suite/src/antigravity-auth/headers.ts +20 -0
- package/external/pi-tools-suite/src/antigravity-auth/index.ts +104 -0
- package/external/pi-tools-suite/src/antigravity-auth/models.ts +86 -0
- package/external/pi-tools-suite/src/antigravity-auth/oauth.ts +305 -0
- package/external/pi-tools-suite/src/antigravity-auth/payload.ts +423 -0
- package/external/pi-tools-suite/src/antigravity-auth/status.ts +78 -0
- package/external/pi-tools-suite/src/antigravity-auth/stream.ts +302 -0
- package/external/pi-tools-suite/src/antigravity-auth/types.ts +113 -0
- package/external/pi-tools-suite/src/ast-grep/index.ts +6 -0
- package/external/pi-tools-suite/src/ast-grep/render.ts +70 -0
- package/external/pi-tools-suite/src/ast-grep/schema.ts +109 -0
- package/external/pi-tools-suite/src/ast-grep/tool.ts +345 -0
- package/external/pi-tools-suite/src/ast-grep/types.ts +55 -0
- package/external/pi-tools-suite/src/ast-grep/utils.ts +65 -0
- package/external/pi-tools-suite/src/async-subagents/async-subagents.sample.jsonc +222 -0
- package/external/pi-tools-suite/src/async-subagents/commands.ts +518 -0
- package/external/pi-tools-suite/src/async-subagents/constants.ts +11 -0
- package/external/pi-tools-suite/src/async-subagents/core/agent-strategy.ts +74 -0
- package/external/pi-tools-suite/src/async-subagents/core/attachment-bridge.ts +133 -0
- package/external/pi-tools-suite/src/async-subagents/core/cleanup.ts +66 -0
- package/external/pi-tools-suite/src/async-subagents/core/concurrency.ts +90 -0
- package/external/pi-tools-suite/src/async-subagents/core/config.ts +819 -0
- package/external/pi-tools-suite/src/async-subagents/core/log-limits.ts +166 -0
- package/external/pi-tools-suite/src/async-subagents/core/model-fallback.ts +133 -0
- package/external/pi-tools-suite/src/async-subagents/core/paths.ts +47 -0
- package/external/pi-tools-suite/src/async-subagents/core/pi-invocation.ts +35 -0
- package/external/pi-tools-suite/src/async-subagents/core/presets.ts +67 -0
- package/external/pi-tools-suite/src/async-subagents/core/process.ts +15 -0
- package/external/pi-tools-suite/src/async-subagents/core/prompt.ts +66 -0
- package/external/pi-tools-suite/src/async-subagents/core/registry.ts +224 -0
- package/external/pi-tools-suite/src/async-subagents/core/retry.ts +191 -0
- package/external/pi-tools-suite/src/async-subagents/core/routing.ts +259 -0
- package/external/pi-tools-suite/src/async-subagents/core/sessions.ts +138 -0
- package/external/pi-tools-suite/src/async-subagents/core/spawn.ts +688 -0
- package/external/pi-tools-suite/src/async-subagents/core/state.ts +281 -0
- package/external/pi-tools-suite/src/async-subagents/core/stop.ts +131 -0
- package/external/pi-tools-suite/src/async-subagents/core/structured-result.ts +237 -0
- package/external/pi-tools-suite/src/async-subagents/core/tool-guard.ts +34 -0
- package/external/pi-tools-suite/src/async-subagents/core/types.ts +150 -0
- package/external/pi-tools-suite/src/async-subagents/core/ultrawork-auto.ts +184 -0
- package/external/pi-tools-suite/src/async-subagents/core/utils.ts +11 -0
- package/external/pi-tools-suite/src/async-subagents/format.ts +41 -0
- package/external/pi-tools-suite/src/async-subagents/index.ts +422 -0
- package/external/pi-tools-suite/src/async-subagents/lib.ts +88 -0
- package/external/pi-tools-suite/src/async-subagents/live.ts +10 -0
- package/external/pi-tools-suite/src/async-subagents/polling.ts +83 -0
- package/external/pi-tools-suite/src/async-subagents/render.ts +230 -0
- package/external/pi-tools-suite/src/async-subagents/subagent-overlay.ts +77 -0
- package/external/pi-tools-suite/src/async-subagents/tasks.ts +120 -0
- package/external/pi-tools-suite/src/async-subagents/tools/cleanup.ts +99 -0
- package/external/pi-tools-suite/src/async-subagents/tools/result.ts +179 -0
- package/external/pi-tools-suite/src/async-subagents/tools/spawn.ts +372 -0
- package/external/pi-tools-suite/src/async-subagents/tools/status.ts +60 -0
- package/external/pi-tools-suite/src/async-subagents/tools/stop.ts +79 -0
- package/external/pi-tools-suite/src/async-subagents/tools/subagents.ts +152 -0
- package/external/pi-tools-suite/src/async-subagents/tools/wait.ts +67 -0
- package/external/pi-tools-suite/src/async-subagents/types.ts +45 -0
- package/external/pi-tools-suite/src/async-subagents/ui.ts +5 -0
- package/external/pi-tools-suite/src/compress/commands.ts +440 -0
- package/external/pi-tools-suite/src/compress/compress-tool.ts +368 -0
- package/external/pi-tools-suite/src/compress/compression-blocks.ts +524 -0
- package/external/pi-tools-suite/src/compress/config.ts +310 -0
- package/external/pi-tools-suite/src/compress/dcp-tui-filter.ts +498 -0
- package/external/pi-tools-suite/src/compress/index.ts +397 -0
- package/external/pi-tools-suite/src/compress/prompts.ts +269 -0
- package/external/pi-tools-suite/src/compress/pruner-candidates.ts +176 -0
- package/external/pi-tools-suite/src/compress/pruner-compression-blocks.ts +260 -0
- package/external/pi-tools-suite/src/compress/pruner-message-ids.ts +147 -0
- package/external/pi-tools-suite/src/compress/pruner-metadata.ts +268 -0
- package/external/pi-tools-suite/src/compress/pruner-nudge.ts +315 -0
- package/external/pi-tools-suite/src/compress/pruner-tools.ts +263 -0
- package/external/pi-tools-suite/src/compress/pruner-types.ts +25 -0
- package/external/pi-tools-suite/src/compress/pruner.ts +92 -0
- package/external/pi-tools-suite/src/compress/state.ts +486 -0
- package/external/pi-tools-suite/src/compress/ui.ts +308 -0
- package/external/pi-tools-suite/src/config.ts +176 -0
- package/external/pi-tools-suite/src/context-usage.ts +31 -0
- package/external/pi-tools-suite/src/default-pi-tools-suite-config.ts +355 -0
- package/external/pi-tools-suite/src/index.ts +46 -0
- package/external/pi-tools-suite/src/lib/lsp.ts +62 -0
- package/external/pi-tools-suite/src/lib/project.ts +42 -0
- package/external/pi-tools-suite/src/lib/tool-args.ts +137 -0
- package/external/pi-tools-suite/src/lsp/_shared/config.ts +156 -0
- package/external/pi-tools-suite/src/lsp/_shared/glob.ts +60 -0
- package/external/pi-tools-suite/src/lsp/_shared/output.ts +102 -0
- package/external/pi-tools-suite/src/lsp/_shared/paths.ts +138 -0
- package/external/pi-tools-suite/src/lsp/_shared/runner.ts +64 -0
- package/external/pi-tools-suite/src/lsp/_shared/template.ts +23 -0
- package/external/pi-tools-suite/src/lsp/_shared/trust.ts +116 -0
- package/external/pi-tools-suite/src/lsp/_shared/types.ts +98 -0
- package/external/pi-tools-suite/src/lsp/async.ts +29 -0
- package/external/pi-tools-suite/src/lsp/child-process.ts +81 -0
- package/external/pi-tools-suite/src/lsp/client.ts +340 -0
- package/external/pi-tools-suite/src/lsp/constants.ts +9 -0
- package/external/pi-tools-suite/src/lsp/diagnostics-store.ts +64 -0
- package/external/pi-tools-suite/src/lsp/documents.ts +24 -0
- package/external/pi-tools-suite/src/lsp/index.ts +31 -0
- package/external/pi-tools-suite/src/lsp/lsp-utils.ts +37 -0
- package/external/pi-tools-suite/src/lsp/manager.ts +190 -0
- package/external/pi-tools-suite/src/lsp/mutation-events.ts +78 -0
- package/external/pi-tools-suite/src/lsp/renderer.ts +1 -0
- package/external/pi-tools-suite/src/lsp/tool-result.ts +6 -0
- package/external/pi-tools-suite/src/lsp/tsserver.ts +107 -0
- package/external/pi-tools-suite/src/lsp/types.ts +15 -0
- package/external/pi-tools-suite/src/model-tools/apply-patch.ts +590 -0
- package/external/pi-tools-suite/src/model-tools/index.ts +430 -0
- package/external/pi-tools-suite/src/model-tools/path-utils.ts +6 -0
- package/external/pi-tools-suite/src/model-tools/tool-args.ts +11 -0
- package/external/pi-tools-suite/src/prompt-commands/index.ts +349 -0
- package/external/pi-tools-suite/src/repo-discovery/index.ts +384 -0
- package/external/pi-tools-suite/src/repo-discovery/project.ts +7 -0
- package/external/pi-tools-suite/src/startup-section.ts +13 -0
- package/external/pi-tools-suite/src/terminal-bell/index.ts +309 -0
- package/external/pi-tools-suite/src/todo/index.ts +201 -0
- package/external/pi-tools-suite/src/todo/state/auto-clear.ts +13 -0
- package/external/pi-tools-suite/src/todo/state/invariants.ts +21 -0
- package/external/pi-tools-suite/src/todo/state/persistence.ts +94 -0
- package/external/pi-tools-suite/src/todo/state/replay.ts +38 -0
- package/external/pi-tools-suite/src/todo/state/selectors.ts +49 -0
- package/external/pi-tools-suite/src/todo/state/state-reducer.ts +362 -0
- package/external/pi-tools-suite/src/todo/state/state.ts +18 -0
- package/external/pi-tools-suite/src/todo/state/store.ts +52 -0
- package/external/pi-tools-suite/src/todo/state/task-graph.ts +57 -0
- package/external/pi-tools-suite/src/todo/todo.ts +487 -0
- package/external/pi-tools-suite/src/todo/tool/response-envelope.ts +143 -0
- package/external/pi-tools-suite/src/todo/tool/types.ts +188 -0
- package/external/pi-tools-suite/src/todo/view/format.ts +18 -0
- package/external/pi-tools-suite/src/todo/view/labels.ts +13 -0
- package/external/pi-tools-suite/src/tool-descriptions.ts +369 -0
- package/external/pi-tools-suite/src/usage/index.ts +152 -0
- package/external/pi-tools-suite/src/usage/lib/copilot.ts +535 -0
- package/external/pi-tools-suite/src/usage/lib/google.ts +478 -0
- package/external/pi-tools-suite/src/usage/lib/openai.ts +268 -0
- package/external/pi-tools-suite/src/usage/lib/types.ts +114 -0
- package/external/pi-tools-suite/src/usage/lib/utils.ts +134 -0
- package/external/pi-tools-suite/src/usage/lib/zhipu.ts +228 -0
- package/external/pi-tools-suite/src/web-search/index.ts +397 -0
- package/package.json +89 -0
- package/skills/context7/SKILL.md +69 -0
- package/skills/context7/scripts/context7.sh +73 -0
- package/skills/handoff/SKILL.md +15 -0
- package/skills/pdf/LICENSE.txt +30 -0
- package/skills/pdf/SKILL.md +314 -0
- package/skills/pdf/forms.md +294 -0
- package/skills/pdf/reference.md +612 -0
- package/skills/pdf/scripts/check_bounding_boxes.py +65 -0
- package/skills/pdf/scripts/check_fillable_fields.py +11 -0
- package/skills/pdf/scripts/convert_pdf_to_images.py +33 -0
- package/skills/pdf/scripts/create_validation_image.py +37 -0
- package/skills/pdf/scripts/extract_form_field_info.py +122 -0
- package/skills/pdf/scripts/extract_form_structure.py +115 -0
- package/skills/pdf/scripts/fill_fillable_fields.py +98 -0
- package/skills/pdf/scripts/fill_pdf_form_with_annotations.py +107 -0
- package/skills/playwright-cli/SKILL.md +388 -0
- package/skills/playwright-cli/references/element-attributes.md +23 -0
- package/skills/playwright-cli/references/playwright-tests.md +39 -0
- package/skills/playwright-cli/references/request-mocking.md +87 -0
- package/skills/playwright-cli/references/running-code.md +241 -0
- package/skills/playwright-cli/references/session-management.md +225 -0
- package/skills/playwright-cli/references/spec-driven-testing.md +305 -0
- package/skills/playwright-cli/references/storage-state.md +275 -0
- package/skills/playwright-cli/references/test-generation.md +134 -0
- package/skills/playwright-cli/references/tracing.md +139 -0
- package/skills/playwright-cli/references/video-recording.md +143 -0
- package/skills/simplify/SKILL.md +51 -0
- package/skills/skill-creator/LICENSE.txt +202 -0
- package/skills/skill-creator/SKILL.md +485 -0
- package/skills/skill-creator/agents/analyzer.md +274 -0
- package/skills/skill-creator/agents/comparator.md +202 -0
- package/skills/skill-creator/agents/grader.md +223 -0
- package/skills/skill-creator/assets/eval_review.html +146 -0
- package/skills/skill-creator/eval-viewer/generate_review.py +471 -0
- package/skills/skill-creator/eval-viewer/viewer.html +1325 -0
- package/skills/skill-creator/references/schemas.md +430 -0
- package/skills/skill-creator/scripts/__init__.py +0 -0
- package/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
- package/skills/skill-creator/scripts/generate_report.py +326 -0
- package/skills/skill-creator/scripts/improve_description.py +247 -0
- package/skills/skill-creator/scripts/package_skill.py +136 -0
- package/skills/skill-creator/scripts/quick_validate.py +103 -0
- package/skills/skill-creator/scripts/run_eval.py +310 -0
- package/skills/skill-creator/scripts/run_loop.py +328 -0
- 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
|
+
}
|