opencode-v2 1.1.53
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/AGENTS.md +27 -0
- package/Dockerfile +18 -0
- package/README.md +15 -0
- package/bin/opencode +84 -0
- package/bunfig.toml +5 -0
- package/package.json +126 -0
- package/parsers-config.ts +253 -0
- package/script/build.ts +193 -0
- package/script/postinstall.mjs +125 -0
- package/script/publish.ts +181 -0
- package/script/schema.ts +47 -0
- package/script/seed-e2e.ts +50 -0
- package/src/acp/README.md +164 -0
- package/src/acp/agent.ts +1676 -0
- package/src/acp/session.ts +117 -0
- package/src/acp/types.ts +23 -0
- package/src/agent/agent.ts +414 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +12 -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/auth/index.ts +70 -0
- package/src/bun/index.ts +137 -0
- package/src/bun/registry.ts +48 -0
- package/src/bus/bus-event.ts +43 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +105 -0
- package/src/cli/bootstrap.ts +17 -0
- package/src/cli/cmd/acp.ts +70 -0
- package/src/cli/cmd/agent.ts +257 -0
- package/src/cli/cmd/auth.ts +400 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/debug/agent.ts +167 -0
- package/src/cli/cmd/debug/config.ts +16 -0
- package/src/cli/cmd/debug/file.ts +97 -0
- package/src/cli/cmd/debug/index.ts +48 -0
- package/src/cli/cmd/debug/lsp.ts +52 -0
- package/src/cli/cmd/debug/ripgrep.ts +87 -0
- package/src/cli/cmd/debug/scrap.ts +16 -0
- package/src/cli/cmd/debug/skill.ts +16 -0
- package/src/cli/cmd/debug/snapshot.ts +52 -0
- package/src/cli/cmd/export.ts +88 -0
- package/src/cli/cmd/generate.ts +38 -0
- package/src/cli/cmd/github.ts +1540 -0
- package/src/cli/cmd/import.ts +147 -0
- package/src/cli/cmd/mcp.ts +755 -0
- package/src/cli/cmd/models.ts +77 -0
- package/src/cli/cmd/pr.ts +112 -0
- package/src/cli/cmd/run.ts +617 -0
- package/src/cli/cmd/serve.ts +20 -0
- package/src/cli/cmd/session.ts +135 -0
- package/src/cli/cmd/stats.ts +426 -0
- package/src/cli/cmd/tui/app.tsx +801 -0
- package/src/cli/cmd/tui/attach.ts +52 -0
- package/src/cli/cmd/tui/component/border.tsx +21 -0
- package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
- package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
- package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
- package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
- package/src/cli/cmd/tui/component/logo.tsx +85 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/spinner.tsx +24 -0
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
- package/src/cli/cmd/tui/component/tips.tsx +153 -0
- package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
- package/src/cli/cmd/tui/context/args.tsx +15 -0
- package/src/cli/cmd/tui/context/directory.ts +13 -0
- package/src/cli/cmd/tui/context/exit.tsx +52 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +100 -0
- package/src/cli/cmd/tui/context/kv.tsx +52 -0
- package/src/cli/cmd/tui/context/local.tsx +409 -0
- package/src/cli/cmd/tui/context/prompt.tsx +18 -0
- package/src/cli/cmd/tui/context/route.tsx +46 -0
- package/src/cli/cmd/tui/context/sdk.tsx +101 -0
- package/src/cli/cmd/tui/context/sync.tsx +470 -0
- package/src/cli/cmd/tui/context/theme/aura.json +69 -0
- package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
- package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
- package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
- package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
- package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
- package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
- package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
- package/src/cli/cmd/tui/context/theme/github.json +233 -0
- package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
- package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
- package/src/cli/cmd/tui/context/theme/material.json +235 -0
- package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
- package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
- package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
- package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
- package/src/cli/cmd/tui/context/theme/nord.json +223 -0
- package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
- package/src/cli/cmd/tui/context/theme/orng.json +249 -0
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
- package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
- package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
- package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
- package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
- package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
- package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
- package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
- package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
- package/src/cli/cmd/tui/context/theme.tsx +1152 -0
- package/src/cli/cmd/tui/event.ts +48 -0
- package/src/cli/cmd/tui/routes/home.tsx +140 -0
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
- package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
- package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
- package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
- package/src/cli/cmd/tui/thread.ts +175 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
- package/src/cli/cmd/tui/ui/link.tsx +28 -0
- package/src/cli/cmd/tui/ui/spinner.ts +368 -0
- package/src/cli/cmd/tui/ui/toast.tsx +100 -0
- package/src/cli/cmd/tui/util/clipboard.ts +159 -0
- package/src/cli/cmd/tui/util/editor.ts +32 -0
- package/src/cli/cmd/tui/util/signal.ts +7 -0
- package/src/cli/cmd/tui/util/terminal.ts +114 -0
- package/src/cli/cmd/tui/util/transcript.ts +98 -0
- package/src/cli/cmd/tui/worker.ts +152 -0
- package/src/cli/cmd/uninstall.ts +357 -0
- package/src/cli/cmd/upgrade.ts +73 -0
- package/src/cli/cmd/web.ts +81 -0
- package/src/cli/error.ts +57 -0
- package/src/cli/logo.ts +6 -0
- package/src/cli/network.ts +60 -0
- package/src/cli/ui.ts +113 -0
- package/src/cli/upgrade.ts +25 -0
- package/src/command/index.ts +150 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/command/template/review.txt +99 -0
- package/src/config/config.ts +1477 -0
- package/src/config/markdown.ts +98 -0
- package/src/env/index.ts +28 -0
- package/src/file/ignore.ts +83 -0
- package/src/file/index.ts +583 -0
- package/src/file/ripgrep.ts +375 -0
- package/src/file/time.ts +69 -0
- package/src/file/watcher.ts +127 -0
- package/src/flag/flag.ts +97 -0
- package/src/format/formatter.ts +366 -0
- package/src/format/index.ts +137 -0
- package/src/global/index.ts +55 -0
- package/src/id/id.ts +83 -0
- package/src/ide/index.ts +76 -0
- package/src/index.ts +159 -0
- package/src/installation/index.ts +246 -0
- package/src/lsp/client.ts +252 -0
- package/src/lsp/index.ts +485 -0
- package/src/lsp/language.ts +119 -0
- package/src/lsp/server.ts +2046 -0
- package/src/mcp/auth.ts +132 -0
- package/src/mcp/index.ts +934 -0
- package/src/mcp/oauth-callback.ts +200 -0
- package/src/mcp/oauth-provider.ts +154 -0
- package/src/patch/index.ts +680 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/index.ts +210 -0
- package/src/permission/next.ts +280 -0
- package/src/plugin/codex.ts +624 -0
- package/src/plugin/copilot.ts +327 -0
- package/src/plugin/index.ts +138 -0
- package/src/project/bootstrap.ts +35 -0
- package/src/project/instance.ts +114 -0
- package/src/project/project.ts +371 -0
- package/src/project/state.ts +70 -0
- package/src/project/vcs.ts +76 -0
- package/src/provider/auth.ts +147 -0
- package/src/provider/models.ts +133 -0
- package/src/provider/provider.ts +1262 -0
- package/src/provider/sdk/copilot/README.md +5 -0
- package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
- package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
- package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
- package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
- package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
- package/src/provider/sdk/copilot/index.ts +2 -0
- package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
- package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
- package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
- package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
- package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
- package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
- package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
- package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
- package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
- package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
- package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
- package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
- package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
- package/src/provider/transform.ts +828 -0
- package/src/pty/index.ts +250 -0
- package/src/question/index.ts +171 -0
- package/src/scheduler/index.ts +61 -0
- package/src/server/error.ts +36 -0
- package/src/server/event.ts +7 -0
- package/src/server/mdns.ts +60 -0
- package/src/server/routes/config.ts +92 -0
- package/src/server/routes/experimental.ts +208 -0
- package/src/server/routes/file.ts +197 -0
- package/src/server/routes/global.ts +183 -0
- package/src/server/routes/mcp.ts +225 -0
- package/src/server/routes/permission.ts +68 -0
- package/src/server/routes/project.ts +82 -0
- package/src/server/routes/provider.ts +165 -0
- package/src/server/routes/pty.ts +169 -0
- package/src/server/routes/question.ts +98 -0
- package/src/server/routes/session.ts +939 -0
- package/src/server/routes/tui.ts +379 -0
- package/src/server/server.ts +613 -0
- package/src/session/compaction.ts +226 -0
- package/src/session/index.ts +524 -0
- package/src/session/instruction.ts +197 -0
- package/src/session/llm.ts +289 -0
- package/src/session/message-v2.ts +802 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +407 -0
- package/src/session/prompt/agent.txt +43 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -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_header.txt +79 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
- package/src/session/prompt/plan.txt +26 -0
- package/src/session/prompt/qwen.txt +109 -0
- package/src/session/prompt/research.txt +81 -0
- package/src/session/prompt/trinity.txt +97 -0
- package/src/session/prompt.ts +1952 -0
- package/src/session/retry.ts +97 -0
- package/src/session/revert.ts +121 -0
- package/src/session/status.ts +76 -0
- package/src/session/summary.ts +217 -0
- package/src/session/system.ts +54 -0
- package/src/session/todo.ts +37 -0
- package/src/share/share-next.ts +200 -0
- package/src/share/share.ts +92 -0
- package/src/shell/shell.ts +67 -0
- package/src/skill/discovery.ts +97 -0
- package/src/skill/index.ts +1 -0
- package/src/skill/skill.ts +188 -0
- package/src/snapshot/index.ts +255 -0
- package/src/storage/storage.ts +227 -0
- package/src/tool/agent-enter.txt +1 -0
- package/src/tool/agent-exit.txt +1 -0
- package/src/tool/agent.ts +237 -0
- package/src/tool/apply_patch.ts +281 -0
- package/src/tool/apply_patch.txt +33 -0
- package/src/tool/bash.ts +269 -0
- package/src/tool/bash.txt +115 -0
- package/src/tool/batch.ts +175 -0
- package/src/tool/batch.txt +24 -0
- package/src/tool/chat-enter.txt +15 -0
- package/src/tool/chat-exit.txt +7 -0
- package/src/tool/chat.ts +217 -0
- package/src/tool/codesearch.ts +132 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +655 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/external-directory.ts +32 -0
- package/src/tool/glob.ts +78 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +147 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +17 -0
- package/src/tool/ls.ts +121 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/lsp.ts +96 -0
- package/src/tool/lsp.txt +19 -0
- package/src/tool/multiedit.ts +46 -0
- package/src/tool/multiedit.txt +41 -0
- package/src/tool/plan-enter.txt +14 -0
- package/src/tool/plan-exit.txt +13 -0
- package/src/tool/plan.ts +130 -0
- package/src/tool/question.ts +33 -0
- package/src/tool/question.txt +10 -0
- package/src/tool/read.ts +211 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +167 -0
- package/src/tool/research-enter.txt +1 -0
- package/src/tool/research-exit.txt +1 -0
- package/src/tool/research.ts +134 -0
- package/src/tool/skill.ts +123 -0
- package/src/tool/task.ts +165 -0
- package/src/tool/task.txt +60 -0
- package/src/tool/todo.ts +53 -0
- package/src/tool/todoread.txt +14 -0
- package/src/tool/todowrite.txt +167 -0
- package/src/tool/tool.ts +89 -0
- package/src/tool/truncation.ts +106 -0
- package/src/tool/webfetch.ts +186 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +150 -0
- package/src/tool/websearch.txt +14 -0
- package/src/tool/write.ts +85 -0
- package/src/tool/write.txt +8 -0
- package/src/util/abort.ts +35 -0
- package/src/util/archive.ts +16 -0
- package/src/util/color.ts +19 -0
- package/src/util/context.ts +25 -0
- package/src/util/defer.ts +12 -0
- package/src/util/eventloop.ts +20 -0
- package/src/util/filesystem.ts +93 -0
- package/src/util/fn.ts +11 -0
- package/src/util/format.ts +20 -0
- package/src/util/iife.ts +3 -0
- package/src/util/keybind.ts +103 -0
- package/src/util/lazy.ts +18 -0
- package/src/util/locale.ts +81 -0
- package/src/util/lock.ts +98 -0
- package/src/util/log.ts +180 -0
- package/src/util/proxied.ts +3 -0
- package/src/util/queue.ts +32 -0
- package/src/util/rpc.ts +66 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +7 -0
- package/src/util/wildcard.ts +56 -0
- package/src/worktree/index.ts +574 -0
- package/sst-env.d.ts +9 -0
- package/test/acp/agent-interface.test.ts +51 -0
- package/test/acp/event-subscription.test.ts +436 -0
- package/test/agent/agent.test.ts +675 -0
- package/test/bun.test.ts +53 -0
- package/test/cli/github-action.test.ts +161 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/cli/import.test.ts +38 -0
- package/test/cli/tui/transcript.test.ts +322 -0
- package/test/config/agent-color.test.ts +71 -0
- package/test/config/config.test.ts +1802 -0
- package/test/config/fixtures/empty-frontmatter.md +4 -0
- package/test/config/fixtures/frontmatter.md +28 -0
- package/test/config/fixtures/markdown-header.md +11 -0
- package/test/config/fixtures/no-frontmatter.md +1 -0
- package/test/config/fixtures/weird-model-id.md +13 -0
- package/test/config/markdown.test.ts +228 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/path-traversal.test.ts +198 -0
- package/test/file/ripgrep.test.ts +39 -0
- package/test/fixture/fixture.ts +45 -0
- package/test/fixture/lsp/fake-lsp-server.js +77 -0
- package/test/ide/ide.test.ts +82 -0
- package/test/keybind.test.ts +421 -0
- package/test/lsp/client.test.ts +95 -0
- package/test/mcp/headers.test.ts +153 -0
- package/test/mcp/oauth-browser.test.ts +249 -0
- package/test/memory/abort-leak.test.ts +136 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +690 -0
- package/test/permission-task.test.ts +319 -0
- package/test/plugin/auth-override.test.ts +44 -0
- package/test/plugin/codex.test.ts +123 -0
- package/test/preload.ts +63 -0
- package/test/project/project.test.ts +120 -0
- package/test/provider/amazon-bedrock.test.ts +445 -0
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
- package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
- package/test/provider/gitlab-duo.test.ts +262 -0
- package/test/provider/provider.test.ts +2129 -0
- package/test/provider/transform.test.ts +2022 -0
- package/test/question/question.test.ts +300 -0
- package/test/scheduler.test.ts +73 -0
- package/test/server/session-list.test.ts +39 -0
- package/test/server/session-select.test.ts +78 -0
- package/test/session/compaction.test.ts +293 -0
- package/test/session/instruction.test.ts +170 -0
- package/test/session/llm.test.ts +691 -0
- package/test/session/message-v2.test.ts +786 -0
- package/test/session/prompt-missing-file.test.ts +53 -0
- package/test/session/prompt-special-chars.test.ts +56 -0
- package/test/session/prompt-variant.test.ts +60 -0
- package/test/session/retry.test.ts +179 -0
- package/test/session/revert-compact.test.ts +285 -0
- package/test/session/session.test.ts +71 -0
- package/test/skill/discovery.test.ts +60 -0
- package/test/skill/skill.test.ts +388 -0
- package/test/snapshot/snapshot.test.ts +1040 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/apply_patch.test.ts +559 -0
- package/test/tool/bash.test.ts +399 -0
- package/test/tool/external-directory.test.ts +127 -0
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +38413 -0
- package/test/tool/grep.test.ts +110 -0
- package/test/tool/question.test.ts +107 -0
- package/test/tool/read.test.ts +358 -0
- package/test/tool/registry.test.ts +122 -0
- package/test/tool/skill.test.ts +112 -0
- package/test/tool/truncation.test.ts +159 -0
- package/test/util/filesystem.test.ts +39 -0
- package/test/util/format.test.ts +59 -0
- package/test/util/iife.test.ts +36 -0
- package/test/util/lazy.test.ts +50 -0
- package/test/util/lock.test.ts +72 -0
- package/test/util/timeout.test.ts +21 -0
- package/test/util/wildcard.test.ts +75 -0
- package/tsconfig.json +16 -0
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
import z from "zod"
|
|
2
|
+
import path from "path"
|
|
3
|
+
import { Tool } from "./tool"
|
|
4
|
+
import { Question } from "../question"
|
|
5
|
+
import { Session } from "../session"
|
|
6
|
+
import { MessageV2 } from "../session/message-v2"
|
|
7
|
+
import { Identifier } from "../id/id"
|
|
8
|
+
import { Provider } from "../provider/provider"
|
|
9
|
+
import { Instance } from "../project/instance"
|
|
10
|
+
import { PermissionNext } from "@/permission/next"
|
|
11
|
+
import EXIT_DESCRIPTION from "./agent-exit.txt"
|
|
12
|
+
import ENTER_DESCRIPTION from "./agent-enter.txt"
|
|
13
|
+
|
|
14
|
+
async function getLastModel(sessionID: string) {
|
|
15
|
+
for await (const item of MessageV2.stream(sessionID)) {
|
|
16
|
+
if (item.info.role === "user" && item.info.model) return item.info.model
|
|
17
|
+
}
|
|
18
|
+
return Provider.defaultModel()
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export const AgentExitTool = Tool.define("agent_exit", {
|
|
22
|
+
description: EXIT_DESCRIPTION,
|
|
23
|
+
parameters: z.object({}),
|
|
24
|
+
async execute(_params, ctx) {
|
|
25
|
+
const session = await Session.get(ctx.sessionID)
|
|
26
|
+
const plan = path.relative(Instance.worktree, Session.plan(session))
|
|
27
|
+
|
|
28
|
+
const answers = await Question.ask({
|
|
29
|
+
sessionID: ctx.sessionID,
|
|
30
|
+
questions: [
|
|
31
|
+
{
|
|
32
|
+
question: "Would you like to exit agent mode and switch to a different mode?",
|
|
33
|
+
header: "Exit Agent Mode",
|
|
34
|
+
custom: false,
|
|
35
|
+
options: [
|
|
36
|
+
{ label: "Chat", description: "Switch to chat agent for general conversation" },
|
|
37
|
+
{ label: "Research", description: "Switch to research agent for web research" },
|
|
38
|
+
],
|
|
39
|
+
},
|
|
40
|
+
],
|
|
41
|
+
tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
|
|
42
|
+
})
|
|
43
|
+
|
|
44
|
+
const answer = answers[0]?.[0]
|
|
45
|
+
if (!answer) throw new Question.RejectedError()
|
|
46
|
+
|
|
47
|
+
const targetAgent = answer.toLowerCase()
|
|
48
|
+
const model = await getLastModel(ctx.sessionID)
|
|
49
|
+
|
|
50
|
+
const userMsg: MessageV2.User = {
|
|
51
|
+
id: Identifier.ascending("message"),
|
|
52
|
+
sessionID: ctx.sessionID,
|
|
53
|
+
role: "user",
|
|
54
|
+
time: {
|
|
55
|
+
created: Date.now(),
|
|
56
|
+
},
|
|
57
|
+
agent: targetAgent,
|
|
58
|
+
model,
|
|
59
|
+
}
|
|
60
|
+
await Session.updateMessage(userMsg)
|
|
61
|
+
await Session.updatePart({
|
|
62
|
+
id: Identifier.ascending("part"),
|
|
63
|
+
messageID: userMsg.id,
|
|
64
|
+
sessionID: ctx.sessionID,
|
|
65
|
+
type: "text",
|
|
66
|
+
text: `User has requested to exit agent mode and switch to ${targetAgent} mode.`,
|
|
67
|
+
synthetic: true,
|
|
68
|
+
} satisfies MessageV2.TextPart)
|
|
69
|
+
|
|
70
|
+
return {
|
|
71
|
+
title: `Switching to ${targetAgent} agent`,
|
|
72
|
+
output: `User confirmed to switch to ${targetAgent} mode.`,
|
|
73
|
+
metadata: {},
|
|
74
|
+
}
|
|
75
|
+
},
|
|
76
|
+
})
|
|
77
|
+
|
|
78
|
+
export const AgentEnterTool = Tool.define("agent_enter", {
|
|
79
|
+
description: ENTER_DESCRIPTION,
|
|
80
|
+
parameters: z.object({}),
|
|
81
|
+
async execute(_params, ctx) {
|
|
82
|
+
const session = await Session.get(ctx.sessionID)
|
|
83
|
+
const messages = await Session.messages({ sessionID: session.id })
|
|
84
|
+
|
|
85
|
+
const hasAssistantMessages = messages.some((msg) => msg.info.role === "assistant")
|
|
86
|
+
|
|
87
|
+
if (hasAssistantMessages) {
|
|
88
|
+
return {
|
|
89
|
+
title: "Cannot enter agent mode",
|
|
90
|
+
output: "Agent mode can only be entered when starting a new conversation, not in an existing thread.",
|
|
91
|
+
metadata: {},
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
const answers = await Question.ask({
|
|
96
|
+
sessionID: ctx.sessionID,
|
|
97
|
+
questions: [
|
|
98
|
+
{
|
|
99
|
+
question: "Would you like to switch to agent mode? Agent mode combines planning and execution - you plan first, then approve execution.",
|
|
100
|
+
header: "Agent Mode",
|
|
101
|
+
custom: false,
|
|
102
|
+
options: [
|
|
103
|
+
{ label: "Yes", description: "Switch to agent mode for unified planning and execution" },
|
|
104
|
+
{ label: "No", description: "Stay with current agent" },
|
|
105
|
+
],
|
|
106
|
+
},
|
|
107
|
+
],
|
|
108
|
+
tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
|
|
109
|
+
})
|
|
110
|
+
|
|
111
|
+
const answer = answers[0]?.[0]
|
|
112
|
+
|
|
113
|
+
if (answer === "No") throw new Question.RejectedError()
|
|
114
|
+
|
|
115
|
+
const model = await getLastModel(ctx.sessionID)
|
|
116
|
+
|
|
117
|
+
const userMsg: MessageV2.User = {
|
|
118
|
+
id: Identifier.ascending("message"),
|
|
119
|
+
sessionID: ctx.sessionID,
|
|
120
|
+
role: "user",
|
|
121
|
+
time: {
|
|
122
|
+
created: Date.now(),
|
|
123
|
+
},
|
|
124
|
+
agent: "agent",
|
|
125
|
+
model,
|
|
126
|
+
}
|
|
127
|
+
await Session.updateMessage(userMsg)
|
|
128
|
+
await Session.updatePart({
|
|
129
|
+
id: Identifier.ascending("part"),
|
|
130
|
+
messageID: userMsg.id,
|
|
131
|
+
sessionID: ctx.sessionID,
|
|
132
|
+
type: "text",
|
|
133
|
+
text: "User has requested to enter agent mode. Begin with planning phase - understand the request, ask clarifying questions, then create a plan for user approval.",
|
|
134
|
+
synthetic: true,
|
|
135
|
+
} satisfies MessageV2.TextPart)
|
|
136
|
+
|
|
137
|
+
return {
|
|
138
|
+
title: "Switching to agent mode",
|
|
139
|
+
output: "User confirmed to switch to agent mode. Begin with the planning workflow.",
|
|
140
|
+
metadata: {},
|
|
141
|
+
}
|
|
142
|
+
},
|
|
143
|
+
})
|
|
144
|
+
|
|
145
|
+
export const AgentApprovePlanTool = Tool.define("agent_approve_plan", {
|
|
146
|
+
description: "Approve the plan and switch from planning phase to execution phase. This grants full file system access for implementation.",
|
|
147
|
+
parameters: z.object({}),
|
|
148
|
+
async execute(_params, ctx) {
|
|
149
|
+
const session = await Session.get(ctx.sessionID)
|
|
150
|
+
const plan = path.relative(Instance.worktree, Session.plan(session))
|
|
151
|
+
|
|
152
|
+
const answers = await Question.ask({
|
|
153
|
+
sessionID: ctx.sessionID,
|
|
154
|
+
questions: [
|
|
155
|
+
{
|
|
156
|
+
question: `Approve the plan at ${plan} and switch to execution mode? You'll have full file system access after approval.`,
|
|
157
|
+
header: "Approve Plan",
|
|
158
|
+
custom: false,
|
|
159
|
+
options: [
|
|
160
|
+
{ label: "Yes", description: "Approve plan and grant full execution permissions" },
|
|
161
|
+
{ label: "No", description: "Continue refining the plan" },
|
|
162
|
+
],
|
|
163
|
+
},
|
|
164
|
+
],
|
|
165
|
+
tool: ctx.callID ? { messageID: ctx.messageID, callID: ctx.callID } : undefined,
|
|
166
|
+
})
|
|
167
|
+
|
|
168
|
+
const answer = answers[0]?.[0]
|
|
169
|
+
if (answer === "No") throw new Question.RejectedError()
|
|
170
|
+
|
|
171
|
+
const model = await getLastModel(ctx.sessionID)
|
|
172
|
+
|
|
173
|
+
const userMsg: MessageV2.User = {
|
|
174
|
+
id: Identifier.ascending("message"),
|
|
175
|
+
sessionID: ctx.sessionID,
|
|
176
|
+
role: "user",
|
|
177
|
+
time: {
|
|
178
|
+
created: Date.now(),
|
|
179
|
+
},
|
|
180
|
+
agent: "agent",
|
|
181
|
+
model,
|
|
182
|
+
}
|
|
183
|
+
await Session.updateMessage(userMsg)
|
|
184
|
+
await Session.updatePart({
|
|
185
|
+
id: Identifier.ascending("part"),
|
|
186
|
+
messageID: userMsg.id,
|
|
187
|
+
sessionID: ctx.sessionID,
|
|
188
|
+
type: "text",
|
|
189
|
+
text: `Plan at ${plan} has been approved. You now have full file system access. Create a todo list and begin implementation.`,
|
|
190
|
+
synthetic: true,
|
|
191
|
+
} satisfies MessageV2.TextPart)
|
|
192
|
+
|
|
193
|
+
return {
|
|
194
|
+
title: "Plan approved - execution mode enabled",
|
|
195
|
+
output: `Plan at ${plan} approved. You now have full file system access. Begin by creating a todo list from the plan.`,
|
|
196
|
+
metadata: {},
|
|
197
|
+
}
|
|
198
|
+
},
|
|
199
|
+
})
|
|
200
|
+
|
|
201
|
+
export const AgentDenyPlanTool = Tool.define("agent_deny_plan", {
|
|
202
|
+
description: "Deny the current plan and return to planning phase. Use this to request changes to the plan before execution.",
|
|
203
|
+
parameters: z.object({
|
|
204
|
+
reason: z.string().optional().describe("Reason for denying the plan"),
|
|
205
|
+
}),
|
|
206
|
+
async execute(params, ctx) {
|
|
207
|
+
const session = await Session.get(ctx.sessionID)
|
|
208
|
+
|
|
209
|
+
const model = await getLastModel(ctx.sessionID)
|
|
210
|
+
|
|
211
|
+
const userMsg: MessageV2.User = {
|
|
212
|
+
id: Identifier.ascending("message"),
|
|
213
|
+
sessionID: ctx.sessionID,
|
|
214
|
+
role: "user",
|
|
215
|
+
time: {
|
|
216
|
+
created: Date.now(),
|
|
217
|
+
},
|
|
218
|
+
agent: "agent",
|
|
219
|
+
model,
|
|
220
|
+
}
|
|
221
|
+
await Session.updateMessage(userMsg)
|
|
222
|
+
await Session.updatePart({
|
|
223
|
+
id: Identifier.ascending("part"),
|
|
224
|
+
messageID: userMsg.id,
|
|
225
|
+
sessionID: ctx.sessionID,
|
|
226
|
+
type: "text",
|
|
227
|
+
text: `Plan has been denied. Reason: ${params.reason ?? "No reason provided"}. Continue planning and revise the plan based on feedback.`,
|
|
228
|
+
synthetic: true,
|
|
229
|
+
} satisfies MessageV2.TextPart)
|
|
230
|
+
|
|
231
|
+
return {
|
|
232
|
+
title: "Plan denied - returning to planning",
|
|
233
|
+
output: "Continue refining the plan based on user feedback.",
|
|
234
|
+
metadata: {},
|
|
235
|
+
}
|
|
236
|
+
},
|
|
237
|
+
})
|
|
@@ -0,0 +1,281 @@
|
|
|
1
|
+
import z from "zod"
|
|
2
|
+
import * as path from "path"
|
|
3
|
+
import * as fs from "fs/promises"
|
|
4
|
+
import { Tool } from "./tool"
|
|
5
|
+
import { Bus } from "../bus"
|
|
6
|
+
import { FileWatcher } from "../file/watcher"
|
|
7
|
+
import { Instance } from "../project/instance"
|
|
8
|
+
import { Patch } from "../patch"
|
|
9
|
+
import { createTwoFilesPatch, diffLines } from "diff"
|
|
10
|
+
import { assertExternalDirectory } from "./external-directory"
|
|
11
|
+
import { trimDiff } from "./edit"
|
|
12
|
+
import { LSP } from "../lsp"
|
|
13
|
+
import { Filesystem } from "../util/filesystem"
|
|
14
|
+
import DESCRIPTION from "./apply_patch.txt"
|
|
15
|
+
import { File } from "../file"
|
|
16
|
+
|
|
17
|
+
const PatchParams = z.object({
|
|
18
|
+
patchText: z.string().describe("The full patch text that describes all changes to be made"),
|
|
19
|
+
})
|
|
20
|
+
|
|
21
|
+
export const ApplyPatchTool = Tool.define("apply_patch", {
|
|
22
|
+
description: DESCRIPTION,
|
|
23
|
+
parameters: PatchParams,
|
|
24
|
+
async execute(params, ctx) {
|
|
25
|
+
if (!params.patchText) {
|
|
26
|
+
throw new Error("patchText is required")
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
// Parse the patch to get hunks
|
|
30
|
+
let hunks: Patch.Hunk[]
|
|
31
|
+
try {
|
|
32
|
+
const parseResult = Patch.parsePatch(params.patchText)
|
|
33
|
+
hunks = parseResult.hunks
|
|
34
|
+
} catch (error) {
|
|
35
|
+
throw new Error(`apply_patch verification failed: ${error}`)
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
if (hunks.length === 0) {
|
|
39
|
+
const normalized = params.patchText.replace(/\r\n/g, "\n").replace(/\r/g, "\n").trim()
|
|
40
|
+
if (normalized === "*** Begin Patch\n*** End Patch") {
|
|
41
|
+
throw new Error("patch rejected: empty patch")
|
|
42
|
+
}
|
|
43
|
+
throw new Error("apply_patch verification failed: no hunks found")
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Validate file paths and check permissions
|
|
47
|
+
const fileChanges: Array<{
|
|
48
|
+
filePath: string
|
|
49
|
+
oldContent: string
|
|
50
|
+
newContent: string
|
|
51
|
+
type: "add" | "update" | "delete" | "move"
|
|
52
|
+
movePath?: string
|
|
53
|
+
diff: string
|
|
54
|
+
additions: number
|
|
55
|
+
deletions: number
|
|
56
|
+
}> = []
|
|
57
|
+
|
|
58
|
+
let totalDiff = ""
|
|
59
|
+
|
|
60
|
+
for (const hunk of hunks) {
|
|
61
|
+
const filePath = path.resolve(Instance.directory, hunk.path)
|
|
62
|
+
await assertExternalDirectory(ctx, filePath)
|
|
63
|
+
|
|
64
|
+
switch (hunk.type) {
|
|
65
|
+
case "add": {
|
|
66
|
+
const oldContent = ""
|
|
67
|
+
const newContent =
|
|
68
|
+
hunk.contents.length === 0 || hunk.contents.endsWith("\n") ? hunk.contents : `${hunk.contents}\n`
|
|
69
|
+
const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, newContent))
|
|
70
|
+
|
|
71
|
+
let additions = 0
|
|
72
|
+
let deletions = 0
|
|
73
|
+
for (const change of diffLines(oldContent, newContent)) {
|
|
74
|
+
if (change.added) additions += change.count || 0
|
|
75
|
+
if (change.removed) deletions += change.count || 0
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
fileChanges.push({
|
|
79
|
+
filePath,
|
|
80
|
+
oldContent,
|
|
81
|
+
newContent,
|
|
82
|
+
type: "add",
|
|
83
|
+
diff,
|
|
84
|
+
additions,
|
|
85
|
+
deletions,
|
|
86
|
+
})
|
|
87
|
+
|
|
88
|
+
totalDiff += diff + "\n"
|
|
89
|
+
break
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
case "update": {
|
|
93
|
+
// Check if file exists for update
|
|
94
|
+
const stats = await fs.stat(filePath).catch(() => null)
|
|
95
|
+
if (!stats || stats.isDirectory()) {
|
|
96
|
+
throw new Error(`apply_patch verification failed: Failed to read file to update: ${filePath}`)
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
const oldContent = await fs.readFile(filePath, "utf-8")
|
|
100
|
+
let newContent = oldContent
|
|
101
|
+
|
|
102
|
+
// Apply the update chunks to get new content
|
|
103
|
+
try {
|
|
104
|
+
const fileUpdate = Patch.deriveNewContentsFromChunks(filePath, hunk.chunks)
|
|
105
|
+
newContent = fileUpdate.content
|
|
106
|
+
} catch (error) {
|
|
107
|
+
throw new Error(`apply_patch verification failed: ${error}`)
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
const diff = trimDiff(createTwoFilesPatch(filePath, filePath, oldContent, newContent))
|
|
111
|
+
|
|
112
|
+
let additions = 0
|
|
113
|
+
let deletions = 0
|
|
114
|
+
for (const change of diffLines(oldContent, newContent)) {
|
|
115
|
+
if (change.added) additions += change.count || 0
|
|
116
|
+
if (change.removed) deletions += change.count || 0
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
const movePath = hunk.move_path ? path.resolve(Instance.directory, hunk.move_path) : undefined
|
|
120
|
+
await assertExternalDirectory(ctx, movePath)
|
|
121
|
+
|
|
122
|
+
fileChanges.push({
|
|
123
|
+
filePath,
|
|
124
|
+
oldContent,
|
|
125
|
+
newContent,
|
|
126
|
+
type: hunk.move_path ? "move" : "update",
|
|
127
|
+
movePath,
|
|
128
|
+
diff,
|
|
129
|
+
additions,
|
|
130
|
+
deletions,
|
|
131
|
+
})
|
|
132
|
+
|
|
133
|
+
totalDiff += diff + "\n"
|
|
134
|
+
break
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
case "delete": {
|
|
138
|
+
const contentToDelete = await fs.readFile(filePath, "utf-8").catch((error) => {
|
|
139
|
+
throw new Error(`apply_patch verification failed: ${error}`)
|
|
140
|
+
})
|
|
141
|
+
const deleteDiff = trimDiff(createTwoFilesPatch(filePath, filePath, contentToDelete, ""))
|
|
142
|
+
|
|
143
|
+
const deletions = contentToDelete.split("\n").length
|
|
144
|
+
|
|
145
|
+
fileChanges.push({
|
|
146
|
+
filePath,
|
|
147
|
+
oldContent: contentToDelete,
|
|
148
|
+
newContent: "",
|
|
149
|
+
type: "delete",
|
|
150
|
+
diff: deleteDiff,
|
|
151
|
+
additions: 0,
|
|
152
|
+
deletions,
|
|
153
|
+
})
|
|
154
|
+
|
|
155
|
+
totalDiff += deleteDiff + "\n"
|
|
156
|
+
break
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// Build per-file metadata for UI rendering (used for both permission and result)
|
|
162
|
+
const files = fileChanges.map((change) => ({
|
|
163
|
+
filePath: change.filePath,
|
|
164
|
+
relativePath: path.relative(Instance.worktree, change.movePath ?? change.filePath),
|
|
165
|
+
type: change.type,
|
|
166
|
+
diff: change.diff,
|
|
167
|
+
before: change.oldContent,
|
|
168
|
+
after: change.newContent,
|
|
169
|
+
additions: change.additions,
|
|
170
|
+
deletions: change.deletions,
|
|
171
|
+
movePath: change.movePath,
|
|
172
|
+
}))
|
|
173
|
+
|
|
174
|
+
// Check permissions if needed
|
|
175
|
+
const relativePaths = fileChanges.map((c) => path.relative(Instance.worktree, c.filePath))
|
|
176
|
+
await ctx.ask({
|
|
177
|
+
permission: "edit",
|
|
178
|
+
patterns: relativePaths,
|
|
179
|
+
always: ["*"],
|
|
180
|
+
metadata: {
|
|
181
|
+
filepath: relativePaths.join(", "),
|
|
182
|
+
diff: totalDiff,
|
|
183
|
+
files,
|
|
184
|
+
},
|
|
185
|
+
})
|
|
186
|
+
|
|
187
|
+
// Apply the changes
|
|
188
|
+
const updates: Array<{ file: string; event: "add" | "change" | "unlink" }> = []
|
|
189
|
+
|
|
190
|
+
for (const change of fileChanges) {
|
|
191
|
+
const edited = change.type === "delete" ? undefined : (change.movePath ?? change.filePath)
|
|
192
|
+
switch (change.type) {
|
|
193
|
+
case "add":
|
|
194
|
+
// Create parent directories (recursive: true is safe on existing/root dirs)
|
|
195
|
+
await fs.mkdir(path.dirname(change.filePath), { recursive: true })
|
|
196
|
+
await fs.writeFile(change.filePath, change.newContent, "utf-8")
|
|
197
|
+
updates.push({ file: change.filePath, event: "add" })
|
|
198
|
+
break
|
|
199
|
+
|
|
200
|
+
case "update":
|
|
201
|
+
await fs.writeFile(change.filePath, change.newContent, "utf-8")
|
|
202
|
+
updates.push({ file: change.filePath, event: "change" })
|
|
203
|
+
break
|
|
204
|
+
|
|
205
|
+
case "move":
|
|
206
|
+
if (change.movePath) {
|
|
207
|
+
// Create parent directories (recursive: true is safe on existing/root dirs)
|
|
208
|
+
await fs.mkdir(path.dirname(change.movePath), { recursive: true })
|
|
209
|
+
await fs.writeFile(change.movePath, change.newContent, "utf-8")
|
|
210
|
+
await fs.unlink(change.filePath)
|
|
211
|
+
updates.push({ file: change.filePath, event: "unlink" })
|
|
212
|
+
updates.push({ file: change.movePath, event: "add" })
|
|
213
|
+
}
|
|
214
|
+
break
|
|
215
|
+
|
|
216
|
+
case "delete":
|
|
217
|
+
await fs.unlink(change.filePath)
|
|
218
|
+
updates.push({ file: change.filePath, event: "unlink" })
|
|
219
|
+
break
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
if (edited) {
|
|
223
|
+
await Bus.publish(File.Event.Edited, {
|
|
224
|
+
file: edited,
|
|
225
|
+
})
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
// Publish file change events
|
|
230
|
+
for (const update of updates) {
|
|
231
|
+
await Bus.publish(FileWatcher.Event.Updated, update)
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// Notify LSP of file changes and collect diagnostics
|
|
235
|
+
for (const change of fileChanges) {
|
|
236
|
+
if (change.type === "delete") continue
|
|
237
|
+
const target = change.movePath ?? change.filePath
|
|
238
|
+
await LSP.touchFile(target, true)
|
|
239
|
+
}
|
|
240
|
+
const diagnostics = await LSP.diagnostics()
|
|
241
|
+
|
|
242
|
+
// Generate output summary
|
|
243
|
+
const summaryLines = fileChanges.map((change) => {
|
|
244
|
+
if (change.type === "add") {
|
|
245
|
+
return `A ${path.relative(Instance.worktree, change.filePath)}`
|
|
246
|
+
}
|
|
247
|
+
if (change.type === "delete") {
|
|
248
|
+
return `D ${path.relative(Instance.worktree, change.filePath)}`
|
|
249
|
+
}
|
|
250
|
+
const target = change.movePath ?? change.filePath
|
|
251
|
+
return `M ${path.relative(Instance.worktree, target)}`
|
|
252
|
+
})
|
|
253
|
+
let output = `Success. Updated the following files:\n${summaryLines.join("\n")}`
|
|
254
|
+
|
|
255
|
+
// Report LSP errors for changed files
|
|
256
|
+
const MAX_DIAGNOSTICS_PER_FILE = 20
|
|
257
|
+
for (const change of fileChanges) {
|
|
258
|
+
if (change.type === "delete") continue
|
|
259
|
+
const target = change.movePath ?? change.filePath
|
|
260
|
+
const normalized = Filesystem.normalizePath(target)
|
|
261
|
+
const issues = diagnostics[normalized] ?? []
|
|
262
|
+
const errors = issues.filter((item) => item.severity === 1)
|
|
263
|
+
if (errors.length > 0) {
|
|
264
|
+
const limited = errors.slice(0, MAX_DIAGNOSTICS_PER_FILE)
|
|
265
|
+
const suffix =
|
|
266
|
+
errors.length > MAX_DIAGNOSTICS_PER_FILE ? `\n... and ${errors.length - MAX_DIAGNOSTICS_PER_FILE} more` : ""
|
|
267
|
+
output += `\n\nLSP errors detected in ${path.relative(Instance.worktree, target)}, please fix:\n<diagnostics file="${target}">\n${limited.map(LSP.Diagnostic.pretty).join("\n")}${suffix}\n</diagnostics>`
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
return {
|
|
272
|
+
title: output,
|
|
273
|
+
metadata: {
|
|
274
|
+
diff: totalDiff,
|
|
275
|
+
files,
|
|
276
|
+
diagnostics,
|
|
277
|
+
},
|
|
278
|
+
output,
|
|
279
|
+
}
|
|
280
|
+
},
|
|
281
|
+
})
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
Use the `apply_patch` tool to edit files. Your patch language is a stripped‑down, file‑oriented diff format designed to be easy to parse and safe to apply. You can think of it as a high‑level envelope:
|
|
2
|
+
|
|
3
|
+
*** Begin Patch
|
|
4
|
+
[ one or more file sections ]
|
|
5
|
+
*** End Patch
|
|
6
|
+
|
|
7
|
+
Within that envelope, you get a sequence of file operations.
|
|
8
|
+
You MUST include a header to specify the action you are taking.
|
|
9
|
+
Each operation starts with one of three headers:
|
|
10
|
+
|
|
11
|
+
*** Add File: <path> - create a new file. Every following line is a + line (the initial contents).
|
|
12
|
+
*** Delete File: <path> - remove an existing file. Nothing follows.
|
|
13
|
+
*** Update File: <path> - patch an existing file in place (optionally with a rename).
|
|
14
|
+
|
|
15
|
+
Example patch:
|
|
16
|
+
|
|
17
|
+
```
|
|
18
|
+
*** Begin Patch
|
|
19
|
+
*** Add File: hello.txt
|
|
20
|
+
+Hello world
|
|
21
|
+
*** Update File: src/app.py
|
|
22
|
+
*** Move to: src/main.py
|
|
23
|
+
@@ def greet():
|
|
24
|
+
-print("Hi")
|
|
25
|
+
+print("Hello, world!")
|
|
26
|
+
*** Delete File: obsolete.txt
|
|
27
|
+
*** End Patch
|
|
28
|
+
```
|
|
29
|
+
|
|
30
|
+
It is important to remember:
|
|
31
|
+
|
|
32
|
+
- You must include a header with your intended action (Add/Delete/Update)
|
|
33
|
+
- You must prefix new lines with `+` even when creating a new file
|