rimuru-ai 1.19.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.rimuru/AGENTS.md +30 -0
- package/.rimuru/agents/backend.md +27 -0
- package/.rimuru/agents/database.md +31 -0
- package/.rimuru/agents/devops.md +30 -0
- package/.rimuru/agents/document-prep.md +49 -0
- package/.rimuru/agents/erp-architect.md +41 -0
- package/.rimuru/agents/ethical-hacking.md +49 -0
- package/.rimuru/agents/frontend.md +31 -0
- package/.rimuru/agents/fullstack.md +24 -0
- package/.rimuru/agents/system-engineer.md +31 -0
- package/.rimuru/agents/veldora-agent-tool-dev.md +30 -0
- package/.rimuru/agents/veldora-backend-dev.md +32 -0
- package/.rimuru/agents/veldora-cicd.md +32 -0
- package/.rimuru/agents/veldora-database.md +32 -0
- package/.rimuru/agents/veldora-doc.md +87 -0
- package/.rimuru/agents/veldora-frontend-dev.md +32 -0
- package/.rimuru/agents/veldora-great-sage.md +33 -0
- package/.rimuru/agents/veldora-mcp-creator.md +30 -0
- package/.rimuru/agents/veldora-pro.md +224 -0
- package/.rimuru/agents/veldora-prompt-enhancer.md +27 -0
- package/.rimuru/agents/veldora-skill-creator.md +28 -0
- package/.rimuru/agents/veldora.md +225 -0
- package/.rimuru/agents/veldorapro-agent-tool-dev.md +29 -0
- package/.rimuru/agents/veldorapro-backend-dev.md +29 -0
- package/.rimuru/agents/veldorapro-cicd.md +29 -0
- package/.rimuru/agents/veldorapro-database.md +29 -0
- package/.rimuru/agents/veldorapro-frontend-dev.md +29 -0
- package/.rimuru/agents/veldorapro-great-sage.md +33 -0
- package/.rimuru/agents/veldorapro-mcp-creator.md +27 -0
- package/.rimuru/agents/veldorapro-prompt-enhancer.md +25 -0
- package/.rimuru/agents/veldorapro-skill-creator.md +27 -0
- package/.rimuru/command/ai-deps.md +24 -0
- package/.rimuru/command/changelog.md +49 -0
- package/.rimuru/command/commit.md +37 -0
- package/.rimuru/command/issues.md +23 -0
- package/.rimuru/command/learn.md +42 -0
- package/.rimuru/command/rmslop.md +15 -0
- package/.rimuru/command/spellcheck.md +5 -0
- package/.rimuru/command/translate.md +14 -0
- package/.rimuru/glossary/README.md +63 -0
- package/.rimuru/glossary/ar.md +28 -0
- package/.rimuru/glossary/br.md +34 -0
- package/.rimuru/glossary/bs.md +33 -0
- package/.rimuru/glossary/da.md +27 -0
- package/.rimuru/glossary/de.md +27 -0
- package/.rimuru/glossary/es.md +27 -0
- package/.rimuru/glossary/fr.md +27 -0
- package/.rimuru/glossary/ja.md +33 -0
- package/.rimuru/glossary/ko.md +27 -0
- package/.rimuru/glossary/no.md +38 -0
- package/.rimuru/glossary/pl.md +27 -0
- package/.rimuru/glossary/ru.md +27 -0
- package/.rimuru/glossary/th.md +34 -0
- package/.rimuru/glossary/tr.md +38 -0
- package/.rimuru/glossary/zh-cn.md +42 -0
- package/.rimuru/glossary/zh-tw.md +42 -0
- package/.rimuru/improver/changelog.md +250 -0
- package/.rimuru/improver/knowledge.md +172 -0
- package/.rimuru/improver/plugins.md +21 -0
- package/.rimuru/improver/skills.md +60 -0
- package/.rimuru/improver/token-audit.md +21 -0
- package/.rimuru/opencode.jsonc +140 -0
- package/.rimuru/plugins/smoke-theme.json +223 -0
- package/.rimuru/plugins/tui-smoke.tsx +1019 -0
- package/.rimuru/skills/effect/SKILL.md +38 -0
- package/.rimuru/themes/mytheme.json +223 -0
- package/.rimuru/tool/github-pr-search.ts +64 -0
- package/.rimuru/tool/github-triage.ts +60 -0
- package/README.md +31 -0
- package/package.json +167 -0
- package/src/account/account.ts +463 -0
- package/src/account/repo.ts +173 -0
- package/src/account/schema.ts +99 -0
- package/src/account/url.ts +8 -0
- package/src/acp/agent.ts +95 -0
- package/src/acp/config-option.ts +203 -0
- package/src/acp/content.ts +250 -0
- package/src/acp/directory.ts +210 -0
- package/src/acp/error.ts +90 -0
- package/src/acp/event.ts +342 -0
- package/src/acp/permission.ts +124 -0
- package/src/acp/profile.ts +42 -0
- package/src/acp/service.ts +1048 -0
- package/src/acp/session.ts +231 -0
- package/src/acp/tool.ts +367 -0
- package/src/acp/usage.ts +232 -0
- package/src/agent/agent.ts +459 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +9 -0
- package/src/agent/prompt/explore.txt +18 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +44 -0
- package/src/agent/subagent-permissions.ts +27 -0
- package/src/audio.d.ts +14 -0
- package/src/auth/index.ts +99 -0
- package/src/background/job.ts +39 -0
- package/src/bus/global.ts +22 -0
- package/src/cli/bootstrap.ts +11 -0
- package/src/cli/cmd/account.ts +264 -0
- package/src/cli/cmd/acp.ts +73 -0
- package/src/cli/cmd/agent.ts +259 -0
- package/src/cli/cmd/attach.ts +97 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/db.ts +62 -0
- package/src/cli/cmd/debug/agent.handler.ts +193 -0
- package/src/cli/cmd/debug/agent.ts +27 -0
- package/src/cli/cmd/debug/config.ts +14 -0
- package/src/cli/cmd/debug/file.ts +73 -0
- package/src/cli/cmd/debug/index.ts +87 -0
- package/src/cli/cmd/debug/lsp.ts +50 -0
- package/src/cli/cmd/debug/ripgrep.ts +79 -0
- package/src/cli/cmd/debug/scrap.ts +15 -0
- package/src/cli/cmd/debug/skill.ts +15 -0
- package/src/cli/cmd/debug/snapshot.ts +50 -0
- package/src/cli/cmd/debug/startup.ts +11 -0
- package/src/cli/cmd/debug/v2.ts +49 -0
- package/src/cli/cmd/export.ts +292 -0
- package/src/cli/cmd/generate.ts +54 -0
- package/src/cli/cmd/github.handler.ts +1593 -0
- package/src/cli/cmd/github.shared.ts +30 -0
- package/src/cli/cmd/github.ts +42 -0
- package/src/cli/cmd/import.ts +224 -0
- package/src/cli/cmd/mcp.ts +850 -0
- package/src/cli/cmd/models.ts +66 -0
- package/src/cli/cmd/plug.ts +230 -0
- package/src/cli/cmd/pr.ts +115 -0
- package/src/cli/cmd/prompt-display.ts +1 -0
- package/src/cli/cmd/providers.ts +534 -0
- package/src/cli/cmd/run/demo.ts +1274 -0
- package/src/cli/cmd/run/entry.body.ts +205 -0
- package/src/cli/cmd/run/footer.command.tsx +1064 -0
- package/src/cli/cmd/run/footer.menu.tsx +351 -0
- package/src/cli/cmd/run/footer.permission.tsx +472 -0
- package/src/cli/cmd/run/footer.prompt.tsx +1306 -0
- package/src/cli/cmd/run/footer.question.tsx +573 -0
- package/src/cli/cmd/run/footer.subagent.tsx +173 -0
- package/src/cli/cmd/run/footer.ts +1129 -0
- package/src/cli/cmd/run/footer.view.tsx +943 -0
- package/src/cli/cmd/run/footer.width.ts +27 -0
- package/src/cli/cmd/run/permission.shared.ts +256 -0
- package/src/cli/cmd/run/prompt.editor.ts +157 -0
- package/src/cli/cmd/run/prompt.shared.ts +153 -0
- package/src/cli/cmd/run/question.shared.ts +340 -0
- package/src/cli/cmd/run/runtime.boot.ts +202 -0
- package/src/cli/cmd/run/runtime.lifecycle.ts +406 -0
- package/src/cli/cmd/run/runtime.queue.ts +349 -0
- package/src/cli/cmd/run/runtime.shared.ts +17 -0
- package/src/cli/cmd/run/runtime.stdin.ts +37 -0
- package/src/cli/cmd/run/runtime.ts +814 -0
- package/src/cli/cmd/run/scrollback.shared.ts +92 -0
- package/src/cli/cmd/run/scrollback.surface.ts +431 -0
- package/src/cli/cmd/run/scrollback.writer.tsx +352 -0
- package/src/cli/cmd/run/session-data.ts +1113 -0
- package/src/cli/cmd/run/session-replay.ts +374 -0
- package/src/cli/cmd/run/session.shared.ts +196 -0
- package/src/cli/cmd/run/splash.ts +280 -0
- package/src/cli/cmd/run/stream.transport.ts +1462 -0
- package/src/cli/cmd/run/stream.ts +175 -0
- package/src/cli/cmd/run/subagent-data.ts +876 -0
- package/src/cli/cmd/run/theme.ts +690 -0
- package/src/cli/cmd/run/tool.ts +1489 -0
- package/src/cli/cmd/run/trace.ts +94 -0
- package/src/cli/cmd/run/turn-summary.ts +47 -0
- package/src/cli/cmd/run/types.ts +350 -0
- package/src/cli/cmd/run/variant.shared.ts +215 -0
- package/src/cli/cmd/run.ts +894 -0
- package/src/cli/cmd/serve.ts +24 -0
- package/src/cli/cmd/session.ts +147 -0
- package/src/cli/cmd/stats.ts +393 -0
- package/src/cli/cmd/tui.ts +224 -0
- package/src/cli/cmd/uninstall.ts +353 -0
- package/src/cli/cmd/upgrade.ts +74 -0
- package/src/cli/cmd/web.ts +84 -0
- package/src/cli/effect/prompt.ts +37 -0
- package/src/cli/effect-cmd.ts +96 -0
- package/src/cli/error.ts +130 -0
- package/src/cli/heap.ts +45 -0
- package/src/cli/logo.ts +1 -0
- package/src/cli/network.ts +64 -0
- package/src/cli/tui/layer.ts +7 -0
- package/src/cli/tui/validate-session.ts +29 -0
- package/src/cli/tui/worker.ts +71 -0
- package/src/cli/ui.ts +98 -0
- package/src/cli/upgrade.ts +53 -0
- package/src/command/index.ts +184 -0
- package/src/command/template/initialize.txt +66 -0
- package/src/command/template/review.txt +101 -0
- package/src/config/agent.ts +59 -0
- package/src/config/command.ts +39 -0
- package/src/config/config.ts +686 -0
- package/src/config/entry-name.ts +19 -0
- package/src/config/managed.ts +69 -0
- package/src/config/markdown.ts +36 -0
- package/src/config/parse.ts +79 -0
- package/src/config/paths.ts +45 -0
- package/src/config/plugin.ts +79 -0
- package/src/config/tui-cwd.ts +5 -0
- package/src/config/tui-host-attention.ts +21 -0
- package/src/config/tui-migrate.ts +132 -0
- package/src/config/tui.ts +274 -0
- package/src/config/variable.ts +91 -0
- package/src/control-plane/adapters/index.ts +41 -0
- package/src/control-plane/adapters/worktree.ts +96 -0
- package/src/control-plane/dev/README.md +19 -0
- package/src/control-plane/dev/debug-workspace-plugin.ts +73 -0
- package/src/control-plane/types.ts +59 -0
- package/src/control-plane/util.ts +39 -0
- package/src/control-plane/workspace-adapter-runtime.ts +51 -0
- package/src/control-plane/workspace-context.ts +26 -0
- package/src/control-plane/workspace.ts +989 -0
- package/src/effect/app-runtime.ts +132 -0
- package/src/effect/bootstrap-runtime.ts +23 -0
- package/src/effect/bridge.ts +84 -0
- package/src/effect/config-service.ts +67 -0
- package/src/effect/instance-ref.ts +11 -0
- package/src/effect/instance-registry.ts +12 -0
- package/src/effect/instance-state.ts +69 -0
- package/src/effect/promise.ts +17 -0
- package/src/effect/run-service.ts +47 -0
- package/src/effect/runner.ts +217 -0
- package/src/effect/runtime-flags.ts +79 -0
- package/src/env/index.ts +43 -0
- package/src/event-v2-bridge.ts +79 -0
- package/src/format/formatter.ts +404 -0
- package/src/format/index.ts +205 -0
- package/src/git/index.ts +350 -0
- package/src/id/id.ts +80 -0
- package/src/ide/index.ts +61 -0
- package/src/image/image.ts +174 -0
- package/src/index.ts +142 -0
- package/src/installation/index.ts +350 -0
- package/src/lsp/client.ts +650 -0
- package/src/lsp/diagnostic.ts +29 -0
- package/src/lsp/language.ts +121 -0
- package/src/lsp/launch.ts +21 -0
- package/src/lsp/lsp.ts +511 -0
- package/src/lsp/server.ts +1983 -0
- package/src/markdown.d.ts +4 -0
- package/src/mcp/auth.ts +174 -0
- package/src/mcp/catalog.ts +153 -0
- package/src/mcp/index.ts +946 -0
- package/src/mcp/oauth-callback.ts +233 -0
- package/src/mcp/oauth-provider.ts +206 -0
- package/src/node.ts +4 -0
- package/src/patch/index.ts +686 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/evaluate.ts +1 -0
- package/src/permission/index.ts +230 -0
- package/src/plugin/azure.ts +26 -0
- package/src/plugin/cloudflare.ts +76 -0
- package/src/plugin/digitalocean.ts +383 -0
- package/src/plugin/github-copilot/copilot.ts +413 -0
- package/src/plugin/github-copilot/models.ts +246 -0
- package/src/plugin/index.ts +316 -0
- package/src/plugin/install.ts +439 -0
- package/src/plugin/loader.ts +237 -0
- package/src/plugin/meta.ts +188 -0
- package/src/plugin/openai/README.md +31 -0
- package/src/plugin/openai/codex.ts +641 -0
- package/src/plugin/openai/ws-pool.ts +270 -0
- package/src/plugin/openai/ws.ts +381 -0
- package/src/plugin/pty-environment.ts +24 -0
- package/src/plugin/shared.ts +323 -0
- package/src/plugin/snowflake-cortex.ts +529 -0
- package/src/plugin/tui/internal.ts +10 -0
- package/src/plugin/tui/runtime.ts +1130 -0
- package/src/plugin/xai.ts +716 -0
- package/src/project/bootstrap-service.ts +9 -0
- package/src/project/bootstrap.ts +76 -0
- package/src/project/instance-context.ts +24 -0
- package/src/project/instance-layer.ts +11 -0
- package/src/project/instance-runtime.ts +16 -0
- package/src/project/instance-store.ts +209 -0
- package/src/project/project.ts +519 -0
- package/src/project/vcs.ts +431 -0
- package/src/provider/auth.ts +233 -0
- package/src/provider/error.ts +188 -0
- package/src/provider/model-status.ts +8 -0
- package/src/provider/provider.ts +1979 -0
- package/src/provider/transform.ts +1426 -0
- package/src/question/index.ts +229 -0
- package/src/question/schema.ts +10 -0
- package/src/server/auth.ts +48 -0
- package/src/server/event.ts +13 -0
- package/src/server/global-lifecycle.ts +28 -0
- package/src/server/init-projectors.ts +3 -0
- package/src/server/mdns.ts +47 -0
- package/src/server/projectors.ts +1 -0
- package/src/server/proxy-util.ts +48 -0
- package/src/server/routes/instance/httpapi/AGENTS.md +39 -0
- package/src/server/routes/instance/httpapi/api.ts +78 -0
- package/src/server/routes/instance/httpapi/errors.ts +193 -0
- package/src/server/routes/instance/httpapi/groups/config.ts +65 -0
- package/src/server/routes/instance/httpapi/groups/control-plane.ts +35 -0
- package/src/server/routes/instance/httpapi/groups/control.ts +76 -0
- package/src/server/routes/instance/httpapi/groups/event.ts +29 -0
- package/src/server/routes/instance/httpapi/groups/experimental.ts +260 -0
- package/src/server/routes/instance/httpapi/groups/file.ts +185 -0
- package/src/server/routes/instance/httpapi/groups/global.ts +138 -0
- package/src/server/routes/instance/httpapi/groups/instance.ts +206 -0
- package/src/server/routes/instance/httpapi/groups/mcp.ts +156 -0
- package/src/server/routes/instance/httpapi/groups/metadata.ts +18 -0
- package/src/server/routes/instance/httpapi/groups/permission.ts +61 -0
- package/src/server/routes/instance/httpapi/groups/project-copy.ts +32 -0
- package/src/server/routes/instance/httpapi/groups/project.ts +93 -0
- package/src/server/routes/instance/httpapi/groups/provider.ts +101 -0
- package/src/server/routes/instance/httpapi/groups/pty.ts +172 -0
- package/src/server/routes/instance/httpapi/groups/query.ts +12 -0
- package/src/server/routes/instance/httpapi/groups/question.ts +74 -0
- package/src/server/routes/instance/httpapi/groups/session.ts +462 -0
- package/src/server/routes/instance/httpapi/groups/sync.ts +113 -0
- package/src/server/routes/instance/httpapi/groups/tui.ts +208 -0
- package/src/server/routes/instance/httpapi/groups/workspace.ts +141 -0
- package/src/server/routes/instance/httpapi/handlers/config.ts +34 -0
- package/src/server/routes/instance/httpapi/handlers/control-plane.ts +37 -0
- package/src/server/routes/instance/httpapi/handlers/control.ts +43 -0
- package/src/server/routes/instance/httpapi/handlers/event.ts +99 -0
- package/src/server/routes/instance/httpapi/handlers/experimental.ts +187 -0
- package/src/server/routes/instance/httpapi/handlers/file.ts +139 -0
- package/src/server/routes/instance/httpapi/handlers/global.ts +156 -0
- package/src/server/routes/instance/httpapi/handlers/instance.ts +110 -0
- package/src/server/routes/instance/httpapi/handlers/mcp.ts +111 -0
- package/src/server/routes/instance/httpapi/handlers/permission.ts +41 -0
- package/src/server/routes/instance/httpapi/handlers/project-copy.ts +83 -0
- package/src/server/routes/instance/httpapi/handlers/project.ts +63 -0
- package/src/server/routes/instance/httpapi/handlers/provider.ts +113 -0
- package/src/server/routes/instance/httpapi/handlers/pty.ts +273 -0
- package/src/server/routes/instance/httpapi/handlers/question.ts +54 -0
- package/src/server/routes/instance/httpapi/handlers/session-errors.ts +21 -0
- package/src/server/routes/instance/httpapi/handlers/session.ts +440 -0
- package/src/server/routes/instance/httpapi/handlers/sync.ts +89 -0
- package/src/server/routes/instance/httpapi/handlers/tui.ts +131 -0
- package/src/server/routes/instance/httpapi/handlers/workspace.ts +102 -0
- package/src/server/routes/instance/httpapi/lifecycle.ts +54 -0
- package/src/server/routes/instance/httpapi/middleware/authorization.ts +150 -0
- package/src/server/routes/instance/httpapi/middleware/compression.ts +64 -0
- package/src/server/routes/instance/httpapi/middleware/cors-vary.ts +29 -0
- package/src/server/routes/instance/httpapi/middleware/error.ts +43 -0
- package/src/server/routes/instance/httpapi/middleware/fence.ts +25 -0
- package/src/server/routes/instance/httpapi/middleware/instance-context.ts +43 -0
- package/src/server/routes/instance/httpapi/middleware/proxy.ts +108 -0
- package/src/server/routes/instance/httpapi/middleware/schema-error.ts +41 -0
- package/src/server/routes/instance/httpapi/middleware/workspace-routing.ts +250 -0
- package/src/server/routes/instance/httpapi/public.ts +535 -0
- package/src/server/routes/instance/httpapi/server.ts +298 -0
- package/src/server/routes/instance/httpapi/websocket-tracker.ts +57 -0
- package/src/server/server.ts +225 -0
- package/src/server/shared/fence.ts +60 -0
- package/src/server/shared/pty-ticket.ts +15 -0
- package/src/server/shared/public-ui.ts +12 -0
- package/src/server/shared/tui-control.ts +28 -0
- package/src/server/shared/ui.ts +122 -0
- package/src/server/shared/workspace-routing.ts +38 -0
- package/src/server/tui-event.ts +53 -0
- package/src/session/compaction.ts +620 -0
- package/src/session/instruction.ts +241 -0
- package/src/session/llm/AGENTS.md +90 -0
- package/src/session/llm/ai-sdk.ts +288 -0
- package/src/session/llm/native-request.ts +196 -0
- package/src/session/llm/native-runtime.ts +196 -0
- package/src/session/llm/request.ts +218 -0
- package/src/session/llm.ts +415 -0
- package/src/session/message-error.ts +14 -0
- package/src/session/message-v2.ts +744 -0
- package/src/session/message.ts +148 -0
- package/src/session/overflow.ts +34 -0
- package/src/session/processor.ts +1084 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/beast.txt +147 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex.txt +79 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/default.txt +95 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/gpt.txt +107 -0
- package/src/session/prompt/kimi.txt +95 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-mode.txt +70 -0
- package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
- package/src/session/prompt/plan.txt +26 -0
- package/src/session/prompt/trinity.txt +97 -0
- package/src/session/prompt.ts +1722 -0
- package/src/session/reminders.ts +92 -0
- package/src/session/retry.ts +201 -0
- package/src/session/revert.ts +160 -0
- package/src/session/run-state.ts +156 -0
- package/src/session/schema.ts +26 -0
- package/src/session/session.ts +1119 -0
- package/src/session/status.ts +97 -0
- package/src/session/summary.ts +165 -0
- package/src/session/system.ts +117 -0
- package/src/session/todo.ts +90 -0
- package/src/session/tools.ts +207 -0
- package/src/share/session.ts +61 -0
- package/src/share/share-next.ts +385 -0
- package/src/skill/discovery.ts +109 -0
- package/src/skill/index.ts +366 -0
- package/src/snapshot/index.ts +808 -0
- package/src/sql.d.ts +4 -0
- package/src/storage/schema.ts +5 -0
- package/src/storage/storage.ts +329 -0
- package/src/sync/README.md +179 -0
- package/src/sync/schema.ts +11 -0
- package/src/temporary.ts +31 -0
- package/src/tool/apply_patch.ts +313 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/edit.ts +737 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +49 -0
- package/src/tool/glob.ts +76 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +112 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +21 -0
- package/src/tool/json-schema.ts +164 -0
- package/src/tool/lsp.ts +113 -0
- package/src/tool/lsp.txt +24 -0
- package/src/tool/mcp-websearch.ts +96 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +79 -0
- package/src/tool/question.ts +44 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +386 -0
- package/src/tool/read.txt +14 -0
- package/src/tool/registry.ts +440 -0
- package/src/tool/schema.ts +14 -0
- package/src/tool/shell/id.ts +19 -0
- package/src/tool/shell/prompt.ts +307 -0
- package/src/tool/shell/shell.txt +21 -0
- package/src/tool/shell.ts +657 -0
- package/src/tool/skill.ts +71 -0
- package/src/tool/skill.txt +5 -0
- package/src/tool/task.ts +346 -0
- package/src/tool/task.txt +19 -0
- package/src/tool/todo.ts +57 -0
- package/src/tool/todowrite.txt +44 -0
- package/src/tool/tool.ts +183 -0
- package/src/tool/truncate.ts +158 -0
- package/src/tool/truncation-dir.ts +4 -0
- package/src/tool/webfetch.ts +192 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +143 -0
- package/src/tool/websearch.txt +14 -0
- package/src/tool/write.ts +104 -0
- package/src/tool/write.txt +8 -0
- package/src/util/archive.ts +17 -0
- package/src/util/bom.ts +27 -0
- package/src/util/data-url.ts +9 -0
- package/src/util/defer.ts +10 -0
- package/src/util/effect-http-client.ts +11 -0
- package/src/util/error.ts +1 -0
- package/src/util/filesystem.ts +251 -0
- package/src/util/html.ts +8 -0
- package/src/util/iife.ts +3 -0
- package/src/util/lazy.ts +20 -0
- package/src/util/local-context.ts +25 -0
- package/src/util/locale.ts +2 -0
- package/src/util/media.ts +26 -0
- package/src/util/process.ts +177 -0
- package/src/util/proxy-env.ts +72 -0
- package/src/util/queue.ts +32 -0
- package/src/util/record.ts +1 -0
- package/src/util/repository.ts +232 -0
- package/src/util/rpc.ts +66 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +13 -0
- package/src/util/token.ts +1 -0
- package/src/util/wildcard.ts +59 -0
- package/src/worktree/index.ts +654 -0
|
@@ -0,0 +1,1489 @@
|
|
|
1
|
+
// Per-tool display rules shared across `rimuru-ai run` output paths.
|
|
2
|
+
//
|
|
3
|
+
// Each known tool (bash, edit, write, task, etc.) has a ToolRule that controls
|
|
4
|
+
// five display hooks:
|
|
5
|
+
//
|
|
6
|
+
// view → visibility policy for progress/final scrollback entries and
|
|
7
|
+
// whether completed finals can render as structured snapshots
|
|
8
|
+
// run → inline summary for the non-interactive `run` command output
|
|
9
|
+
// scroll → text formatting for start/progress/final scrollback entries
|
|
10
|
+
// permission → display info for the permission UI (icon, title, diff)
|
|
11
|
+
// snap → structured snapshot (code block, diff, task card) for rich
|
|
12
|
+
// scrollback entries
|
|
13
|
+
//
|
|
14
|
+
// Tools not in TOOL_RULES get fallback formatting.
|
|
15
|
+
import os from "os"
|
|
16
|
+
import path from "path"
|
|
17
|
+
import stripAnsi from "strip-ansi"
|
|
18
|
+
import type { ToolPart } from "@rimurucode-ai/sdk/v2"
|
|
19
|
+
import type * as Tool from "@/tool/tool"
|
|
20
|
+
import type { ApplyPatchTool } from "@/tool/apply_patch"
|
|
21
|
+
import type { ShellTool as BashTool } from "@/tool/shell"
|
|
22
|
+
import type { EditTool } from "@/tool/edit"
|
|
23
|
+
import type { GlobTool } from "@/tool/glob"
|
|
24
|
+
import type { GrepTool } from "@/tool/grep"
|
|
25
|
+
import type { InvalidTool } from "@/tool/invalid"
|
|
26
|
+
import type { LspTool } from "@/tool/lsp"
|
|
27
|
+
import type { PlanExitTool } from "@/tool/plan"
|
|
28
|
+
import type { QuestionTool } from "@/tool/question"
|
|
29
|
+
import type { ReadTool } from "@/tool/read"
|
|
30
|
+
import type { SkillTool } from "@/tool/skill"
|
|
31
|
+
import type { TaskTool } from "@/tool/task"
|
|
32
|
+
import type { TodoWriteTool } from "@/tool/todo"
|
|
33
|
+
import type { WebFetchTool } from "@/tool/webfetch"
|
|
34
|
+
import { webSearchProviderLabel, type WebSearchTool } from "@/tool/websearch"
|
|
35
|
+
import type { WriteTool } from "@/tool/write"
|
|
36
|
+
import { LANGUAGE_EXTENSIONS } from "@/lsp/language"
|
|
37
|
+
import * as Locale from "@/util/locale"
|
|
38
|
+
import type { RunEntryBody, StreamCommit, ToolSnapshot } from "./types"
|
|
39
|
+
|
|
40
|
+
export type ToolView = {
|
|
41
|
+
output: boolean
|
|
42
|
+
final: boolean
|
|
43
|
+
snap?: "code" | "diff" | "structured"
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export type ToolPhase = "start" | "progress" | "final"
|
|
47
|
+
|
|
48
|
+
export type ToolDict = Record<string, unknown>
|
|
49
|
+
|
|
50
|
+
export type ToolFrame = {
|
|
51
|
+
raw: string
|
|
52
|
+
name: string
|
|
53
|
+
input: ToolDict
|
|
54
|
+
meta: ToolDict
|
|
55
|
+
state: ToolDict
|
|
56
|
+
status: string
|
|
57
|
+
error: string
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
export type ToolInline = {
|
|
61
|
+
icon: string
|
|
62
|
+
title: string
|
|
63
|
+
description?: string
|
|
64
|
+
mode?: "inline" | "block"
|
|
65
|
+
body?: string
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
export type ToolPermissionInfo = {
|
|
69
|
+
icon: string
|
|
70
|
+
title: string
|
|
71
|
+
lines: string[]
|
|
72
|
+
diff?: string
|
|
73
|
+
file?: string
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
export type ToolProps<T = Tool.Info> = {
|
|
77
|
+
input: Partial<Tool.InferParameters<T>>
|
|
78
|
+
metadata: Partial<Tool.InferMetadata<T>>
|
|
79
|
+
frame: ToolFrame
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
type ToolPermissionProps<T = Tool.Info> = {
|
|
83
|
+
input: Partial<Tool.InferParameters<T>>
|
|
84
|
+
metadata: Partial<Tool.InferMetadata<T>>
|
|
85
|
+
patterns: string[]
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
type ToolPermissionCtx = {
|
|
89
|
+
input: ToolDict
|
|
90
|
+
meta: ToolDict
|
|
91
|
+
patterns: string[]
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
type ToolDefs = {
|
|
95
|
+
invalid: typeof InvalidTool
|
|
96
|
+
bash: typeof BashTool
|
|
97
|
+
write: typeof WriteTool
|
|
98
|
+
edit: typeof EditTool
|
|
99
|
+
apply_patch: typeof ApplyPatchTool
|
|
100
|
+
batch: Tool.Info
|
|
101
|
+
task: typeof TaskTool
|
|
102
|
+
todowrite: typeof TodoWriteTool
|
|
103
|
+
question: typeof QuestionTool
|
|
104
|
+
read: typeof ReadTool
|
|
105
|
+
glob: typeof GlobTool
|
|
106
|
+
grep: typeof GrepTool
|
|
107
|
+
list: Tool.Info
|
|
108
|
+
lsp: typeof LspTool
|
|
109
|
+
webfetch: typeof WebFetchTool
|
|
110
|
+
websearch: typeof WebSearchTool
|
|
111
|
+
skill: typeof SkillTool
|
|
112
|
+
plan_exit: typeof PlanExitTool
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
type ToolName = keyof ToolDefs
|
|
116
|
+
|
|
117
|
+
type ToolRule<T = Tool.Info> = {
|
|
118
|
+
view: ToolView
|
|
119
|
+
run: (props: ToolProps<T>) => ToolInline
|
|
120
|
+
scroll?: Partial<Record<ToolPhase, (props: ToolProps<T>) => string>>
|
|
121
|
+
permission?: (props: ToolPermissionProps<T>) => ToolPermissionInfo
|
|
122
|
+
snap?: (props: ToolProps<T>) => ToolSnapshot | undefined
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
type ToolRegistry = {
|
|
126
|
+
[K in ToolName]: ToolRule<ToolDefs[K]>
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
type AnyToolRule = ToolRule
|
|
130
|
+
|
|
131
|
+
function dict(v: unknown): ToolDict {
|
|
132
|
+
if (!v || typeof v !== "object" || Array.isArray(v)) {
|
|
133
|
+
return {}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
return { ...v }
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
function props<T = Tool.Info>(frame: ToolFrame): ToolProps<T> {
|
|
140
|
+
return {
|
|
141
|
+
input: Object.assign(Object.create(null), frame.input),
|
|
142
|
+
metadata: Object.assign(Object.create(null), frame.meta),
|
|
143
|
+
frame,
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
function permission<T = Tool.Info>(ctx: ToolPermissionCtx): ToolPermissionProps<T> {
|
|
148
|
+
return {
|
|
149
|
+
input: Object.assign(Object.create(null), ctx.input),
|
|
150
|
+
metadata: Object.assign(Object.create(null), ctx.meta),
|
|
151
|
+
patterns: ctx.patterns,
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
function text(v: unknown): string {
|
|
156
|
+
return typeof v === "string" ? v : ""
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
function num(v: unknown): number | undefined {
|
|
160
|
+
if (typeof v !== "number" || !Number.isFinite(v)) {
|
|
161
|
+
return undefined
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
return v
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
function list<T>(v: unknown): T[] {
|
|
168
|
+
if (!Array.isArray(v)) {
|
|
169
|
+
return []
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
return v
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
function info(data: ToolDict, skip: string[] = []): string {
|
|
176
|
+
const list = Object.entries(data).filter(([key, val]) => {
|
|
177
|
+
if (skip.includes(key)) {
|
|
178
|
+
return false
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
return typeof val === "string" || typeof val === "number" || typeof val === "boolean"
|
|
182
|
+
})
|
|
183
|
+
|
|
184
|
+
if (list.length === 0) {
|
|
185
|
+
return ""
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
return `[${list.map(([key, val]) => `${key}=${String(val)}`).join(", ")}]`
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
function span(state: ToolDict): string {
|
|
192
|
+
const time = dict(state.time)
|
|
193
|
+
const start = num(time.start)
|
|
194
|
+
const end = num(time.end)
|
|
195
|
+
if (start === undefined || end === undefined || end <= start) {
|
|
196
|
+
return ""
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
return Locale.duration(end - start)
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
function fail(ctx: ToolFrame): string {
|
|
203
|
+
const error = toolError(ctx)
|
|
204
|
+
if (error) {
|
|
205
|
+
return `✖ ${ctx.name} failed: ${error}`
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
return `✖ ${ctx.name} failed`
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
function toolError(ctx: ToolFrame): string {
|
|
212
|
+
if (ctx.error) {
|
|
213
|
+
return ctx.error
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
const state = text(ctx.state.error).trim()
|
|
217
|
+
if (state) {
|
|
218
|
+
return state
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
return ctx.raw.trim()
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
function fallbackStart(ctx: ToolFrame): string {
|
|
225
|
+
const extra = info(ctx.input)
|
|
226
|
+
if (!extra) {
|
|
227
|
+
return `⚙ ${ctx.name}`
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
return `⚙ ${ctx.name} ${extra}`
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
function fallbackFinal(ctx: ToolFrame): string {
|
|
234
|
+
if (ctx.status === "error") {
|
|
235
|
+
return fail(ctx)
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
if (ctx.status && ctx.status !== "completed") {
|
|
239
|
+
return ctx.raw.trim()
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
const time = span(ctx.state)
|
|
243
|
+
if (!time) {
|
|
244
|
+
return `${ctx.name} completed`
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
return `${ctx.name} completed · ${time}`
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
export function toolPath(input?: string, opts: { home?: boolean } = {}): string {
|
|
251
|
+
if (!input) {
|
|
252
|
+
return ""
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
const cwd = process.cwd()
|
|
256
|
+
const home = os.homedir()
|
|
257
|
+
const abs = path.isAbsolute(input) ? input : path.resolve(cwd, input)
|
|
258
|
+
const rel = path.relative(cwd, abs)
|
|
259
|
+
|
|
260
|
+
if (!rel) {
|
|
261
|
+
return "."
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
if (!rel.startsWith("..")) {
|
|
265
|
+
return rel.replaceAll("\\", "/")
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
if (opts.home && home && (abs === home || abs.startsWith(home + path.sep))) {
|
|
269
|
+
return abs.replace(home, "~").replaceAll("\\", "/")
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
return abs.replaceAll("\\", "/")
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
function fallbackInline(ctx: ToolFrame): ToolInline {
|
|
276
|
+
const title = text(ctx.state.title) || (Object.keys(ctx.input).length > 0 ? JSON.stringify(ctx.input) : "Unknown")
|
|
277
|
+
|
|
278
|
+
return {
|
|
279
|
+
icon: "⚙",
|
|
280
|
+
title: `${ctx.name} ${title}`,
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
function count(n: number, label: string): string {
|
|
285
|
+
return `${n} ${label}${n === 1 ? "" : "es"}`
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
function runGlob(p: ToolProps<typeof GlobTool>): ToolInline {
|
|
289
|
+
const root = p.input.path ?? ""
|
|
290
|
+
const title = `Glob "${p.input.pattern ?? ""}"`
|
|
291
|
+
const suffix = root ? `in ${toolPath(root)}` : ""
|
|
292
|
+
const matches = p.metadata.count
|
|
293
|
+
const description = matches === undefined ? suffix : `${suffix}${suffix ? " · " : ""}${count(matches, "match")}`
|
|
294
|
+
return {
|
|
295
|
+
icon: "✱",
|
|
296
|
+
title,
|
|
297
|
+
...(description && { description }),
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
function runGrep(p: ToolProps<typeof GrepTool>): ToolInline {
|
|
302
|
+
const root = p.input.path ?? ""
|
|
303
|
+
const title = `Grep "${p.input.pattern ?? ""}"`
|
|
304
|
+
const suffix = root ? `in ${toolPath(root)}` : ""
|
|
305
|
+
const matches = p.metadata.matches
|
|
306
|
+
const description = matches === undefined ? suffix : `${suffix}${suffix ? " · " : ""}${count(matches, "match")}`
|
|
307
|
+
return {
|
|
308
|
+
icon: "✱",
|
|
309
|
+
title,
|
|
310
|
+
...(description && { description }),
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
function runList(p: ToolProps): ToolInline {
|
|
315
|
+
const dir = text(dict(p.input).path)
|
|
316
|
+
return {
|
|
317
|
+
icon: "→",
|
|
318
|
+
title: dir ? `List ${toolPath(dir)}` : "List",
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
function runRead(p: ToolProps<typeof ReadTool>): ToolInline {
|
|
323
|
+
const file = toolPath(p.input.filePath)
|
|
324
|
+
const description = info(p.frame.input, ["filePath"]) || undefined
|
|
325
|
+
return {
|
|
326
|
+
icon: "→",
|
|
327
|
+
title: `Read ${file}`,
|
|
328
|
+
...(description && { description }),
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
function runWrite(p: ToolProps<typeof WriteTool>): ToolInline {
|
|
333
|
+
return {
|
|
334
|
+
icon: "←",
|
|
335
|
+
title: `Write ${toolPath(p.input.filePath)}`,
|
|
336
|
+
mode: "block",
|
|
337
|
+
body: p.frame.status === "completed" ? text(p.frame.state.output) : undefined,
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
function runWebfetch(p: ToolProps<typeof WebFetchTool>): ToolInline {
|
|
342
|
+
const url = p.input.url ?? ""
|
|
343
|
+
return {
|
|
344
|
+
icon: "%",
|
|
345
|
+
title: url ? `WebFetch ${url}` : "WebFetch",
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
function runEdit(p: ToolProps<typeof EditTool>): ToolInline {
|
|
350
|
+
return {
|
|
351
|
+
icon: "←",
|
|
352
|
+
title: `Edit ${toolPath(p.input.filePath)}`,
|
|
353
|
+
mode: "block",
|
|
354
|
+
body: p.metadata.diff,
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
function runWebSearch(p: ToolProps<typeof WebSearchTool>): ToolInline {
|
|
359
|
+
const title = webSearchProviderLabel(p.metadata.provider)
|
|
360
|
+
return {
|
|
361
|
+
icon: "◈",
|
|
362
|
+
title: p.input.query ? `${title} "${p.input.query}"` : title,
|
|
363
|
+
}
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
function runTask(p: ToolProps<typeof TaskTool>): ToolInline {
|
|
367
|
+
const kind = Locale.titlecase(p.input.subagent_type || "unknown")
|
|
368
|
+
const desc = p.input.description
|
|
369
|
+
const icon = p.frame.status === "error" ? "✗" : p.frame.status === "running" ? "•" : "✓"
|
|
370
|
+
return {
|
|
371
|
+
icon,
|
|
372
|
+
title: desc || `${kind} Task`,
|
|
373
|
+
description: desc ? `${kind} Agent` : undefined,
|
|
374
|
+
}
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
function runTodo(p: ToolProps<typeof TodoWriteTool>): ToolInline {
|
|
378
|
+
return {
|
|
379
|
+
icon: "#",
|
|
380
|
+
title: "Todos",
|
|
381
|
+
mode: "block",
|
|
382
|
+
body: list<{ status?: string; content?: string }>(p.frame.input.todos)
|
|
383
|
+
.flatMap((item) => {
|
|
384
|
+
const body = typeof item?.content === "string" ? item.content : ""
|
|
385
|
+
if (!body) {
|
|
386
|
+
return []
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
const mark = item.status === "completed" ? "[✓]" : item.status === "in_progress" ? "[•]" : "[ ]"
|
|
390
|
+
return [`${mark} ${body}`]
|
|
391
|
+
})
|
|
392
|
+
.join("\n"),
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
function runSkill(p: ToolProps<typeof SkillTool>): ToolInline {
|
|
397
|
+
return {
|
|
398
|
+
icon: "→",
|
|
399
|
+
title: `Skill "${p.input.name ?? ""}"`,
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
function runPatch(p: ToolProps<typeof ApplyPatchTool>): ToolInline {
|
|
404
|
+
const files = p.metadata.files?.length ?? 0
|
|
405
|
+
if (files === 0) {
|
|
406
|
+
return {
|
|
407
|
+
icon: "%",
|
|
408
|
+
title: "Patch",
|
|
409
|
+
}
|
|
410
|
+
}
|
|
411
|
+
|
|
412
|
+
return {
|
|
413
|
+
icon: "%",
|
|
414
|
+
title: `Patch ${files} file${files === 1 ? "" : "s"}`,
|
|
415
|
+
}
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
function runQuestion(p: ToolProps<typeof QuestionTool>): ToolInline {
|
|
419
|
+
const total = list(p.frame.input.questions).length
|
|
420
|
+
return {
|
|
421
|
+
icon: "→",
|
|
422
|
+
title: `Asked ${total} question${total === 1 ? "" : "s"}`,
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
function runInvalid(p: ToolProps<typeof InvalidTool>): ToolInline {
|
|
427
|
+
return {
|
|
428
|
+
icon: "✗",
|
|
429
|
+
title: text(p.frame.state.title) || "Invalid Tool",
|
|
430
|
+
mode: "block",
|
|
431
|
+
body: p.frame.status === "completed" ? text(p.frame.state.output) : undefined,
|
|
432
|
+
}
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
function runBatch(p: ToolProps): ToolInline {
|
|
436
|
+
const calls = list(dict(p.input).tool_calls).length
|
|
437
|
+
return {
|
|
438
|
+
icon: "#",
|
|
439
|
+
title: text(p.frame.state.title) || (calls > 0 ? `Batch ${calls} tool${calls === 1 ? "" : "s"}` : "Batch"),
|
|
440
|
+
mode: "block",
|
|
441
|
+
body: p.frame.status === "completed" ? text(p.frame.state.output) : undefined,
|
|
442
|
+
}
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
function lspTitle(
|
|
446
|
+
input: {
|
|
447
|
+
operation?: string
|
|
448
|
+
filePath?: string
|
|
449
|
+
line?: number
|
|
450
|
+
character?: number
|
|
451
|
+
},
|
|
452
|
+
opts: { home?: boolean } = {},
|
|
453
|
+
): string {
|
|
454
|
+
const op = input.operation || "request"
|
|
455
|
+
const file = input.filePath ? toolPath(input.filePath, opts) : ""
|
|
456
|
+
const line = typeof input.line === "number" ? input.line : undefined
|
|
457
|
+
const char = typeof input.character === "number" ? input.character : undefined
|
|
458
|
+
const pos = line !== undefined && char !== undefined ? `:${line}:${char}` : ""
|
|
459
|
+
if (!file) {
|
|
460
|
+
return `LSP ${op}`
|
|
461
|
+
}
|
|
462
|
+
|
|
463
|
+
return `LSP ${op} ${file}${pos}`
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
function runLsp(p: ToolProps<typeof LspTool>): ToolInline {
|
|
467
|
+
return {
|
|
468
|
+
icon: "→",
|
|
469
|
+
title: text(p.frame.state.title) || lspTitle(p.input),
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
function runPlanExit(p: ToolProps<typeof PlanExitTool>): ToolInline {
|
|
474
|
+
return {
|
|
475
|
+
icon: "→",
|
|
476
|
+
title: text(p.frame.state.title) || "Switching to build agent",
|
|
477
|
+
mode: "block",
|
|
478
|
+
body: p.frame.status === "completed" ? text(p.frame.state.output) : undefined,
|
|
479
|
+
}
|
|
480
|
+
}
|
|
481
|
+
|
|
482
|
+
type PatchFile = Tool.InferMetadata<typeof ApplyPatchTool>["files"][number]
|
|
483
|
+
|
|
484
|
+
function patchTitle(file: PatchFile): string {
|
|
485
|
+
const rel = file.relativePath
|
|
486
|
+
const from = file.filePath
|
|
487
|
+
if (file.type === "add") {
|
|
488
|
+
return `# Created ${rel || toolPath(from)}`
|
|
489
|
+
}
|
|
490
|
+
if (file.type === "delete") {
|
|
491
|
+
return `# Deleted ${rel || toolPath(from)}`
|
|
492
|
+
}
|
|
493
|
+
if (file.type === "move") {
|
|
494
|
+
return `# Moved ${toolPath(from)} -> ${rel || toolPath(file.movePath)}`
|
|
495
|
+
}
|
|
496
|
+
|
|
497
|
+
return `# Patched ${rel || toolPath(from)}`
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
function snapWrite(p: ToolProps<typeof WriteTool>): ToolSnapshot | undefined {
|
|
501
|
+
const file = p.input.filePath || ""
|
|
502
|
+
const content = p.input.content || ""
|
|
503
|
+
if (!file && !content) {
|
|
504
|
+
return undefined
|
|
505
|
+
}
|
|
506
|
+
|
|
507
|
+
return {
|
|
508
|
+
kind: "code",
|
|
509
|
+
title: `# Wrote ${toolPath(file)}`,
|
|
510
|
+
content,
|
|
511
|
+
file,
|
|
512
|
+
}
|
|
513
|
+
}
|
|
514
|
+
|
|
515
|
+
function snapEdit(p: ToolProps<typeof EditTool>): ToolSnapshot | undefined {
|
|
516
|
+
const file = p.input.filePath || ""
|
|
517
|
+
const diff = p.metadata.diff || ""
|
|
518
|
+
if (!file || !diff.trim()) {
|
|
519
|
+
return undefined
|
|
520
|
+
}
|
|
521
|
+
|
|
522
|
+
return {
|
|
523
|
+
kind: "diff",
|
|
524
|
+
items: [
|
|
525
|
+
{
|
|
526
|
+
title: `# Edited ${toolPath(file)}`,
|
|
527
|
+
diff,
|
|
528
|
+
file,
|
|
529
|
+
},
|
|
530
|
+
],
|
|
531
|
+
}
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
function snapPatch(p: ToolProps<typeof ApplyPatchTool>): ToolSnapshot | undefined {
|
|
535
|
+
const files = list<PatchFile>(p.frame.meta.files)
|
|
536
|
+
if (files.length === 0) {
|
|
537
|
+
return undefined
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
const items = files.flatMap((file) => {
|
|
541
|
+
if (!file || typeof file !== "object") {
|
|
542
|
+
return []
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
const diff = typeof file.patch === "string" ? file.patch : ""
|
|
546
|
+
if (!diff.trim()) {
|
|
547
|
+
return []
|
|
548
|
+
}
|
|
549
|
+
|
|
550
|
+
const name = file.movePath || file.filePath || file.relativePath
|
|
551
|
+
return [
|
|
552
|
+
{
|
|
553
|
+
title: patchTitle(file),
|
|
554
|
+
diff,
|
|
555
|
+
file: name,
|
|
556
|
+
deletions: typeof file.deletions === "number" ? file.deletions : 0,
|
|
557
|
+
},
|
|
558
|
+
]
|
|
559
|
+
})
|
|
560
|
+
|
|
561
|
+
if (items.length === 0) {
|
|
562
|
+
return undefined
|
|
563
|
+
}
|
|
564
|
+
|
|
565
|
+
return {
|
|
566
|
+
kind: "diff",
|
|
567
|
+
items,
|
|
568
|
+
}
|
|
569
|
+
}
|
|
570
|
+
|
|
571
|
+
function snapTask(p: ToolProps<typeof TaskTool>): ToolSnapshot {
|
|
572
|
+
const kind = Locale.titlecase(p.input.subagent_type || "general")
|
|
573
|
+
const desc = p.input.description
|
|
574
|
+
const title = text(p.frame.state.title)
|
|
575
|
+
const rows = [desc || title].filter((item): item is string => Boolean(item))
|
|
576
|
+
|
|
577
|
+
return {
|
|
578
|
+
kind: "task",
|
|
579
|
+
title: `# ${kind} Task`,
|
|
580
|
+
rows,
|
|
581
|
+
tail: "",
|
|
582
|
+
}
|
|
583
|
+
}
|
|
584
|
+
|
|
585
|
+
function snapTodo(p: ToolProps<typeof TodoWriteTool>): ToolSnapshot {
|
|
586
|
+
const items = list<{ status?: string; content?: string }>(p.frame.input.todos).flatMap((item) => {
|
|
587
|
+
const content = typeof item?.content === "string" ? item.content : ""
|
|
588
|
+
if (!content) {
|
|
589
|
+
return []
|
|
590
|
+
}
|
|
591
|
+
|
|
592
|
+
return [
|
|
593
|
+
{
|
|
594
|
+
status: typeof item.status === "string" ? item.status : "",
|
|
595
|
+
content,
|
|
596
|
+
},
|
|
597
|
+
]
|
|
598
|
+
})
|
|
599
|
+
|
|
600
|
+
return {
|
|
601
|
+
kind: "todo",
|
|
602
|
+
items,
|
|
603
|
+
tail: "",
|
|
604
|
+
}
|
|
605
|
+
}
|
|
606
|
+
|
|
607
|
+
function snapQuestion(p: ToolProps<typeof QuestionTool>): ToolSnapshot {
|
|
608
|
+
const answers = list<unknown[]>(p.frame.meta.answers)
|
|
609
|
+
const items = list<{ question?: string }>(p.frame.input.questions).map((item, i) => {
|
|
610
|
+
const answer = list<string>(answers[i]).filter((entry) => typeof entry === "string")
|
|
611
|
+
return {
|
|
612
|
+
question: item.question || `Question ${i + 1}`,
|
|
613
|
+
answer: answer.length > 0 ? answer.join(", ") : "(no answer)",
|
|
614
|
+
}
|
|
615
|
+
})
|
|
616
|
+
|
|
617
|
+
return {
|
|
618
|
+
kind: "question",
|
|
619
|
+
items,
|
|
620
|
+
tail: "",
|
|
621
|
+
}
|
|
622
|
+
}
|
|
623
|
+
|
|
624
|
+
function scrollBashStart(p: ToolProps<typeof BashTool>): string {
|
|
625
|
+
const cmd = p.input.command ?? ""
|
|
626
|
+
const desc = p.input.description || "Shell"
|
|
627
|
+
const wd = p.input.workdir ?? ""
|
|
628
|
+
const dir = wd && wd !== "." ? toolPath(wd) : ""
|
|
629
|
+
if (cmd && desc === "Shell" && !dir) {
|
|
630
|
+
return `$ ${cmd}`
|
|
631
|
+
}
|
|
632
|
+
|
|
633
|
+
const title = dir && !desc.includes(dir) ? `${desc} in ${dir}` : desc
|
|
634
|
+
|
|
635
|
+
if (!cmd) {
|
|
636
|
+
return `# ${title}`
|
|
637
|
+
}
|
|
638
|
+
|
|
639
|
+
return `# ${title}\n$ ${cmd}`
|
|
640
|
+
}
|
|
641
|
+
|
|
642
|
+
function scrollBashProgress(p: ToolProps<typeof BashTool>): string {
|
|
643
|
+
const out = stripAnsi(p.frame.raw)
|
|
644
|
+
const cmd = (p.input.command ?? "").trim()
|
|
645
|
+
const fmt = (text: string) => {
|
|
646
|
+
const body = text.replace(/^\n+/, "").replace(/\n+$/, "")
|
|
647
|
+
return body ? `\n${body}` : ""
|
|
648
|
+
}
|
|
649
|
+
|
|
650
|
+
if (!cmd) {
|
|
651
|
+
return out.replace(/\n+$/, "")
|
|
652
|
+
}
|
|
653
|
+
|
|
654
|
+
const wdRaw = (p.input.workdir ?? "").trim()
|
|
655
|
+
const wd = wdRaw ? toolPath(wdRaw) : ""
|
|
656
|
+
const lines = out.split("\n")
|
|
657
|
+
const first = (lines[0] || "").trim()
|
|
658
|
+
const second = (lines[1] || "").trim()
|
|
659
|
+
|
|
660
|
+
if (wd && (first === wd || first === wdRaw) && second === cmd) {
|
|
661
|
+
return fmt(lines.slice(2).join("\n"))
|
|
662
|
+
}
|
|
663
|
+
|
|
664
|
+
if (first === cmd || first === `$ ${cmd}`) {
|
|
665
|
+
return fmt(lines.slice(1).join("\n"))
|
|
666
|
+
}
|
|
667
|
+
|
|
668
|
+
if (wd && (first === `${wd} ${cmd}` || first === `${wdRaw} ${cmd}`)) {
|
|
669
|
+
return fmt(lines.slice(1).join("\n"))
|
|
670
|
+
}
|
|
671
|
+
|
|
672
|
+
return fmt(out)
|
|
673
|
+
}
|
|
674
|
+
|
|
675
|
+
function scrollBashFinal(p: ToolProps<typeof BashTool>): string {
|
|
676
|
+
const code = p.metadata.exit ?? num(p.frame.meta.exitCode) ?? num(p.frame.meta.exit_code)
|
|
677
|
+
const time = span(p.frame.state)
|
|
678
|
+
if (code === undefined) {
|
|
679
|
+
if (!time) {
|
|
680
|
+
return "bash completed"
|
|
681
|
+
}
|
|
682
|
+
|
|
683
|
+
return `bash completed · ${time}`
|
|
684
|
+
}
|
|
685
|
+
|
|
686
|
+
return `bash completed (exit ${code})${time ? ` · ${time}` : ""}`
|
|
687
|
+
}
|
|
688
|
+
|
|
689
|
+
function scrollReadStart(p: ToolProps<typeof ReadTool>): string {
|
|
690
|
+
const file = toolPath(p.input.filePath)
|
|
691
|
+
const extra = info(p.frame.input, ["filePath"])
|
|
692
|
+
const tail = extra ? ` ${extra}` : ""
|
|
693
|
+
return `→ Read ${file}${tail}`.trim()
|
|
694
|
+
}
|
|
695
|
+
|
|
696
|
+
function scrollWriteStart(_: ToolProps<typeof WriteTool>): string {
|
|
697
|
+
return ""
|
|
698
|
+
}
|
|
699
|
+
|
|
700
|
+
function scrollEditStart(_: ToolProps<typeof EditTool>): string {
|
|
701
|
+
return ""
|
|
702
|
+
}
|
|
703
|
+
|
|
704
|
+
function scrollPatchStart(_: ToolProps<typeof ApplyPatchTool>): string {
|
|
705
|
+
return ""
|
|
706
|
+
}
|
|
707
|
+
|
|
708
|
+
function patchLine(file: PatchFile): string {
|
|
709
|
+
const type = file.type
|
|
710
|
+
const rel = file.relativePath
|
|
711
|
+
const from = file.filePath
|
|
712
|
+
|
|
713
|
+
if (type === "add") {
|
|
714
|
+
return `+ Created ${rel || toolPath(from)}`
|
|
715
|
+
}
|
|
716
|
+
|
|
717
|
+
if (type === "delete") {
|
|
718
|
+
return `- Deleted ${rel || toolPath(from)}`
|
|
719
|
+
}
|
|
720
|
+
|
|
721
|
+
if (type === "move") {
|
|
722
|
+
return `→ Moved ${toolPath(from)} → ${rel || toolPath(file.movePath)}`
|
|
723
|
+
}
|
|
724
|
+
|
|
725
|
+
return `~ Patched ${rel || toolPath(from)}`
|
|
726
|
+
}
|
|
727
|
+
|
|
728
|
+
function scrollPatchFinal(p: ToolProps<typeof ApplyPatchTool>): string {
|
|
729
|
+
if (p.frame.status === "error") {
|
|
730
|
+
return fail(p.frame)
|
|
731
|
+
}
|
|
732
|
+
|
|
733
|
+
const files = list<PatchFile>(p.frame.meta.files)
|
|
734
|
+
if (files.length === 0) {
|
|
735
|
+
const time = span(p.frame.state)
|
|
736
|
+
if (!time) {
|
|
737
|
+
return "patch"
|
|
738
|
+
}
|
|
739
|
+
|
|
740
|
+
return `patch · ${time}`
|
|
741
|
+
}
|
|
742
|
+
|
|
743
|
+
const show_updates = !files.some((file) => file?.type && file.type !== "update")
|
|
744
|
+
const shown = files.filter((file) => show_updates || file.type !== "update")
|
|
745
|
+
const rows = shown.slice(0, 6).map(patchLine)
|
|
746
|
+
if (shown.length > 6) {
|
|
747
|
+
rows.push(`... and ${shown.length - 6} more`)
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
if (rows.length > 0) {
|
|
751
|
+
return rows.join("\n")
|
|
752
|
+
}
|
|
753
|
+
|
|
754
|
+
return patchLine(files[0]!)
|
|
755
|
+
}
|
|
756
|
+
|
|
757
|
+
function scrollTaskStart(_: ToolProps<typeof TaskTool>): string {
|
|
758
|
+
return ""
|
|
759
|
+
}
|
|
760
|
+
|
|
761
|
+
function taskResult(output: string): string | undefined {
|
|
762
|
+
if (!output.trim()) {
|
|
763
|
+
return undefined
|
|
764
|
+
}
|
|
765
|
+
|
|
766
|
+
const match = output.match(/<task_result>\s*([\s\S]*?)\s*<\/task_result>/)
|
|
767
|
+
if (match) {
|
|
768
|
+
return match[1].trim() || undefined
|
|
769
|
+
}
|
|
770
|
+
|
|
771
|
+
const next = output
|
|
772
|
+
.split("\n")
|
|
773
|
+
.filter((line) => !line.startsWith("task_id:"))
|
|
774
|
+
.join("\n")
|
|
775
|
+
.trim()
|
|
776
|
+
return next || undefined
|
|
777
|
+
}
|
|
778
|
+
|
|
779
|
+
function scrollTaskFinal(p: ToolProps<typeof TaskTool>): string {
|
|
780
|
+
if (p.frame.status === "error") {
|
|
781
|
+
return fail(p.frame)
|
|
782
|
+
}
|
|
783
|
+
|
|
784
|
+
const kind = Locale.titlecase(p.input.subagent_type || "general")
|
|
785
|
+
const row = p.input.description || text(p.frame.state.title)
|
|
786
|
+
if (!row) {
|
|
787
|
+
return `# ${kind} Task`
|
|
788
|
+
}
|
|
789
|
+
|
|
790
|
+
return `# ${kind} Task\n${row}`
|
|
791
|
+
}
|
|
792
|
+
|
|
793
|
+
function scrollTodoStart(_: ToolProps<typeof TodoWriteTool>): string {
|
|
794
|
+
return ""
|
|
795
|
+
}
|
|
796
|
+
|
|
797
|
+
function scrollTodoFinal(p: ToolProps<typeof TodoWriteTool>): string {
|
|
798
|
+
const items = list<{ status?: string }>(p.input.todos)
|
|
799
|
+
const time = span(p.frame.state)
|
|
800
|
+
if (items.length === 0) {
|
|
801
|
+
if (!time) {
|
|
802
|
+
return "0 todos"
|
|
803
|
+
}
|
|
804
|
+
|
|
805
|
+
return `0 todos · ${time}`
|
|
806
|
+
}
|
|
807
|
+
|
|
808
|
+
const doneN = items.filter((item) => item.status === "completed").length
|
|
809
|
+
const runN = items.filter((item) => item.status === "in_progress").length
|
|
810
|
+
const left = items.length - doneN - runN
|
|
811
|
+
const tail = [`${items.length} total`]
|
|
812
|
+
if (doneN > 0) {
|
|
813
|
+
tail.push(`${doneN} done`)
|
|
814
|
+
}
|
|
815
|
+
if (runN > 0) {
|
|
816
|
+
tail.push(`${runN} active`)
|
|
817
|
+
}
|
|
818
|
+
if (left > 0) {
|
|
819
|
+
tail.push(`${left} pending`)
|
|
820
|
+
}
|
|
821
|
+
|
|
822
|
+
if (time) {
|
|
823
|
+
tail.push(time)
|
|
824
|
+
}
|
|
825
|
+
|
|
826
|
+
return tail.join(" · ")
|
|
827
|
+
}
|
|
828
|
+
|
|
829
|
+
function scrollQuestionStart(_: ToolProps<typeof QuestionTool>): string {
|
|
830
|
+
return ""
|
|
831
|
+
}
|
|
832
|
+
|
|
833
|
+
function scrollQuestionFinal(p: ToolProps<typeof QuestionTool>): string {
|
|
834
|
+
const q = p.input.questions ?? []
|
|
835
|
+
const a = p.metadata.answers ?? []
|
|
836
|
+
const time = span(p.frame.state)
|
|
837
|
+
if (q.length === 0) {
|
|
838
|
+
if (!time) {
|
|
839
|
+
return "0 questions"
|
|
840
|
+
}
|
|
841
|
+
|
|
842
|
+
return `0 questions · ${time}`
|
|
843
|
+
}
|
|
844
|
+
|
|
845
|
+
const rows: string[] = []
|
|
846
|
+
for (const [i, item] of q.slice(0, 4).entries()) {
|
|
847
|
+
const prompt = item.question
|
|
848
|
+
const reply = a[i] ?? []
|
|
849
|
+
rows.push(`? ${prompt || `Question ${i + 1}`}`)
|
|
850
|
+
rows.push(` ${reply.length > 0 ? reply.join(", ") : "(no answer)"}`)
|
|
851
|
+
}
|
|
852
|
+
|
|
853
|
+
if (q.length > 4) {
|
|
854
|
+
rows.push(`... and ${q.length - 4} more`)
|
|
855
|
+
}
|
|
856
|
+
|
|
857
|
+
return rows.join("\n")
|
|
858
|
+
}
|
|
859
|
+
|
|
860
|
+
function scrollLspStart(p: ToolProps<typeof LspTool>): string {
|
|
861
|
+
return `→ ${lspTitle(p.input)}`
|
|
862
|
+
}
|
|
863
|
+
|
|
864
|
+
function scrollSkillStart(p: ToolProps<typeof SkillTool>): string {
|
|
865
|
+
return `→ Skill "${p.input.name ?? ""}"`
|
|
866
|
+
}
|
|
867
|
+
|
|
868
|
+
function scrollGlobStart(p: ToolProps<typeof GlobTool>): string {
|
|
869
|
+
const pattern = p.input.pattern ?? ""
|
|
870
|
+
const head = pattern ? `✱ Glob "${pattern}"` : "✱ Glob"
|
|
871
|
+
const dir = p.input.path ?? ""
|
|
872
|
+
if (!dir) {
|
|
873
|
+
return head
|
|
874
|
+
}
|
|
875
|
+
|
|
876
|
+
return `${head} in ${toolPath(dir)}`
|
|
877
|
+
}
|
|
878
|
+
|
|
879
|
+
function scrollGlobFinal(p: ToolProps<typeof GlobTool>): string {
|
|
880
|
+
return toolError(p.frame) || fail(p.frame)
|
|
881
|
+
}
|
|
882
|
+
|
|
883
|
+
function scrollGrepStart(p: ToolProps<typeof GrepTool>): string {
|
|
884
|
+
const pattern = p.input.pattern ?? ""
|
|
885
|
+
const head = pattern ? `✱ Grep "${pattern}"` : "✱ Grep"
|
|
886
|
+
const dir = p.input.path ?? ""
|
|
887
|
+
if (!dir) {
|
|
888
|
+
return head
|
|
889
|
+
}
|
|
890
|
+
|
|
891
|
+
return `${head} in ${toolPath(dir)}`
|
|
892
|
+
}
|
|
893
|
+
|
|
894
|
+
function scrollListStart(p: ToolProps): string {
|
|
895
|
+
const dir = text(dict(p.input).path)
|
|
896
|
+
if (!dir) {
|
|
897
|
+
return "→ List"
|
|
898
|
+
}
|
|
899
|
+
|
|
900
|
+
return `→ List ${toolPath(dir)}`
|
|
901
|
+
}
|
|
902
|
+
|
|
903
|
+
function scrollWebfetchStart(p: ToolProps<typeof WebFetchTool>): string {
|
|
904
|
+
const url = p.input.url ?? ""
|
|
905
|
+
if (!url) {
|
|
906
|
+
return "% WebFetch"
|
|
907
|
+
}
|
|
908
|
+
|
|
909
|
+
return `% WebFetch ${url}`
|
|
910
|
+
}
|
|
911
|
+
|
|
912
|
+
function scrollWebSearchStart(p: ToolProps<typeof WebSearchTool>): string {
|
|
913
|
+
const title = webSearchProviderLabel(p.metadata.provider)
|
|
914
|
+
const query = p.input.query ?? ""
|
|
915
|
+
if (!query) {
|
|
916
|
+
return `◈ ${title}`
|
|
917
|
+
}
|
|
918
|
+
|
|
919
|
+
return `◈ ${title} "${query}"`
|
|
920
|
+
}
|
|
921
|
+
|
|
922
|
+
function permEdit(p: ToolPermissionProps<typeof EditTool>): ToolPermissionInfo {
|
|
923
|
+
const input = p.input as { filePath?: string; filepath?: string; diff?: string }
|
|
924
|
+
const file = input.filePath || input.filepath || p.patterns[0] || ""
|
|
925
|
+
return {
|
|
926
|
+
icon: "→",
|
|
927
|
+
title: `Edit ${toolPath(file, { home: true })}`,
|
|
928
|
+
lines: [],
|
|
929
|
+
diff: p.metadata.diff ?? input.diff,
|
|
930
|
+
file,
|
|
931
|
+
}
|
|
932
|
+
}
|
|
933
|
+
|
|
934
|
+
function permRead(p: ToolPermissionProps<typeof ReadTool>): ToolPermissionInfo {
|
|
935
|
+
const file = p.input.filePath || p.patterns[0] || ""
|
|
936
|
+
return {
|
|
937
|
+
icon: "→",
|
|
938
|
+
title: `Read ${toolPath(file, { home: true })}`,
|
|
939
|
+
lines: file ? [`Path: ${toolPath(file, { home: true })}`] : [],
|
|
940
|
+
}
|
|
941
|
+
}
|
|
942
|
+
|
|
943
|
+
function permGlob(p: ToolPermissionProps<typeof GlobTool>): ToolPermissionInfo {
|
|
944
|
+
const pattern = p.input.pattern || p.patterns[0] || ""
|
|
945
|
+
return {
|
|
946
|
+
icon: "✱",
|
|
947
|
+
title: `Glob "${pattern}"`,
|
|
948
|
+
lines: pattern ? [`Pattern: ${pattern}`] : [],
|
|
949
|
+
}
|
|
950
|
+
}
|
|
951
|
+
|
|
952
|
+
function permGrep(p: ToolPermissionProps<typeof GrepTool>): ToolPermissionInfo {
|
|
953
|
+
const pattern = p.input.pattern || p.patterns[0] || ""
|
|
954
|
+
return {
|
|
955
|
+
icon: "✱",
|
|
956
|
+
title: `Grep "${pattern}"`,
|
|
957
|
+
lines: pattern ? [`Pattern: ${pattern}`] : [],
|
|
958
|
+
}
|
|
959
|
+
}
|
|
960
|
+
|
|
961
|
+
function permList(p: ToolPermissionProps): ToolPermissionInfo {
|
|
962
|
+
const dir = text(dict(p.input).path) || p.patterns[0] || ""
|
|
963
|
+
return {
|
|
964
|
+
icon: "→",
|
|
965
|
+
title: `List ${toolPath(dir, { home: true })}`,
|
|
966
|
+
lines: dir ? [`Path: ${toolPath(dir, { home: true })}`] : [],
|
|
967
|
+
}
|
|
968
|
+
}
|
|
969
|
+
|
|
970
|
+
function permBash(p: ToolPermissionProps<typeof BashTool>): ToolPermissionInfo {
|
|
971
|
+
const title = p.input.description || "Shell command"
|
|
972
|
+
const cmd = p.input.command || ""
|
|
973
|
+
return {
|
|
974
|
+
icon: "#",
|
|
975
|
+
title,
|
|
976
|
+
lines: cmd ? [`$ ${cmd}`] : p.patterns.map((item) => `- ${item}`),
|
|
977
|
+
}
|
|
978
|
+
}
|
|
979
|
+
|
|
980
|
+
function permTask(p: ToolPermissionProps<typeof TaskTool>): ToolPermissionInfo {
|
|
981
|
+
const type = p.input.subagent_type || "general"
|
|
982
|
+
const desc = p.input.description
|
|
983
|
+
return {
|
|
984
|
+
icon: "#",
|
|
985
|
+
title: `${Locale.titlecase(type)} Task`,
|
|
986
|
+
lines: desc ? [`◉ ${desc}`] : [],
|
|
987
|
+
}
|
|
988
|
+
}
|
|
989
|
+
|
|
990
|
+
function permWebfetch(p: ToolPermissionProps<typeof WebFetchTool>): ToolPermissionInfo {
|
|
991
|
+
const url = p.input.url || ""
|
|
992
|
+
return {
|
|
993
|
+
icon: "%",
|
|
994
|
+
title: `WebFetch ${url}`,
|
|
995
|
+
lines: url ? [`URL: ${url}`] : [],
|
|
996
|
+
}
|
|
997
|
+
}
|
|
998
|
+
|
|
999
|
+
function permWebSearch(p: ToolPermissionProps<typeof WebSearchTool>): ToolPermissionInfo {
|
|
1000
|
+
const query = p.input.query || ""
|
|
1001
|
+
const title = webSearchProviderLabel(p.metadata.provider)
|
|
1002
|
+
return {
|
|
1003
|
+
icon: "◈",
|
|
1004
|
+
title: query ? `${title} "${query}"` : title,
|
|
1005
|
+
lines: query ? [`Query: ${query}`] : [],
|
|
1006
|
+
}
|
|
1007
|
+
}
|
|
1008
|
+
|
|
1009
|
+
function permLsp(p: ToolPermissionProps<typeof LspTool>): ToolPermissionInfo {
|
|
1010
|
+
const file = p.input.filePath || ""
|
|
1011
|
+
const line = typeof p.input.line === "number" ? p.input.line : undefined
|
|
1012
|
+
const char = typeof p.input.character === "number" ? p.input.character : undefined
|
|
1013
|
+
const pos = line !== undefined && char !== undefined ? `${line}:${char}` : undefined
|
|
1014
|
+
return {
|
|
1015
|
+
icon: "→",
|
|
1016
|
+
title: lspTitle(p.input, { home: true }),
|
|
1017
|
+
lines: [
|
|
1018
|
+
...(p.input.operation ? [`Operation: ${p.input.operation}`] : []),
|
|
1019
|
+
...(file ? [`Path: ${toolPath(file, { home: true })}`] : []),
|
|
1020
|
+
...(pos ? [`Position: ${pos}`] : []),
|
|
1021
|
+
],
|
|
1022
|
+
}
|
|
1023
|
+
}
|
|
1024
|
+
|
|
1025
|
+
const TOOL_RULES = {
|
|
1026
|
+
invalid: {
|
|
1027
|
+
view: {
|
|
1028
|
+
output: true,
|
|
1029
|
+
final: false,
|
|
1030
|
+
},
|
|
1031
|
+
run: runInvalid,
|
|
1032
|
+
scroll: {
|
|
1033
|
+
start: () => "",
|
|
1034
|
+
},
|
|
1035
|
+
},
|
|
1036
|
+
bash: {
|
|
1037
|
+
view: {
|
|
1038
|
+
output: true,
|
|
1039
|
+
final: false,
|
|
1040
|
+
},
|
|
1041
|
+
run: runBash,
|
|
1042
|
+
scroll: {
|
|
1043
|
+
start: scrollBashStart,
|
|
1044
|
+
progress: scrollBashProgress,
|
|
1045
|
+
final: scrollBashFinal,
|
|
1046
|
+
},
|
|
1047
|
+
permission: permBash,
|
|
1048
|
+
},
|
|
1049
|
+
write: {
|
|
1050
|
+
view: {
|
|
1051
|
+
output: false,
|
|
1052
|
+
final: true,
|
|
1053
|
+
snap: "code",
|
|
1054
|
+
},
|
|
1055
|
+
run: runWrite,
|
|
1056
|
+
snap: snapWrite,
|
|
1057
|
+
scroll: {
|
|
1058
|
+
start: scrollWriteStart,
|
|
1059
|
+
},
|
|
1060
|
+
},
|
|
1061
|
+
edit: {
|
|
1062
|
+
view: {
|
|
1063
|
+
output: false,
|
|
1064
|
+
final: true,
|
|
1065
|
+
snap: "diff",
|
|
1066
|
+
},
|
|
1067
|
+
run: runEdit,
|
|
1068
|
+
snap: snapEdit,
|
|
1069
|
+
scroll: {
|
|
1070
|
+
start: scrollEditStart,
|
|
1071
|
+
},
|
|
1072
|
+
permission: permEdit,
|
|
1073
|
+
},
|
|
1074
|
+
apply_patch: {
|
|
1075
|
+
view: {
|
|
1076
|
+
output: false,
|
|
1077
|
+
final: true,
|
|
1078
|
+
snap: "diff",
|
|
1079
|
+
},
|
|
1080
|
+
run: runPatch,
|
|
1081
|
+
snap: snapPatch,
|
|
1082
|
+
scroll: {
|
|
1083
|
+
start: scrollPatchStart,
|
|
1084
|
+
final: scrollPatchFinal,
|
|
1085
|
+
},
|
|
1086
|
+
},
|
|
1087
|
+
batch: {
|
|
1088
|
+
view: {
|
|
1089
|
+
output: true,
|
|
1090
|
+
final: false,
|
|
1091
|
+
},
|
|
1092
|
+
run: runBatch,
|
|
1093
|
+
scroll: {
|
|
1094
|
+
start: () => "",
|
|
1095
|
+
},
|
|
1096
|
+
},
|
|
1097
|
+
task: {
|
|
1098
|
+
view: {
|
|
1099
|
+
output: false,
|
|
1100
|
+
final: true,
|
|
1101
|
+
snap: "structured",
|
|
1102
|
+
},
|
|
1103
|
+
run: runTask,
|
|
1104
|
+
snap: snapTask,
|
|
1105
|
+
scroll: {
|
|
1106
|
+
start: scrollTaskStart,
|
|
1107
|
+
final: scrollTaskFinal,
|
|
1108
|
+
},
|
|
1109
|
+
permission: permTask,
|
|
1110
|
+
},
|
|
1111
|
+
todowrite: {
|
|
1112
|
+
view: {
|
|
1113
|
+
output: false,
|
|
1114
|
+
final: true,
|
|
1115
|
+
snap: "structured",
|
|
1116
|
+
},
|
|
1117
|
+
run: runTodo,
|
|
1118
|
+
snap: snapTodo,
|
|
1119
|
+
scroll: {
|
|
1120
|
+
start: scrollTodoStart,
|
|
1121
|
+
final: scrollTodoFinal,
|
|
1122
|
+
},
|
|
1123
|
+
},
|
|
1124
|
+
question: {
|
|
1125
|
+
view: {
|
|
1126
|
+
output: false,
|
|
1127
|
+
final: true,
|
|
1128
|
+
snap: "structured",
|
|
1129
|
+
},
|
|
1130
|
+
run: runQuestion,
|
|
1131
|
+
snap: snapQuestion,
|
|
1132
|
+
scroll: {
|
|
1133
|
+
start: scrollQuestionStart,
|
|
1134
|
+
final: scrollQuestionFinal,
|
|
1135
|
+
},
|
|
1136
|
+
},
|
|
1137
|
+
read: {
|
|
1138
|
+
view: {
|
|
1139
|
+
output: false,
|
|
1140
|
+
final: false,
|
|
1141
|
+
},
|
|
1142
|
+
run: runRead,
|
|
1143
|
+
scroll: {
|
|
1144
|
+
start: scrollReadStart,
|
|
1145
|
+
},
|
|
1146
|
+
permission: permRead,
|
|
1147
|
+
},
|
|
1148
|
+
glob: {
|
|
1149
|
+
view: {
|
|
1150
|
+
output: false,
|
|
1151
|
+
final: false,
|
|
1152
|
+
},
|
|
1153
|
+
run: runGlob,
|
|
1154
|
+
scroll: {
|
|
1155
|
+
start: scrollGlobStart,
|
|
1156
|
+
final: scrollGlobFinal,
|
|
1157
|
+
},
|
|
1158
|
+
permission: permGlob,
|
|
1159
|
+
},
|
|
1160
|
+
grep: {
|
|
1161
|
+
view: {
|
|
1162
|
+
output: false,
|
|
1163
|
+
final: false,
|
|
1164
|
+
},
|
|
1165
|
+
run: runGrep,
|
|
1166
|
+
scroll: {
|
|
1167
|
+
start: scrollGrepStart,
|
|
1168
|
+
},
|
|
1169
|
+
permission: permGrep,
|
|
1170
|
+
},
|
|
1171
|
+
list: {
|
|
1172
|
+
view: {
|
|
1173
|
+
output: false,
|
|
1174
|
+
final: false,
|
|
1175
|
+
},
|
|
1176
|
+
run: runList,
|
|
1177
|
+
scroll: {
|
|
1178
|
+
start: scrollListStart,
|
|
1179
|
+
},
|
|
1180
|
+
permission: permList,
|
|
1181
|
+
},
|
|
1182
|
+
lsp: {
|
|
1183
|
+
view: {
|
|
1184
|
+
output: false,
|
|
1185
|
+
final: false,
|
|
1186
|
+
},
|
|
1187
|
+
run: runLsp,
|
|
1188
|
+
scroll: {
|
|
1189
|
+
start: scrollLspStart,
|
|
1190
|
+
},
|
|
1191
|
+
permission: permLsp,
|
|
1192
|
+
},
|
|
1193
|
+
webfetch: {
|
|
1194
|
+
view: {
|
|
1195
|
+
output: false,
|
|
1196
|
+
final: false,
|
|
1197
|
+
},
|
|
1198
|
+
run: runWebfetch,
|
|
1199
|
+
scroll: {
|
|
1200
|
+
start: scrollWebfetchStart,
|
|
1201
|
+
},
|
|
1202
|
+
permission: permWebfetch,
|
|
1203
|
+
},
|
|
1204
|
+
websearch: {
|
|
1205
|
+
view: {
|
|
1206
|
+
output: false,
|
|
1207
|
+
final: false,
|
|
1208
|
+
},
|
|
1209
|
+
run: runWebSearch,
|
|
1210
|
+
scroll: {
|
|
1211
|
+
start: scrollWebSearchStart,
|
|
1212
|
+
},
|
|
1213
|
+
permission: permWebSearch,
|
|
1214
|
+
},
|
|
1215
|
+
skill: {
|
|
1216
|
+
view: {
|
|
1217
|
+
output: false,
|
|
1218
|
+
final: false,
|
|
1219
|
+
},
|
|
1220
|
+
run: runSkill,
|
|
1221
|
+
scroll: {
|
|
1222
|
+
start: scrollSkillStart,
|
|
1223
|
+
},
|
|
1224
|
+
},
|
|
1225
|
+
plan_exit: {
|
|
1226
|
+
view: {
|
|
1227
|
+
output: true,
|
|
1228
|
+
final: false,
|
|
1229
|
+
},
|
|
1230
|
+
run: runPlanExit,
|
|
1231
|
+
scroll: {
|
|
1232
|
+
start: () => "",
|
|
1233
|
+
},
|
|
1234
|
+
},
|
|
1235
|
+
} as const satisfies ToolRegistry
|
|
1236
|
+
|
|
1237
|
+
function key(name: string): name is ToolName {
|
|
1238
|
+
return Object.prototype.hasOwnProperty.call(TOOL_RULES, name)
|
|
1239
|
+
}
|
|
1240
|
+
|
|
1241
|
+
function rule(name?: string): AnyToolRule | undefined {
|
|
1242
|
+
if (!name || !key(name)) {
|
|
1243
|
+
return undefined
|
|
1244
|
+
}
|
|
1245
|
+
|
|
1246
|
+
return TOOL_RULES[name]
|
|
1247
|
+
}
|
|
1248
|
+
|
|
1249
|
+
function frame(part: ToolPart): ToolFrame {
|
|
1250
|
+
const state = dict(part.state)
|
|
1251
|
+
return {
|
|
1252
|
+
raw: "",
|
|
1253
|
+
name: part.tool,
|
|
1254
|
+
input: dict(state.input),
|
|
1255
|
+
meta: "metadata" in part.state ? dict(part.state.metadata) : {},
|
|
1256
|
+
state,
|
|
1257
|
+
status: text(state.status),
|
|
1258
|
+
error: text(state.error),
|
|
1259
|
+
}
|
|
1260
|
+
}
|
|
1261
|
+
|
|
1262
|
+
export function toolFrame(commit: StreamCommit, raw: string): ToolFrame {
|
|
1263
|
+
const state = dict(commit.part?.state)
|
|
1264
|
+
return {
|
|
1265
|
+
raw,
|
|
1266
|
+
name: commit.tool || commit.part?.tool || "tool",
|
|
1267
|
+
input: dict(state.input),
|
|
1268
|
+
meta: commit.part?.state && "metadata" in commit.part.state ? dict(commit.part.state.metadata) : {},
|
|
1269
|
+
state,
|
|
1270
|
+
status: commit.toolState ?? text(state.status),
|
|
1271
|
+
error: (commit.toolError ?? "").trim(),
|
|
1272
|
+
}
|
|
1273
|
+
}
|
|
1274
|
+
|
|
1275
|
+
function runBash(p: ToolProps<typeof BashTool>): ToolInline {
|
|
1276
|
+
return {
|
|
1277
|
+
icon: "$",
|
|
1278
|
+
title: p.input.command || "",
|
|
1279
|
+
mode: "block",
|
|
1280
|
+
body: p.frame.status === "completed" ? text(p.frame.state.output).trim() : undefined,
|
|
1281
|
+
}
|
|
1282
|
+
}
|
|
1283
|
+
|
|
1284
|
+
export function toolView(name?: string): ToolView {
|
|
1285
|
+
return (
|
|
1286
|
+
rule(name)?.view ?? {
|
|
1287
|
+
output: true,
|
|
1288
|
+
final: true,
|
|
1289
|
+
}
|
|
1290
|
+
)
|
|
1291
|
+
}
|
|
1292
|
+
|
|
1293
|
+
export function toolStructuredFinal(commit: StreamCommit): boolean {
|
|
1294
|
+
const state = commit.toolState ?? commit.part?.state.status
|
|
1295
|
+
return (
|
|
1296
|
+
commit.kind === "tool" &&
|
|
1297
|
+
commit.phase === "final" &&
|
|
1298
|
+
state === "completed" &&
|
|
1299
|
+
Boolean(toolView(commit.tool ?? commit.part?.tool).snap)
|
|
1300
|
+
)
|
|
1301
|
+
}
|
|
1302
|
+
|
|
1303
|
+
export function toolInlineInfo(part: ToolPart): ToolInline {
|
|
1304
|
+
const ctx = frame(part)
|
|
1305
|
+
const draw = rule(ctx.name)?.run
|
|
1306
|
+
try {
|
|
1307
|
+
if (draw) {
|
|
1308
|
+
return draw(props(ctx))
|
|
1309
|
+
}
|
|
1310
|
+
} catch {
|
|
1311
|
+
return fallbackInline(ctx)
|
|
1312
|
+
}
|
|
1313
|
+
|
|
1314
|
+
return fallbackInline(ctx)
|
|
1315
|
+
}
|
|
1316
|
+
|
|
1317
|
+
export function toolScroll(phase: ToolPhase, ctx: ToolFrame): string {
|
|
1318
|
+
const draw = rule(ctx.name)?.scroll?.[phase]
|
|
1319
|
+
try {
|
|
1320
|
+
if (draw) {
|
|
1321
|
+
return draw(props(ctx))
|
|
1322
|
+
}
|
|
1323
|
+
} catch {
|
|
1324
|
+
if (phase === "start") {
|
|
1325
|
+
return fallbackStart(ctx)
|
|
1326
|
+
}
|
|
1327
|
+
if (phase === "progress") {
|
|
1328
|
+
return ctx.raw
|
|
1329
|
+
}
|
|
1330
|
+
return fallbackFinal(ctx)
|
|
1331
|
+
}
|
|
1332
|
+
|
|
1333
|
+
if (phase === "start") {
|
|
1334
|
+
return fallbackStart(ctx)
|
|
1335
|
+
}
|
|
1336
|
+
|
|
1337
|
+
if (phase === "progress") {
|
|
1338
|
+
return ctx.raw
|
|
1339
|
+
}
|
|
1340
|
+
|
|
1341
|
+
return fallbackFinal(ctx)
|
|
1342
|
+
}
|
|
1343
|
+
|
|
1344
|
+
export function toolPermissionInfo(
|
|
1345
|
+
name: string,
|
|
1346
|
+
input: ToolDict,
|
|
1347
|
+
meta: ToolDict,
|
|
1348
|
+
patterns: string[],
|
|
1349
|
+
): ToolPermissionInfo | undefined {
|
|
1350
|
+
const draw = rule(name)?.permission
|
|
1351
|
+
if (!draw) {
|
|
1352
|
+
return undefined
|
|
1353
|
+
}
|
|
1354
|
+
|
|
1355
|
+
try {
|
|
1356
|
+
return draw(permission({ input, meta, patterns }))
|
|
1357
|
+
} catch {
|
|
1358
|
+
return undefined
|
|
1359
|
+
}
|
|
1360
|
+
}
|
|
1361
|
+
|
|
1362
|
+
export function toolSnapshot(commit: StreamCommit, raw: string): ToolSnapshot | undefined {
|
|
1363
|
+
const ctx = toolFrame(commit, raw)
|
|
1364
|
+
const draw = rule(ctx.name)?.snap
|
|
1365
|
+
if (!draw) {
|
|
1366
|
+
return undefined
|
|
1367
|
+
}
|
|
1368
|
+
|
|
1369
|
+
try {
|
|
1370
|
+
return draw(props(ctx))
|
|
1371
|
+
} catch {
|
|
1372
|
+
return undefined
|
|
1373
|
+
}
|
|
1374
|
+
}
|
|
1375
|
+
|
|
1376
|
+
function textBody(content: string): RunEntryBody | undefined {
|
|
1377
|
+
if (!content) {
|
|
1378
|
+
return undefined
|
|
1379
|
+
}
|
|
1380
|
+
|
|
1381
|
+
return {
|
|
1382
|
+
type: "text",
|
|
1383
|
+
content,
|
|
1384
|
+
}
|
|
1385
|
+
}
|
|
1386
|
+
|
|
1387
|
+
function markdownBody(content: string): RunEntryBody | undefined {
|
|
1388
|
+
if (!content) {
|
|
1389
|
+
return undefined
|
|
1390
|
+
}
|
|
1391
|
+
|
|
1392
|
+
return {
|
|
1393
|
+
type: "markdown",
|
|
1394
|
+
content,
|
|
1395
|
+
}
|
|
1396
|
+
}
|
|
1397
|
+
|
|
1398
|
+
function structuredBody(commit: StreamCommit, raw: string): RunEntryBody | undefined {
|
|
1399
|
+
const snap = toolSnapshot(commit, raw)
|
|
1400
|
+
if (!snap) {
|
|
1401
|
+
return undefined
|
|
1402
|
+
}
|
|
1403
|
+
|
|
1404
|
+
return {
|
|
1405
|
+
type: "structured",
|
|
1406
|
+
snapshot: snap,
|
|
1407
|
+
}
|
|
1408
|
+
}
|
|
1409
|
+
|
|
1410
|
+
function shellOutput(command: string, raw: string): string | undefined {
|
|
1411
|
+
const body = stripAnsi(raw).replace(/^\n+/, "").replace(/\n+$/, "")
|
|
1412
|
+
if (!body) {
|
|
1413
|
+
return undefined
|
|
1414
|
+
}
|
|
1415
|
+
|
|
1416
|
+
if (!command) {
|
|
1417
|
+
return body
|
|
1418
|
+
}
|
|
1419
|
+
|
|
1420
|
+
return `\n${body}`
|
|
1421
|
+
}
|
|
1422
|
+
|
|
1423
|
+
export function toolEntryBody(commit: StreamCommit, raw: string): RunEntryBody | undefined {
|
|
1424
|
+
if (commit.shell) {
|
|
1425
|
+
if (commit.phase === "start") {
|
|
1426
|
+
return textBody(`$ ${commit.shell.command}`)
|
|
1427
|
+
}
|
|
1428
|
+
|
|
1429
|
+
if (commit.phase === "progress") {
|
|
1430
|
+
return textBody(shellOutput(commit.shell.command, raw) ?? "")
|
|
1431
|
+
}
|
|
1432
|
+
|
|
1433
|
+
return undefined
|
|
1434
|
+
}
|
|
1435
|
+
|
|
1436
|
+
const ctx = toolFrame(commit, raw)
|
|
1437
|
+
const view = toolView(ctx.name)
|
|
1438
|
+
|
|
1439
|
+
if (ctx.name === "task") {
|
|
1440
|
+
if (commit.phase === "start") {
|
|
1441
|
+
return undefined
|
|
1442
|
+
}
|
|
1443
|
+
|
|
1444
|
+
if (commit.phase === "final" && ctx.status === "completed") {
|
|
1445
|
+
const result = taskResult(text(ctx.state.output))
|
|
1446
|
+
if (result) {
|
|
1447
|
+
return markdownBody(result)
|
|
1448
|
+
}
|
|
1449
|
+
}
|
|
1450
|
+
}
|
|
1451
|
+
|
|
1452
|
+
if (commit.phase === "progress" && !view.output) {
|
|
1453
|
+
return undefined
|
|
1454
|
+
}
|
|
1455
|
+
|
|
1456
|
+
if (commit.phase === "final") {
|
|
1457
|
+
if (ctx.status === "error") {
|
|
1458
|
+
return textBody(toolScroll("final", ctx))
|
|
1459
|
+
}
|
|
1460
|
+
|
|
1461
|
+
if (!view.final) {
|
|
1462
|
+
return undefined
|
|
1463
|
+
}
|
|
1464
|
+
|
|
1465
|
+
if (ctx.status && ctx.status !== "completed") {
|
|
1466
|
+
return textBody(ctx.raw.trim())
|
|
1467
|
+
}
|
|
1468
|
+
|
|
1469
|
+
if (toolStructuredFinal(commit)) {
|
|
1470
|
+
return structuredBody(commit, raw) ?? textBody(toolScroll("final", ctx))
|
|
1471
|
+
}
|
|
1472
|
+
}
|
|
1473
|
+
|
|
1474
|
+
return textBody(toolScroll(commit.phase, ctx))
|
|
1475
|
+
}
|
|
1476
|
+
|
|
1477
|
+
export function toolFiletype(input?: string): string | undefined {
|
|
1478
|
+
if (!input) {
|
|
1479
|
+
return undefined
|
|
1480
|
+
}
|
|
1481
|
+
|
|
1482
|
+
const ext = path.extname(input)
|
|
1483
|
+
const lang = LANGUAGE_EXTENSIONS[ext]
|
|
1484
|
+
if (["typescriptreact", "javascriptreact", "javascript"].includes(lang)) {
|
|
1485
|
+
return "typescript"
|
|
1486
|
+
}
|
|
1487
|
+
|
|
1488
|
+
return lang
|
|
1489
|
+
}
|