@gguf/coder 0.2.9 → 0.3.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.editorconfig +16 -0
- package/.env.example +63 -0
- package/.gitattributes +1 -0
- package/.semgrepignore +19 -0
- package/coder-dummy-file.ts +52 -0
- package/coder.config.example.json +59 -0
- package/coder.config.json +13 -0
- package/color_picker.html +36 -0
- package/dist/tools/execute-bash.js +3 -3
- package/dist/tools/execute-bash.js.map +1 -1
- package/dist/tools/fetch-url.js +3 -3
- package/dist/tools/fetch-url.js.map +1 -1
- package/dist/tools/find-files.d.ts.map +1 -1
- package/dist/tools/find-files.js +1 -1
- package/dist/tools/find-files.js.map +1 -1
- package/dist/tools/lsp-get-diagnostics.js +1 -1
- package/dist/tools/lsp-get-diagnostics.js.map +1 -1
- package/dist/tools/read-file.d.ts.map +1 -1
- package/dist/tools/read-file.js +6 -6
- package/dist/tools/read-file.js.map +1 -1
- package/dist/tools/search-file-contents.d.ts.map +1 -1
- package/dist/tools/search-file-contents.js +1 -1
- package/dist/tools/search-file-contents.js.map +1 -1
- package/dist/tools/string-replace.js +11 -11
- package/dist/tools/string-replace.js.map +1 -1
- package/dist/tools/web-search.d.ts.map +1 -1
- package/dist/tools/web-search.js +3 -3
- package/dist/tools/web-search.js.map +1 -1
- package/dist/tools/write-file.js +4 -4
- package/dist/tools/write-file.js.map +1 -1
- package/dist/utils/tool-result-display.d.ts.map +1 -1
- package/dist/utils/tool-result-display.js +3 -3
- package/dist/utils/tool-result-display.js.map +1 -1
- package/package.json +2 -14
- package/scripts/extract-changelog.js +73 -0
- package/scripts/fetch-models.js +143 -0
- package/scripts/test.sh +40 -0
- package/scripts/update-homebrew-formula.sh +125 -0
- package/scripts/update-nix-version.sh +157 -0
- package/source/ai-sdk-client/AISDKClient.spec.ts +117 -0
- package/source/ai-sdk-client/AISDKClient.ts +155 -0
- package/source/ai-sdk-client/chat/chat-handler.spec.ts +121 -0
- package/source/ai-sdk-client/chat/chat-handler.ts +276 -0
- package/source/ai-sdk-client/chat/streaming-handler.spec.ts +173 -0
- package/source/ai-sdk-client/chat/streaming-handler.ts +110 -0
- package/source/ai-sdk-client/chat/tool-processor.spec.ts +92 -0
- package/source/ai-sdk-client/chat/tool-processor.ts +70 -0
- package/source/ai-sdk-client/converters/message-converter.spec.ts +220 -0
- package/source/ai-sdk-client/converters/message-converter.ts +113 -0
- package/source/ai-sdk-client/converters/tool-converter.spec.ts +90 -0
- package/source/ai-sdk-client/converters/tool-converter.ts +46 -0
- package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +55 -0
- package/source/ai-sdk-client/error-handling/error-extractor.ts +15 -0
- package/source/ai-sdk-client/error-handling/error-parser.spec.ts +169 -0
- package/source/ai-sdk-client/error-handling/error-parser.ts +161 -0
- package/source/ai-sdk-client/index.ts +7 -0
- package/source/ai-sdk-client/providers/provider-factory.spec.ts +71 -0
- package/source/ai-sdk-client/providers/provider-factory.ts +41 -0
- package/source/ai-sdk-client/types.ts +9 -0
- package/source/ai-sdk-client-empty-message.spec.ts +141 -0
- package/source/ai-sdk-client-error-handling.spec.ts +186 -0
- package/source/ai-sdk-client-maxretries.spec.ts +114 -0
- package/source/ai-sdk-client-preparestep.spec.ts +279 -0
- package/source/app/App.spec.tsx +32 -0
- package/source/app/App.tsx +480 -0
- package/source/app/components/AppContainer.spec.tsx +96 -0
- package/source/app/components/AppContainer.tsx +56 -0
- package/source/app/components/ChatInterface.spec.tsx +163 -0
- package/source/app/components/ChatInterface.tsx +144 -0
- package/source/app/components/ModalSelectors.spec.tsx +141 -0
- package/source/app/components/ModalSelectors.tsx +135 -0
- package/source/app/helpers.spec.ts +97 -0
- package/source/app/helpers.ts +63 -0
- package/source/app/index.ts +4 -0
- package/source/app/types.ts +39 -0
- package/source/app/utils/appUtils.ts +294 -0
- package/source/app/utils/conversationState.ts +310 -0
- package/source/app.spec.tsx +244 -0
- package/source/cli.spec.ts +73 -0
- package/source/cli.tsx +51 -0
- package/source/client-factory.spec.ts +48 -0
- package/source/client-factory.ts +178 -0
- package/source/command-parser.spec.ts +127 -0
- package/source/command-parser.ts +36 -0
- package/source/commands/checkpoint.spec.tsx +277 -0
- package/source/commands/checkpoint.tsx +366 -0
- package/source/commands/clear.tsx +22 -0
- package/source/commands/custom-commands.tsx +121 -0
- package/source/commands/exit.ts +21 -0
- package/source/commands/export.spec.tsx +131 -0
- package/source/commands/export.tsx +79 -0
- package/source/commands/help.tsx +120 -0
- package/source/commands/index.ts +17 -0
- package/source/commands/init.tsx +339 -0
- package/source/commands/lsp-command.spec.tsx +281 -0
- package/source/commands/lsp.tsx +120 -0
- package/source/commands/mcp-command.spec.tsx +313 -0
- package/source/commands/mcp.tsx +162 -0
- package/source/commands/model-database.spec.tsx +758 -0
- package/source/commands/model-database.tsx +418 -0
- package/source/commands/model.ts +12 -0
- package/source/commands/provider.ts +12 -0
- package/source/commands/setup-config.tsx +16 -0
- package/source/commands/simple-commands.spec.tsx +175 -0
- package/source/commands/status.ts +12 -0
- package/source/commands/theme.ts +12 -0
- package/source/commands/update.spec.tsx +261 -0
- package/source/commands/update.tsx +201 -0
- package/source/commands/usage.spec.tsx +495 -0
- package/source/commands/usage.tsx +100 -0
- package/source/commands.spec.ts +436 -0
- package/source/commands.ts +83 -0
- package/source/components/assistant-message.spec.tsx +796 -0
- package/source/components/assistant-message.tsx +34 -0
- package/source/components/bash-execution-indicator.tsx +21 -0
- package/source/components/cancelling-indicator.tsx +16 -0
- package/source/components/chat-queue.spec.tsx +83 -0
- package/source/components/chat-queue.tsx +36 -0
- package/source/components/checkpoint-display.spec.tsx +219 -0
- package/source/components/checkpoint-display.tsx +126 -0
- package/source/components/checkpoint-selector.spec.tsx +173 -0
- package/source/components/checkpoint-selector.tsx +173 -0
- package/source/components/development-mode-indicator.spec.tsx +268 -0
- package/source/components/development-mode-indicator.tsx +38 -0
- package/source/components/message-box.spec.tsx +427 -0
- package/source/components/message-box.tsx +87 -0
- package/source/components/model-selector.tsx +132 -0
- package/source/components/provider-selector.tsx +75 -0
- package/source/components/random-spinner.tsx +19 -0
- package/source/components/security-disclaimer.tsx +73 -0
- package/source/components/status-connection-display.spec.tsx +133 -0
- package/source/components/status.tsx +267 -0
- package/source/components/theme-selector.tsx +126 -0
- package/source/components/tool-confirmation.tsx +190 -0
- package/source/components/tool-execution-indicator.tsx +33 -0
- package/source/components/tool-message.tsx +85 -0
- package/source/components/ui/titled-box.spec.tsx +207 -0
- package/source/components/ui/titled-box.tsx +57 -0
- package/source/components/usage/progress-bar.spec.tsx +398 -0
- package/source/components/usage/progress-bar.tsx +30 -0
- package/source/components/usage/usage-display.spec.tsx +780 -0
- package/source/components/usage/usage-display.tsx +291 -0
- package/source/components/user-input.spec.tsx +327 -0
- package/source/components/user-input.tsx +533 -0
- package/source/components/user-message.spec.tsx +230 -0
- package/source/components/user-message.tsx +84 -0
- package/source/components/welcome-message.tsx +76 -0
- package/source/config/env-substitution.ts +65 -0
- package/source/config/index.spec.ts +171 -0
- package/source/config/index.ts +154 -0
- package/source/config/paths.spec.ts +241 -0
- package/source/config/paths.ts +55 -0
- package/source/config/preferences.ts +51 -0
- package/source/config/themes.ts +315 -0
- package/source/constants.ts +130 -0
- package/source/context/mode-context.spec.ts +79 -0
- package/source/context/mode-context.ts +24 -0
- package/source/custom-commands/executor.spec.ts +142 -0
- package/source/custom-commands/executor.ts +64 -0
- package/source/custom-commands/loader.spec.ts +314 -0
- package/source/custom-commands/loader.ts +153 -0
- package/source/custom-commands/parser.ts +196 -0
- package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +39 -0
- package/source/hooks/chat-handler/conversation/conversation-loop.tsx +511 -0
- package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +50 -0
- package/source/hooks/chat-handler/conversation/tool-executor.tsx +109 -0
- package/source/hooks/chat-handler/index.ts +12 -0
- package/source/hooks/chat-handler/state/streaming-state.spec.ts +26 -0
- package/source/hooks/chat-handler/state/streaming-state.ts +19 -0
- package/source/hooks/chat-handler/types.ts +38 -0
- package/source/hooks/chat-handler/useChatHandler.spec.tsx +321 -0
- package/source/hooks/chat-handler/useChatHandler.tsx +194 -0
- package/source/hooks/chat-handler/utils/context-checker.spec.ts +60 -0
- package/source/hooks/chat-handler/utils/context-checker.tsx +73 -0
- package/source/hooks/chat-handler/utils/message-helpers.spec.ts +42 -0
- package/source/hooks/chat-handler/utils/message-helpers.tsx +36 -0
- package/source/hooks/chat-handler/utils/tool-filters.spec.ts +109 -0
- package/source/hooks/chat-handler/utils/tool-filters.ts +64 -0
- package/source/hooks/useAppHandlers.tsx +291 -0
- package/source/hooks/useAppInitialization.tsx +422 -0
- package/source/hooks/useAppState.tsx +311 -0
- package/source/hooks/useDirectoryTrust.tsx +98 -0
- package/source/hooks/useInputState.ts +414 -0
- package/source/hooks/useModeHandlers.tsx +302 -0
- package/source/hooks/useNonInteractiveMode.ts +140 -0
- package/source/hooks/useTerminalWidth.tsx +81 -0
- package/source/hooks/useTheme.ts +18 -0
- package/source/hooks/useToolHandler.tsx +349 -0
- package/source/hooks/useUIState.ts +61 -0
- package/source/init/agents-template-generator.ts +421 -0
- package/source/init/existing-rules-extractor.ts +319 -0
- package/source/init/file-scanner.spec.ts +227 -0
- package/source/init/file-scanner.ts +238 -0
- package/source/init/framework-detector.ts +382 -0
- package/source/init/language-detector.ts +269 -0
- package/source/init/project-analyzer.spec.ts +231 -0
- package/source/init/project-analyzer.ts +458 -0
- package/source/lsp/index.ts +31 -0
- package/source/lsp/lsp-client.spec.ts +508 -0
- package/source/lsp/lsp-client.ts +487 -0
- package/source/lsp/lsp-manager.spec.ts +477 -0
- package/source/lsp/lsp-manager.ts +419 -0
- package/source/lsp/protocol.spec.ts +502 -0
- package/source/lsp/protocol.ts +360 -0
- package/source/lsp/server-discovery.spec.ts +654 -0
- package/source/lsp/server-discovery.ts +515 -0
- package/source/markdown-parser/html-entities.spec.ts +88 -0
- package/source/markdown-parser/html-entities.ts +45 -0
- package/source/markdown-parser/index.spec.ts +281 -0
- package/source/markdown-parser/index.ts +126 -0
- package/source/markdown-parser/table-parser.spec.ts +133 -0
- package/source/markdown-parser/table-parser.ts +114 -0
- package/source/markdown-parser/utils.spec.ts +70 -0
- package/source/markdown-parser/utils.ts +13 -0
- package/source/mcp/mcp-client.spec.ts +81 -0
- package/source/mcp/mcp-client.ts +625 -0
- package/source/mcp/transport-factory.spec.ts +406 -0
- package/source/mcp/transport-factory.ts +312 -0
- package/source/message-handler.ts +67 -0
- package/source/model-database/database-engine.spec.ts +494 -0
- package/source/model-database/database-engine.ts +50 -0
- package/source/model-database/model-database.spec.ts +363 -0
- package/source/model-database/model-database.ts +91 -0
- package/source/model-database/model-engine.spec.ts +447 -0
- package/source/model-database/model-engine.ts +65 -0
- package/source/model-database/model-fetcher.spec.ts +583 -0
- package/source/model-database/model-fetcher.ts +330 -0
- package/source/models/index.ts +1 -0
- package/source/models/models-cache.spec.ts +214 -0
- package/source/models/models-cache.ts +78 -0
- package/source/models/models-dev-client.spec.ts +379 -0
- package/source/models/models-dev-client.ts +329 -0
- package/source/models/models-types.ts +68 -0
- package/source/prompt-history.ts +155 -0
- package/source/security/command-injection.spec.ts +240 -0
- package/source/services/checkpoint-manager.spec.ts +523 -0
- package/source/services/checkpoint-manager.ts +466 -0
- package/source/services/file-snapshot.spec.ts +569 -0
- package/source/services/file-snapshot.ts +220 -0
- package/source/test-utils/render-with-theme.tsx +48 -0
- package/source/tokenization/index.ts +1 -0
- package/source/tokenization/tokenizer-factory.spec.ts +170 -0
- package/source/tokenization/tokenizer-factory.ts +125 -0
- package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +200 -0
- package/source/tokenization/tokenizers/anthropic-tokenizer.ts +43 -0
- package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +236 -0
- package/source/tokenization/tokenizers/fallback-tokenizer.ts +26 -0
- package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +224 -0
- package/source/tokenization/tokenizers/llama-tokenizer.ts +41 -0
- package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +184 -0
- package/source/tokenization/tokenizers/openai-tokenizer.ts +57 -0
- package/source/tool-calling/index.ts +5 -0
- package/source/tool-calling/json-parser.spec.ts +639 -0
- package/source/tool-calling/json-parser.ts +247 -0
- package/source/tool-calling/tool-parser.spec.ts +395 -0
- package/source/tool-calling/tool-parser.ts +120 -0
- package/source/tool-calling/xml-parser.spec.ts +662 -0
- package/source/tool-calling/xml-parser.ts +289 -0
- package/source/tools/execute-bash.spec.tsx +353 -0
- package/source/tools/execute-bash.tsx +219 -0
- package/source/tools/execute-function.spec.ts +130 -0
- package/source/tools/fetch-url.spec.tsx +342 -0
- package/source/tools/fetch-url.tsx +172 -0
- package/source/tools/find-files.spec.tsx +924 -0
- package/source/tools/find-files.tsx +293 -0
- package/source/tools/index.ts +102 -0
- package/source/tools/lsp-get-diagnostics.tsx +192 -0
- package/source/tools/needs-approval.spec.ts +282 -0
- package/source/tools/read-file.spec.tsx +801 -0
- package/source/tools/read-file.tsx +387 -0
- package/source/tools/search-file-contents.spec.tsx +1273 -0
- package/source/tools/search-file-contents.tsx +293 -0
- package/source/tools/string-replace.spec.tsx +730 -0
- package/source/tools/string-replace.tsx +548 -0
- package/source/tools/tool-manager.ts +210 -0
- package/source/tools/tool-registry.spec.ts +415 -0
- package/source/tools/tool-registry.ts +228 -0
- package/source/tools/web-search.tsx +223 -0
- package/source/tools/write-file.spec.tsx +559 -0
- package/source/tools/write-file.tsx +228 -0
- package/source/types/app.ts +37 -0
- package/source/types/checkpoint.ts +48 -0
- package/source/types/commands.ts +46 -0
- package/source/types/components.ts +27 -0
- package/source/types/config.ts +103 -0
- package/source/types/core-connection-status.spec.ts +67 -0
- package/source/types/core.ts +181 -0
- package/source/types/hooks.ts +50 -0
- package/source/types/index.ts +12 -0
- package/source/types/markdown-parser.ts +11 -0
- package/source/types/mcp.ts +52 -0
- package/source/types/system.ts +16 -0
- package/source/types/tokenization.ts +41 -0
- package/source/types/ui.ts +40 -0
- package/source/types/usage.ts +58 -0
- package/source/types/utils.ts +16 -0
- package/source/usage/calculator.spec.ts +385 -0
- package/source/usage/calculator.ts +104 -0
- package/source/usage/storage.spec.ts +703 -0
- package/source/usage/storage.ts +238 -0
- package/source/usage/tracker.spec.ts +456 -0
- package/source/usage/tracker.ts +102 -0
- package/source/utils/atomic-deletion.spec.ts +194 -0
- package/source/utils/atomic-deletion.ts +127 -0
- package/source/utils/bounded-map.spec.ts +300 -0
- package/source/utils/bounded-map.ts +193 -0
- package/source/utils/checkpoint-utils.spec.ts +222 -0
- package/source/utils/checkpoint-utils.ts +92 -0
- package/source/utils/error-formatter.spec.ts +169 -0
- package/source/utils/error-formatter.ts +194 -0
- package/source/utils/file-autocomplete.spec.ts +173 -0
- package/source/utils/file-autocomplete.ts +196 -0
- package/source/utils/file-cache.spec.ts +309 -0
- package/source/utils/file-cache.ts +195 -0
- package/source/utils/file-content-loader.spec.ts +180 -0
- package/source/utils/file-content-loader.ts +179 -0
- package/source/utils/file-mention-handler.spec.ts +261 -0
- package/source/utils/file-mention-handler.ts +84 -0
- package/source/utils/file-mention-parser.spec.ts +182 -0
- package/source/utils/file-mention-parser.ts +170 -0
- package/source/utils/fuzzy-matching.spec.ts +149 -0
- package/source/utils/fuzzy-matching.ts +146 -0
- package/source/utils/indentation-normalizer.spec.ts +216 -0
- package/source/utils/indentation-normalizer.ts +76 -0
- package/source/utils/installation-detector.spec.ts +178 -0
- package/source/utils/installation-detector.ts +153 -0
- package/source/utils/logging/config.spec.ts +311 -0
- package/source/utils/logging/config.ts +210 -0
- package/source/utils/logging/console-facade.spec.ts +184 -0
- package/source/utils/logging/console-facade.ts +384 -0
- package/source/utils/logging/correlation.spec.ts +679 -0
- package/source/utils/logging/correlation.ts +474 -0
- package/source/utils/logging/formatters.spec.ts +464 -0
- package/source/utils/logging/formatters.ts +207 -0
- package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +93 -0
- package/source/utils/logging/health-monitor/alerts/alert-manager.ts +79 -0
- package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/configuration-check.ts +43 -0
- package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/logging-check.ts +58 -0
- package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +100 -0
- package/source/utils/logging/health-monitor/checks/memory-check.ts +78 -0
- package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/performance-check.ts +56 -0
- package/source/utils/logging/health-monitor/checks/request-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/request-check.ts +76 -0
- package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +70 -0
- package/source/utils/logging/health-monitor/core/health-check-runner.ts +138 -0
- package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +58 -0
- package/source/utils/logging/health-monitor/core/health-monitor.ts +344 -0
- package/source/utils/logging/health-monitor/core/scoring.spec.ts +65 -0
- package/source/utils/logging/health-monitor/core/scoring.ts +91 -0
- package/source/utils/logging/health-monitor/index.ts +15 -0
- package/source/utils/logging/health-monitor/instances.ts +48 -0
- package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +141 -0
- package/source/utils/logging/health-monitor/middleware/http-middleware.ts +75 -0
- package/source/utils/logging/health-monitor/types.ts +126 -0
- package/source/utils/logging/index.spec.ts +284 -0
- package/source/utils/logging/index.ts +236 -0
- package/source/utils/logging/integration.spec.ts +441 -0
- package/source/utils/logging/log-method-factory.spec.ts +573 -0
- package/source/utils/logging/log-method-factory.ts +233 -0
- package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +277 -0
- package/source/utils/logging/log-query/aggregation/aggregator.ts +159 -0
- package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +159 -0
- package/source/utils/logging/log-query/aggregation/facet-generator.ts +47 -0
- package/source/utils/logging/log-query/index.ts +23 -0
- package/source/utils/logging/log-query/query/filter-predicates.spec.ts +247 -0
- package/source/utils/logging/log-query/query/filter-predicates.ts +154 -0
- package/source/utils/logging/log-query/query/query-builder.spec.ts +182 -0
- package/source/utils/logging/log-query/query/query-builder.ts +151 -0
- package/source/utils/logging/log-query/query/query-engine.spec.ts +214 -0
- package/source/utils/logging/log-query/query/query-engine.ts +45 -0
- package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +143 -0
- package/source/utils/logging/log-query/storage/circular-buffer.ts +75 -0
- package/source/utils/logging/log-query/storage/index-manager.spec.ts +150 -0
- package/source/utils/logging/log-query/storage/index-manager.ts +71 -0
- package/source/utils/logging/log-query/storage/log-storage.spec.ts +257 -0
- package/source/utils/logging/log-query/storage/log-storage.ts +80 -0
- package/source/utils/logging/log-query/types.ts +163 -0
- package/source/utils/logging/log-query/utils/helpers.spec.ts +263 -0
- package/source/utils/logging/log-query/utils/helpers.ts +72 -0
- package/source/utils/logging/log-query/utils/sorting.spec.ts +182 -0
- package/source/utils/logging/log-query/utils/sorting.ts +61 -0
- package/source/utils/logging/logger-provider.spec.ts +262 -0
- package/source/utils/logging/logger-provider.ts +362 -0
- package/source/utils/logging/performance.spec.ts +209 -0
- package/source/utils/logging/performance.ts +757 -0
- package/source/utils/logging/pino-logger.spec.ts +425 -0
- package/source/utils/logging/pino-logger.ts +514 -0
- package/source/utils/logging/redaction.spec.ts +490 -0
- package/source/utils/logging/redaction.ts +267 -0
- package/source/utils/logging/request-tracker.spec.ts +1198 -0
- package/source/utils/logging/request-tracker.ts +803 -0
- package/source/utils/logging/transports.spec.ts +505 -0
- package/source/utils/logging/transports.ts +305 -0
- package/source/utils/logging/types.ts +216 -0
- package/source/utils/message-builder.spec.ts +179 -0
- package/source/utils/message-builder.ts +101 -0
- package/source/utils/message-queue.tsx +486 -0
- package/source/utils/paste-detection.spec.ts +69 -0
- package/source/utils/paste-detection.ts +124 -0
- package/source/utils/paste-roundtrip.spec.ts +442 -0
- package/source/utils/paste-utils.spec.ts +128 -0
- package/source/utils/paste-utils.ts +52 -0
- package/source/utils/programming-language-helper.spec.ts +74 -0
- package/source/utils/programming-language-helper.ts +32 -0
- package/source/utils/prompt-assembly.spec.ts +221 -0
- package/source/utils/prompt-processor.ts +173 -0
- package/source/utils/tool-args-parser.spec.ts +136 -0
- package/source/utils/tool-args-parser.ts +54 -0
- package/source/utils/tool-cancellation.spec.ts +230 -0
- package/source/utils/tool-cancellation.ts +28 -0
- package/source/utils/tool-result-display.spec.tsx +469 -0
- package/source/utils/tool-result-display.tsx +90 -0
- package/source/utils/update-checker.spec.ts +383 -0
- package/source/utils/update-checker.ts +183 -0
- package/source/wizard/config-wizard.spec.tsx +103 -0
- package/source/wizard/config-wizard.tsx +382 -0
- package/source/wizard/steps/location-step.spec.tsx +186 -0
- package/source/wizard/steps/location-step.tsx +147 -0
- package/source/wizard/steps/mcp-step.spec.tsx +607 -0
- package/source/wizard/steps/mcp-step.tsx +632 -0
- package/source/wizard/steps/provider-step.spec.tsx +342 -0
- package/source/wizard/steps/provider-step.tsx +957 -0
- package/source/wizard/steps/summary-step.spec.tsx +749 -0
- package/source/wizard/steps/summary-step.tsx +228 -0
- package/source/wizard/templates/mcp-templates.spec.ts +613 -0
- package/source/wizard/templates/mcp-templates.ts +570 -0
- package/source/wizard/templates/provider-templates.spec.ts +152 -0
- package/source/wizard/templates/provider-templates.ts +485 -0
- package/source/wizard/utils/fetch-cloud-models.spec.ts +428 -0
- package/source/wizard/utils/fetch-cloud-models.ts +223 -0
- package/source/wizard/utils/fetch-local-models.spec.ts +297 -0
- package/source/wizard/utils/fetch-local-models.ts +192 -0
- package/source/wizard/validation-array.spec.ts +264 -0
- package/source/wizard/validation.spec.ts +373 -0
- package/source/wizard/validation.ts +232 -0
- package/source/app/prompts/main-prompt.md +0 -122
|
@@ -0,0 +1,548 @@
|
|
|
1
|
+
import { constants } from 'node:fs';
|
|
2
|
+
import { access, writeFile } from 'node:fs/promises';
|
|
3
|
+
import { resolve } from 'node:path';
|
|
4
|
+
import { highlight } from 'cli-highlight';
|
|
5
|
+
import { Box, Text } from 'ink';
|
|
6
|
+
import React from 'react';
|
|
7
|
+
|
|
8
|
+
import ToolMessage from '@/components/tool-message';
|
|
9
|
+
import { getColors } from '@/config/index';
|
|
10
|
+
import { getCurrentMode } from '@/context/mode-context';
|
|
11
|
+
import { jsonSchema, tool } from '@/types/core';
|
|
12
|
+
import type { Colors } from '@/types/index';
|
|
13
|
+
import { getCachedFileContent, invalidateCache } from '@/utils/file-cache';
|
|
14
|
+
import { normalizeIndentation } from '@/utils/indentation-normalizer';
|
|
15
|
+
import { getLanguageFromExtension } from '@/utils/programming-language-helper';
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
interface StringReplaceArgs {
|
|
19
|
+
path: string;
|
|
20
|
+
old_str: string;
|
|
21
|
+
new_str: string;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const executeStringReplace = async (
|
|
25
|
+
args: StringReplaceArgs,
|
|
26
|
+
): Promise<string> => {
|
|
27
|
+
const { path, old_str, new_str } = args;
|
|
28
|
+
|
|
29
|
+
// Validate old_str is not empty
|
|
30
|
+
if (!old_str || old_str.length === 0) {
|
|
31
|
+
throw new Error(
|
|
32
|
+
'old_str cannot be empty. Provide the exact content to find and replace.',
|
|
33
|
+
);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
const absPath = resolve(path);
|
|
37
|
+
const cached = await getCachedFileContent(absPath);
|
|
38
|
+
const fileContent = cached.content;
|
|
39
|
+
|
|
40
|
+
// Count occurrences of old_str
|
|
41
|
+
const occurrences = fileContent.split(old_str).length - 1;
|
|
42
|
+
|
|
43
|
+
if (occurrences === 0) {
|
|
44
|
+
throw new Error(
|
|
45
|
+
`Content not found in file. The file may have changed since you last read it.\n\nSearching for:\n${old_str}\n\nSuggestion: Read the file again to see current contents.`,
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
if (occurrences > 1) {
|
|
50
|
+
throw new Error(
|
|
51
|
+
`Found ${occurrences} matches for the search string. Please provide more surrounding context to make the match unique.\n\nSearching for:\n${old_str}`,
|
|
52
|
+
);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// Perform the replacement
|
|
56
|
+
const newContent = fileContent.replace(old_str, new_str);
|
|
57
|
+
|
|
58
|
+
// Write updated content
|
|
59
|
+
await writeFile(absPath, newContent, 'utf-8');
|
|
60
|
+
|
|
61
|
+
// Invalidate cache after write
|
|
62
|
+
invalidateCache(absPath);
|
|
63
|
+
|
|
64
|
+
// Calculate line numbers where change occurred
|
|
65
|
+
const beforeLines = fileContent.split('\n');
|
|
66
|
+
const oldStrLines = old_str.split('\n');
|
|
67
|
+
const newStrLines = new_str.split('\n');
|
|
68
|
+
|
|
69
|
+
// Find the line where the change started
|
|
70
|
+
let startLine = 0;
|
|
71
|
+
let searchIndex = 0;
|
|
72
|
+
for (let i = 0; i < beforeLines.length; i++) {
|
|
73
|
+
const lineWithNewline =
|
|
74
|
+
beforeLines[i] + (i < beforeLines.length - 1 ? '\n' : '');
|
|
75
|
+
if (fileContent.indexOf(old_str, searchIndex) === searchIndex) {
|
|
76
|
+
startLine = i + 1;
|
|
77
|
+
break;
|
|
78
|
+
}
|
|
79
|
+
searchIndex += lineWithNewline.length;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
const endLine = startLine + oldStrLines.length - 1;
|
|
83
|
+
const newEndLine = startLine + newStrLines.length - 1;
|
|
84
|
+
|
|
85
|
+
// Generate full file contents to show the model the current file state
|
|
86
|
+
const newLines = newContent.split('\n');
|
|
87
|
+
let fileContext = '\n\nUpdated file contents:\n';
|
|
88
|
+
for (let i = 0; i < newLines.length; i++) {
|
|
89
|
+
const lineNumStr = String(i + 1).padStart(4, ' ');
|
|
90
|
+
const line = newLines[i] || '';
|
|
91
|
+
fileContext += `${lineNumStr}: ${line}\n`;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
const rangeDesc =
|
|
95
|
+
startLine === endLine
|
|
96
|
+
? `line ${startLine}`
|
|
97
|
+
: `lines ${startLine}-${endLine}`;
|
|
98
|
+
const newRangeDesc =
|
|
99
|
+
startLine === newEndLine
|
|
100
|
+
? `line ${startLine}`
|
|
101
|
+
: `lines ${startLine}-${newEndLine}`;
|
|
102
|
+
|
|
103
|
+
return `Successfully replaced content at ${rangeDesc} (now ${newRangeDesc}).${fileContext}`;
|
|
104
|
+
};
|
|
105
|
+
|
|
106
|
+
const stringReplaceCoreTool = tool({
|
|
107
|
+
description:
|
|
108
|
+
'Replace exact string content in a file. IMPORTANT: Provide exact content including whitespace and surrounding context. For unique matching, include 2-3 lines before/after the change. Break large changes into multiple small replacements.',
|
|
109
|
+
inputSchema: jsonSchema<StringReplaceArgs>({
|
|
110
|
+
type: 'object',
|
|
111
|
+
properties: {
|
|
112
|
+
path: {
|
|
113
|
+
type: 'string',
|
|
114
|
+
description: 'The path to the file to edit.',
|
|
115
|
+
},
|
|
116
|
+
old_str: {
|
|
117
|
+
type: 'string',
|
|
118
|
+
description:
|
|
119
|
+
'The EXACT string to find and replace, including all whitespace, newlines, and indentation. Must match exactly. Include surrounding context (2-3 lines) to ensure unique match.',
|
|
120
|
+
},
|
|
121
|
+
new_str: {
|
|
122
|
+
type: 'string',
|
|
123
|
+
description:
|
|
124
|
+
'The replacement string. Can be empty to delete content. Must preserve proper indentation and formatting.',
|
|
125
|
+
},
|
|
126
|
+
},
|
|
127
|
+
required: ['path', 'old_str', 'new_str'],
|
|
128
|
+
}),
|
|
129
|
+
// Medium risk: file write operation, requires approval except in auto-accept mode
|
|
130
|
+
needsApproval: () => {
|
|
131
|
+
const mode = getCurrentMode();
|
|
132
|
+
return mode !== 'auto-accept'; // true in normal/plan, false in auto-accept
|
|
133
|
+
},
|
|
134
|
+
execute: async (args, _options) => {
|
|
135
|
+
return await executeStringReplace(args);
|
|
136
|
+
},
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
const StringReplaceFormatter = React.memo(
|
|
140
|
+
({ preview }: { preview: React.ReactElement }) => {
|
|
141
|
+
return preview;
|
|
142
|
+
},
|
|
143
|
+
);
|
|
144
|
+
|
|
145
|
+
async function formatStringReplacePreview(
|
|
146
|
+
args: StringReplaceArgs,
|
|
147
|
+
result?: string,
|
|
148
|
+
colors?: Colors,
|
|
149
|
+
): Promise<React.ReactElement> {
|
|
150
|
+
const themeColors = colors || getColors();
|
|
151
|
+
const { path, old_str, new_str } = args;
|
|
152
|
+
|
|
153
|
+
const isResult = result !== undefined;
|
|
154
|
+
|
|
155
|
+
try {
|
|
156
|
+
const absPath = resolve(path);
|
|
157
|
+
const cached = await getCachedFileContent(absPath);
|
|
158
|
+
const fileContent = cached.content;
|
|
159
|
+
const ext = path.split('.').pop()?.toLowerCase() ?? '';
|
|
160
|
+
const language = getLanguageFromExtension(ext);
|
|
161
|
+
|
|
162
|
+
// In result mode, skip validation since file has already been modified
|
|
163
|
+
if (isResult) {
|
|
164
|
+
const messageContent = (
|
|
165
|
+
<Box flexDirection="column">
|
|
166
|
+
<Text color={themeColors.tool}>☕︎ string_replace</Text>
|
|
167
|
+
|
|
168
|
+
<Box>
|
|
169
|
+
<Text color={themeColors.secondary}>Path: </Text>
|
|
170
|
+
<Text color={themeColors.primary}>{path}</Text>
|
|
171
|
+
</Box>
|
|
172
|
+
|
|
173
|
+
<Box flexDirection="column" marginTop={1}>
|
|
174
|
+
<Text color={themeColors.success}>
|
|
175
|
+
✓ String replacement completed successfully
|
|
176
|
+
</Text>
|
|
177
|
+
</Box>
|
|
178
|
+
</Box>
|
|
179
|
+
);
|
|
180
|
+
|
|
181
|
+
return <ToolMessage message={messageContent} hideBox={true} />;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
// Preview mode - validate old_str exists and is unique
|
|
185
|
+
const occurrences = fileContent.split(old_str).length - 1;
|
|
186
|
+
|
|
187
|
+
if (occurrences === 0) {
|
|
188
|
+
const errorContent = (
|
|
189
|
+
<Box flexDirection="column">
|
|
190
|
+
<Text color={themeColors.tool}>☕︎ string_replace</Text>
|
|
191
|
+
|
|
192
|
+
<Box>
|
|
193
|
+
<Text color={themeColors.secondary}>Path: </Text>
|
|
194
|
+
<Text color={themeColors.primary}>{path}</Text>
|
|
195
|
+
</Box>
|
|
196
|
+
|
|
197
|
+
<Box flexDirection="column" marginTop={1}>
|
|
198
|
+
<Text color={themeColors.error}>
|
|
199
|
+
✗ Error: Content not found in file. The file may have changed
|
|
200
|
+
since you last read it.
|
|
201
|
+
</Text>
|
|
202
|
+
</Box>
|
|
203
|
+
|
|
204
|
+
<Box flexDirection="column" marginTop={1}>
|
|
205
|
+
<Text color={themeColors.secondary}>Searching for:</Text>
|
|
206
|
+
{old_str.split('\n').map((line, i) => (
|
|
207
|
+
<Text key={i} color={themeColors.white}>
|
|
208
|
+
{line}
|
|
209
|
+
</Text>
|
|
210
|
+
))}
|
|
211
|
+
</Box>
|
|
212
|
+
</Box>
|
|
213
|
+
);
|
|
214
|
+
return <ToolMessage message={errorContent} hideBox={true} />;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
if (occurrences > 1) {
|
|
218
|
+
const errorContent = (
|
|
219
|
+
<Box flexDirection="column">
|
|
220
|
+
<Text color={themeColors.tool}>☕︎ string_replace</Text>
|
|
221
|
+
|
|
222
|
+
<Box>
|
|
223
|
+
<Text color={themeColors.secondary}>Path: </Text>
|
|
224
|
+
<Text color={themeColors.primary}>{path}</Text>
|
|
225
|
+
</Box>
|
|
226
|
+
|
|
227
|
+
<Box flexDirection="column" marginTop={1}>
|
|
228
|
+
<Text color={themeColors.error}>
|
|
229
|
+
✗ Error: Found {occurrences} matches
|
|
230
|
+
</Text>
|
|
231
|
+
<Text color={themeColors.secondary}>
|
|
232
|
+
Add more surrounding context to make the match unique.
|
|
233
|
+
</Text>
|
|
234
|
+
</Box>
|
|
235
|
+
|
|
236
|
+
<Box flexDirection="column" marginTop={1}>
|
|
237
|
+
<Text color={themeColors.secondary}>Searching for:</Text>
|
|
238
|
+
{old_str.split('\n').map((line, i) => (
|
|
239
|
+
<Text key={i} color={themeColors.white}>
|
|
240
|
+
{line}
|
|
241
|
+
</Text>
|
|
242
|
+
))}
|
|
243
|
+
</Box>
|
|
244
|
+
</Box>
|
|
245
|
+
);
|
|
246
|
+
return <ToolMessage message={errorContent} hideBox={true} />;
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
// Find location of the match in the file
|
|
250
|
+
const matchIndex = fileContent.indexOf(old_str);
|
|
251
|
+
const beforeContent = fileContent.substring(0, matchIndex);
|
|
252
|
+
const beforeLines = beforeContent.split('\n');
|
|
253
|
+
const startLine = beforeLines.length;
|
|
254
|
+
|
|
255
|
+
const oldStrLines = old_str.split('\n');
|
|
256
|
+
const newStrLines = new_str.split('\n');
|
|
257
|
+
const endLine = startLine + oldStrLines.length - 1;
|
|
258
|
+
|
|
259
|
+
const allLines = fileContent.split('\n');
|
|
260
|
+
const contextLines = 3;
|
|
261
|
+
const showStart = Math.max(0, startLine - 1 - contextLines);
|
|
262
|
+
const showEnd = Math.min(allLines.length - 1, endLine - 1 + contextLines);
|
|
263
|
+
|
|
264
|
+
// Collect all lines to be displayed for normalization
|
|
265
|
+
const linesToNormalize: string[] = [];
|
|
266
|
+
|
|
267
|
+
// Context before
|
|
268
|
+
for (let i = showStart; i < startLine - 1; i++) {
|
|
269
|
+
linesToNormalize.push(allLines[i] || '');
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
// Old lines
|
|
273
|
+
for (let i = 0; i < oldStrLines.length; i++) {
|
|
274
|
+
linesToNormalize.push(oldStrLines[i] || '');
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
// New lines
|
|
278
|
+
for (let i = 0; i < newStrLines.length; i++) {
|
|
279
|
+
linesToNormalize.push(newStrLines[i] || '');
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
// Context after
|
|
283
|
+
for (let i = endLine; i <= showEnd; i++) {
|
|
284
|
+
linesToNormalize.push(allLines[i] || '');
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
// Normalize indentation
|
|
288
|
+
const normalizedLines = normalizeIndentation(linesToNormalize);
|
|
289
|
+
|
|
290
|
+
// Split normalized lines back into sections
|
|
291
|
+
let lineIndex = 0;
|
|
292
|
+
const contextBeforeCount = startLine - 1 - showStart;
|
|
293
|
+
const normalizedContextBefore = normalizedLines.slice(
|
|
294
|
+
lineIndex,
|
|
295
|
+
lineIndex + contextBeforeCount,
|
|
296
|
+
);
|
|
297
|
+
lineIndex += contextBeforeCount;
|
|
298
|
+
|
|
299
|
+
const normalizedOldLines = normalizedLines.slice(
|
|
300
|
+
lineIndex,
|
|
301
|
+
lineIndex + oldStrLines.length,
|
|
302
|
+
);
|
|
303
|
+
lineIndex += oldStrLines.length;
|
|
304
|
+
|
|
305
|
+
const normalizedNewLines = normalizedLines.slice(
|
|
306
|
+
lineIndex,
|
|
307
|
+
lineIndex + newStrLines.length,
|
|
308
|
+
);
|
|
309
|
+
lineIndex += newStrLines.length;
|
|
310
|
+
|
|
311
|
+
const normalizedContextAfter = normalizedLines.slice(lineIndex);
|
|
312
|
+
|
|
313
|
+
const contextBefore: React.ReactElement[] = [];
|
|
314
|
+
const removedLines: React.ReactElement[] = [];
|
|
315
|
+
const addedLines: React.ReactElement[] = [];
|
|
316
|
+
const contextAfter: React.ReactElement[] = [];
|
|
317
|
+
|
|
318
|
+
// Show context before
|
|
319
|
+
for (let i = 0; i < normalizedContextBefore.length; i++) {
|
|
320
|
+
const actualLineNum = showStart + i;
|
|
321
|
+
const lineNumStr = String(actualLineNum + 1).padStart(4, ' ');
|
|
322
|
+
const line = normalizedContextBefore[i] || '';
|
|
323
|
+
let displayLine: string;
|
|
324
|
+
try {
|
|
325
|
+
displayLine = highlight(line, { language, theme: 'default' });
|
|
326
|
+
} catch {
|
|
327
|
+
displayLine = line;
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
contextBefore.push(
|
|
331
|
+
<Text key={`before-${i}`} color={themeColors.secondary}>
|
|
332
|
+
{lineNumStr} {displayLine}
|
|
333
|
+
</Text>,
|
|
334
|
+
);
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
// Show removed lines (old_str)
|
|
338
|
+
for (let i = 0; i < normalizedOldLines.length; i++) {
|
|
339
|
+
const lineNumStr = String(startLine + i).padStart(4, ' ');
|
|
340
|
+
const line = normalizedOldLines[i] || '';
|
|
341
|
+
let displayLine: string;
|
|
342
|
+
try {
|
|
343
|
+
displayLine = highlight(line, { language, theme: 'default' });
|
|
344
|
+
} catch {
|
|
345
|
+
displayLine = line;
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
removedLines.push(
|
|
349
|
+
<Text
|
|
350
|
+
key={`remove-${i}`}
|
|
351
|
+
backgroundColor={themeColors.diffRemoved}
|
|
352
|
+
color={themeColors.diffRemovedText}
|
|
353
|
+
wrap="wrap"
|
|
354
|
+
>
|
|
355
|
+
{lineNumStr} - {displayLine}
|
|
356
|
+
</Text>,
|
|
357
|
+
);
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
// Show added lines (new_str)
|
|
361
|
+
for (let i = 0; i < normalizedNewLines.length; i++) {
|
|
362
|
+
const lineNumStr = String(startLine + i).padStart(4, ' ');
|
|
363
|
+
const line = normalizedNewLines[i] || '';
|
|
364
|
+
let displayLine: string;
|
|
365
|
+
try {
|
|
366
|
+
displayLine = highlight(line, { language, theme: 'default' });
|
|
367
|
+
} catch {
|
|
368
|
+
displayLine = line;
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
addedLines.push(
|
|
372
|
+
<Text
|
|
373
|
+
key={`add-${i}`}
|
|
374
|
+
backgroundColor={themeColors.diffAdded}
|
|
375
|
+
color={themeColors.diffAddedText}
|
|
376
|
+
wrap="wrap"
|
|
377
|
+
>
|
|
378
|
+
{lineNumStr} + {displayLine}
|
|
379
|
+
</Text>,
|
|
380
|
+
);
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
// Show context after
|
|
384
|
+
const lineDelta = newStrLines.length - oldStrLines.length;
|
|
385
|
+
for (let i = 0; i < normalizedContextAfter.length; i++) {
|
|
386
|
+
const actualLineNum = endLine + i;
|
|
387
|
+
const lineNumStr = String(actualLineNum + lineDelta + 1).padStart(4, ' ');
|
|
388
|
+
const line = normalizedContextAfter[i] || '';
|
|
389
|
+
let displayLine: string;
|
|
390
|
+
try {
|
|
391
|
+
displayLine = highlight(line, { language, theme: 'default' });
|
|
392
|
+
} catch {
|
|
393
|
+
displayLine = line;
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
contextAfter.push(
|
|
397
|
+
<Text key={`after-${i}`} color={themeColors.secondary}>
|
|
398
|
+
{lineNumStr} {displayLine}
|
|
399
|
+
</Text>,
|
|
400
|
+
);
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
const rangeDesc =
|
|
404
|
+
startLine === endLine
|
|
405
|
+
? `line ${startLine}`
|
|
406
|
+
: `lines ${startLine}-${endLine}`;
|
|
407
|
+
|
|
408
|
+
const messageContent = (
|
|
409
|
+
<Box flexDirection="column">
|
|
410
|
+
<Text color={themeColors.tool}>☕︎ string_replace</Text>
|
|
411
|
+
|
|
412
|
+
<Box>
|
|
413
|
+
<Text color={themeColors.secondary}>Path: </Text>
|
|
414
|
+
<Text color={themeColors.primary}>{path}</Text>
|
|
415
|
+
</Box>
|
|
416
|
+
|
|
417
|
+
<Box>
|
|
418
|
+
<Text color={themeColors.secondary}>Location: </Text>
|
|
419
|
+
<Text color={themeColors.white}>{rangeDesc}</Text>
|
|
420
|
+
</Box>
|
|
421
|
+
|
|
422
|
+
<Box flexDirection="column" marginTop={1}>
|
|
423
|
+
<Text color={themeColors.success}>
|
|
424
|
+
{isResult ? '✓ Replace completed' : '✓ Replacing'}{' '}
|
|
425
|
+
{oldStrLines.length} line{oldStrLines.length > 1 ? 's' : ''} with{' '}
|
|
426
|
+
{newStrLines.length} line
|
|
427
|
+
{newStrLines.length > 1 ? 's' : ''}
|
|
428
|
+
</Text>
|
|
429
|
+
<Box flexDirection="column">
|
|
430
|
+
{contextBefore}
|
|
431
|
+
{removedLines}
|
|
432
|
+
{addedLines}
|
|
433
|
+
{contextAfter}
|
|
434
|
+
</Box>
|
|
435
|
+
</Box>
|
|
436
|
+
</Box>
|
|
437
|
+
);
|
|
438
|
+
|
|
439
|
+
return <ToolMessage message={messageContent} hideBox={true} />;
|
|
440
|
+
} catch (error) {
|
|
441
|
+
const errorContent = (
|
|
442
|
+
<Box flexDirection="column">
|
|
443
|
+
<Text color={themeColors.tool}>☕︎ string_replace</Text>
|
|
444
|
+
|
|
445
|
+
<Box>
|
|
446
|
+
<Text color={themeColors.secondary}>Path: </Text>
|
|
447
|
+
<Text color={themeColors.primary}>{path}</Text>
|
|
448
|
+
</Box>
|
|
449
|
+
|
|
450
|
+
<Box>
|
|
451
|
+
<Text color={themeColors.error}>Error: </Text>
|
|
452
|
+
<Text color={themeColors.error}>
|
|
453
|
+
{error instanceof Error ? error.message : String(error)}
|
|
454
|
+
</Text>
|
|
455
|
+
</Box>
|
|
456
|
+
</Box>
|
|
457
|
+
);
|
|
458
|
+
|
|
459
|
+
return <ToolMessage message={errorContent} hideBox={true} />;
|
|
460
|
+
}
|
|
461
|
+
}
|
|
462
|
+
|
|
463
|
+
|
|
464
|
+
|
|
465
|
+
const stringReplaceFormatter = async (
|
|
466
|
+
args: StringReplaceArgs,
|
|
467
|
+
result?: string,
|
|
468
|
+
): Promise<React.ReactElement> => {
|
|
469
|
+
const colors = getColors();
|
|
470
|
+
const { path, old_str, new_str } = args;
|
|
471
|
+
const absPath = resolve(path);
|
|
472
|
+
|
|
473
|
+
// VS Code preview logic removed
|
|
474
|
+
|
|
475
|
+
const preview = await formatStringReplacePreview(args, result, colors);
|
|
476
|
+
return <StringReplaceFormatter preview={preview} />;
|
|
477
|
+
};
|
|
478
|
+
|
|
479
|
+
const stringReplaceValidator = async (
|
|
480
|
+
args: StringReplaceArgs,
|
|
481
|
+
): Promise<{ valid: true } | { valid: false; error: string }> => {
|
|
482
|
+
const { path, old_str } = args;
|
|
483
|
+
|
|
484
|
+
// Check if file exists
|
|
485
|
+
const absPath = resolve(path);
|
|
486
|
+
try {
|
|
487
|
+
await access(absPath, constants.F_OK);
|
|
488
|
+
} catch (error) {
|
|
489
|
+
if (error && typeof error === 'object' && 'code' in error) {
|
|
490
|
+
if (error.code === 'ENOENT') {
|
|
491
|
+
return {
|
|
492
|
+
valid: false,
|
|
493
|
+
error: `☕︎ File "${path}" does not exist`,
|
|
494
|
+
};
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
498
|
+
return {
|
|
499
|
+
valid: false,
|
|
500
|
+
error: `☕︎ Cannot access file "${path}": ${errorMessage}`,
|
|
501
|
+
};
|
|
502
|
+
}
|
|
503
|
+
|
|
504
|
+
// Validate old_str is not empty
|
|
505
|
+
if (!old_str || old_str.length === 0) {
|
|
506
|
+
return {
|
|
507
|
+
valid: false,
|
|
508
|
+
error:
|
|
509
|
+
'☕︎ old_str cannot be empty. Provide the exact content to find and replace.',
|
|
510
|
+
};
|
|
511
|
+
}
|
|
512
|
+
|
|
513
|
+
// Check if content exists in file and is unique
|
|
514
|
+
try {
|
|
515
|
+
const cached = await getCachedFileContent(absPath);
|
|
516
|
+
const fileContent = cached.content;
|
|
517
|
+
const occurrences = fileContent.split(old_str).length - 1;
|
|
518
|
+
|
|
519
|
+
if (occurrences === 0) {
|
|
520
|
+
return {
|
|
521
|
+
valid: false,
|
|
522
|
+
error: `☕︎ Content not found in file. The file may have changed since you last read it.\n\nSearching for:\n${old_str}\n\nSuggestion: Read the file again to see current contents.`,
|
|
523
|
+
};
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
if (occurrences > 1) {
|
|
527
|
+
return {
|
|
528
|
+
valid: false,
|
|
529
|
+
error: `☕︎ Found ${occurrences} matches for the search string. Please provide more surrounding context to make the match unique.\n\nSearching for:\n${old_str}`,
|
|
530
|
+
};
|
|
531
|
+
}
|
|
532
|
+
} catch (error) {
|
|
533
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
534
|
+
return {
|
|
535
|
+
valid: false,
|
|
536
|
+
error: `☕︎ Error reading file "${path}": ${errorMessage}`,
|
|
537
|
+
};
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
return { valid: true };
|
|
541
|
+
};
|
|
542
|
+
|
|
543
|
+
export const stringReplaceTool = {
|
|
544
|
+
name: 'string_replace' as const,
|
|
545
|
+
tool: stringReplaceCoreTool,
|
|
546
|
+
formatter: stringReplaceFormatter,
|
|
547
|
+
validator: stringReplaceValidator,
|
|
548
|
+
};
|