@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,182 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import test from 'ava';
|
|
3
|
+
import {
|
|
4
|
+
isValidFilePath,
|
|
5
|
+
parseFileMentions,
|
|
6
|
+
parseLineRange,
|
|
7
|
+
resolveFilePath,
|
|
8
|
+
} from './file-mention-parser.js';
|
|
9
|
+
|
|
10
|
+
console.log(`\nfile-mention-parser.spec.ts`);
|
|
11
|
+
|
|
12
|
+
// Test parseFileMentions()
|
|
13
|
+
test('parses single file mention', t => {
|
|
14
|
+
const result = parseFileMentions('Check @app.tsx');
|
|
15
|
+
t.is(result.length, 1);
|
|
16
|
+
t.is(result[0].filePath, 'app.tsx');
|
|
17
|
+
t.is(result[0].rawText, '@app.tsx');
|
|
18
|
+
t.is(result[0].startIndex, 6);
|
|
19
|
+
t.is(result[0].endIndex, 14);
|
|
20
|
+
t.is(result[0].lineRange, undefined);
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
test('parses file with path', t => {
|
|
24
|
+
const result = parseFileMentions('Check @src/components/Button.tsx');
|
|
25
|
+
t.is(result.length, 1);
|
|
26
|
+
t.is(result[0].filePath, 'src/components/Button.tsx');
|
|
27
|
+
t.is(result[0].rawText, '@src/components/Button.tsx');
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
test('parses file with single line number', t => {
|
|
31
|
+
const result = parseFileMentions('@app.tsx:10');
|
|
32
|
+
t.is(result.length, 1);
|
|
33
|
+
t.is(result[0].filePath, 'app.tsx');
|
|
34
|
+
t.deepEqual(result[0].lineRange, {start: 10, end: undefined});
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
test('parses file with line range', t => {
|
|
38
|
+
const result = parseFileMentions('@app.tsx:10-20');
|
|
39
|
+
t.is(result.length, 1);
|
|
40
|
+
t.is(result[0].filePath, 'app.tsx');
|
|
41
|
+
t.deepEqual(result[0].lineRange, {start: 10, end: 20});
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
test('parses multiple file mentions', t => {
|
|
45
|
+
const result = parseFileMentions('Compare @a.ts and @b.ts');
|
|
46
|
+
t.is(result.length, 2);
|
|
47
|
+
t.is(result[0].filePath, 'a.ts');
|
|
48
|
+
t.is(result[1].filePath, 'b.ts');
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
test('parses file mentions with different line ranges', t => {
|
|
52
|
+
const result = parseFileMentions('Check @app.tsx:1-5 and @utils.ts:10-15');
|
|
53
|
+
t.is(result.length, 2);
|
|
54
|
+
t.deepEqual(result[0].lineRange, {start: 1, end: 5});
|
|
55
|
+
t.deepEqual(result[1].lineRange, {start: 10, end: 15});
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
test('handles text without file mentions', t => {
|
|
59
|
+
const result = parseFileMentions('This is just regular text');
|
|
60
|
+
t.is(result.length, 0);
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
test('ignores @ symbols in email addresses', t => {
|
|
64
|
+
const result = parseFileMentions('Email: user@example.com');
|
|
65
|
+
// Should not match email addresses (they have spaces or are at word boundaries)
|
|
66
|
+
// However, this might match "example.com" - let's verify behavior
|
|
67
|
+
t.true(result.length <= 1);
|
|
68
|
+
if (result.length === 1) {
|
|
69
|
+
t.is(result[0].filePath, 'example.com');
|
|
70
|
+
}
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
test('rejects invalid line ranges in file mentions', t => {
|
|
74
|
+
// Line range where end < start should not have lineRange
|
|
75
|
+
const result = parseFileMentions('@app.tsx:20-10');
|
|
76
|
+
t.is(result.length, 1);
|
|
77
|
+
t.is(result[0].lineRange, undefined);
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
test('rejects zero or negative line numbers', t => {
|
|
81
|
+
const result1 = parseFileMentions('@app.tsx:0');
|
|
82
|
+
t.is(result1[0].lineRange, undefined);
|
|
83
|
+
|
|
84
|
+
const result2 = parseFileMentions('@app.tsx:0-10');
|
|
85
|
+
t.is(result2[0].lineRange, undefined);
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
test('handles complex file paths', t => {
|
|
89
|
+
const result = parseFileMentions(
|
|
90
|
+
'@src/components/ui/Button/index.tsx:100-200',
|
|
91
|
+
);
|
|
92
|
+
t.is(result.length, 1);
|
|
93
|
+
t.is(result[0].filePath, 'src/components/ui/Button/index.tsx');
|
|
94
|
+
t.deepEqual(result[0].lineRange, {start: 100, end: 200});
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
// Test isValidFilePath()
|
|
98
|
+
test('accepts valid relative paths', t => {
|
|
99
|
+
t.true(isValidFilePath('app.tsx'));
|
|
100
|
+
t.true(isValidFilePath('src/app.tsx'));
|
|
101
|
+
t.true(isValidFilePath('src/components/Button.tsx'));
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
test('rejects directory traversal attempts', t => {
|
|
105
|
+
t.false(isValidFilePath('../../etc/passwd'));
|
|
106
|
+
t.false(isValidFilePath('../../../secret.txt'));
|
|
107
|
+
t.false(isValidFilePath('src/../../../etc/passwd'));
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
test('rejects absolute paths', t => {
|
|
111
|
+
t.false(isValidFilePath('/etc/passwd'));
|
|
112
|
+
t.false(isValidFilePath('/home/user/file.txt'));
|
|
113
|
+
t.false(isValidFilePath('C:\\Windows\\System32\\file.txt'));
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
test('rejects empty or whitespace paths', t => {
|
|
117
|
+
t.false(isValidFilePath(''));
|
|
118
|
+
t.false(isValidFilePath(' '));
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
test('rejects paths with null bytes', t => {
|
|
122
|
+
t.false(isValidFilePath('file.txt\0'));
|
|
123
|
+
t.false(isValidFilePath('file\0.txt'));
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
test('rejects paths starting with slashes', t => {
|
|
127
|
+
t.false(isValidFilePath('/file.txt'));
|
|
128
|
+
t.false(isValidFilePath('\\file.txt'));
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
// Test resolveFilePath()
|
|
132
|
+
test('resolves relative path to absolute', t => {
|
|
133
|
+
const cwd = '/home/user/project';
|
|
134
|
+
const resolved = resolveFilePath('src/app.tsx', cwd);
|
|
135
|
+
t.is(resolved, path.join(cwd, 'src/app.tsx'));
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
test('throws on invalid file paths', t => {
|
|
139
|
+
const cwd = '/home/user/project';
|
|
140
|
+
t.throws(() => resolveFilePath('../../etc/passwd', cwd), {
|
|
141
|
+
message: /Invalid file path/,
|
|
142
|
+
});
|
|
143
|
+
});
|
|
144
|
+
|
|
145
|
+
test('throws when resolved path escapes project directory', t => {
|
|
146
|
+
const cwd = '/home/user/project';
|
|
147
|
+
// Even if validation passes, resolve should catch escaping
|
|
148
|
+
t.throws(() => resolveFilePath('/etc/passwd', cwd), {
|
|
149
|
+
message: /Invalid file path/,
|
|
150
|
+
});
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
test('resolves nested paths correctly', t => {
|
|
154
|
+
const cwd = '/home/user/project';
|
|
155
|
+
const resolved = resolveFilePath('src/components/ui/Button.tsx', cwd);
|
|
156
|
+
t.is(resolved, path.join(cwd, 'src/components/ui/Button.tsx'));
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
// Test parseLineRange()
|
|
160
|
+
test('parses single line number', t => {
|
|
161
|
+
const result = parseLineRange('10');
|
|
162
|
+
t.deepEqual(result, {start: 10, end: undefined});
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
test('parses line range', t => {
|
|
166
|
+
const result = parseLineRange('10-20');
|
|
167
|
+
t.deepEqual(result, {start: 10, end: 20});
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
test('rejects invalid line ranges', t => {
|
|
171
|
+
t.is(parseLineRange('20-10'), null); // end < start
|
|
172
|
+
t.is(parseLineRange('0'), null); // zero
|
|
173
|
+
t.is(parseLineRange('-5'), null); // negative
|
|
174
|
+
t.is(parseLineRange('abc'), null); // non-numeric
|
|
175
|
+
t.is(parseLineRange(''), null); // empty
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
test('rejects malformed ranges', t => {
|
|
179
|
+
t.is(parseLineRange('10-20-30'), null); // too many parts
|
|
180
|
+
t.is(parseLineRange('10-'), null); // missing end
|
|
181
|
+
t.is(parseLineRange('-20'), null); // missing start
|
|
182
|
+
});
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Represents a parsed file mention from user input
|
|
5
|
+
* Supports:
|
|
6
|
+
* - @filename.ts
|
|
7
|
+
* - @src/components/Button.tsx
|
|
8
|
+
* - @file.ts:10-20 (line ranges)
|
|
9
|
+
* - @file.ts:10 (single line)
|
|
10
|
+
*/
|
|
11
|
+
interface FileMention {
|
|
12
|
+
rawText: string; // "@src/app.tsx:10-20"
|
|
13
|
+
filePath: string; // "src/app.tsx"
|
|
14
|
+
lineRange?: {
|
|
15
|
+
start: number;
|
|
16
|
+
end?: number; // undefined for single line
|
|
17
|
+
};
|
|
18
|
+
startIndex: number; // Position in input string
|
|
19
|
+
endIndex: number;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Regex pattern to match @file mentions
|
|
24
|
+
* Matches:
|
|
25
|
+
* - @file.ts
|
|
26
|
+
* - @src/path/to/file.tsx
|
|
27
|
+
* - @file.ts:10
|
|
28
|
+
* - @file.ts:10-20
|
|
29
|
+
*/
|
|
30
|
+
const FILE_MENTION_REGEX = /@([^\s:]+)(?::(\d+)(?:-(\d+))?)?/g;
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Parse all @mentions from user input
|
|
34
|
+
*/
|
|
35
|
+
export function parseFileMentions(input: string): FileMention[] {
|
|
36
|
+
const mentions: FileMention[] = [];
|
|
37
|
+
let match: RegExpExecArray | null;
|
|
38
|
+
|
|
39
|
+
// Reset regex state
|
|
40
|
+
FILE_MENTION_REGEX.lastIndex = 0;
|
|
41
|
+
|
|
42
|
+
while ((match = FILE_MENTION_REGEX.exec(input)) !== null) {
|
|
43
|
+
const rawText = match[0]; // Full match: "@src/app.tsx:10-20"
|
|
44
|
+
const filePath = match[1]; // Captured group 1: "src/app.tsx"
|
|
45
|
+
const lineStart = match[2]; // Captured group 2: "10"
|
|
46
|
+
const lineEnd = match[3]; // Captured group 3: "20"
|
|
47
|
+
|
|
48
|
+
// Skip if the file path is empty or invalid
|
|
49
|
+
if (!filePath || !isValidFilePath(filePath)) {
|
|
50
|
+
continue;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
const mention: FileMention = {
|
|
54
|
+
rawText,
|
|
55
|
+
filePath,
|
|
56
|
+
startIndex: match.index,
|
|
57
|
+
endIndex: match.index + rawText.length,
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
// Parse line range if present
|
|
61
|
+
if (lineStart) {
|
|
62
|
+
const start = parseInt(lineStart, 10);
|
|
63
|
+
const end = lineEnd ? parseInt(lineEnd, 10) : undefined;
|
|
64
|
+
|
|
65
|
+
// Validate line numbers
|
|
66
|
+
if (start > 0 && (!end || end >= start)) {
|
|
67
|
+
mention.lineRange = {start, end};
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
mentions.push(mention);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
return mentions;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Validate file path to prevent directory traversal attacks
|
|
79
|
+
* and ensure it's within the project directory
|
|
80
|
+
*/
|
|
81
|
+
export function isValidFilePath(filePath: string): boolean {
|
|
82
|
+
// Reject empty paths
|
|
83
|
+
if (!filePath || filePath.trim().length === 0) {
|
|
84
|
+
return false;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
// Reject paths that try to escape parent directories
|
|
88
|
+
if (filePath.includes('..')) {
|
|
89
|
+
return false;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Reject absolute paths (outside project)
|
|
93
|
+
if (path.isAbsolute(filePath)) {
|
|
94
|
+
return false;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// Reject Windows absolute paths (C:\, D:\, etc.) even on Unix systems
|
|
98
|
+
if (/^[A-Za-z]:[/\\]/.test(filePath)) {
|
|
99
|
+
return false;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
// Reject paths with null bytes (security)
|
|
103
|
+
if (filePath.includes('\0')) {
|
|
104
|
+
return false;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// Reject paths that start with special characters that could be problematic
|
|
108
|
+
if (filePath.startsWith('/') || filePath.startsWith('\\')) {
|
|
109
|
+
return false;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
return true;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Resolve a relative file path to an absolute path within the project
|
|
117
|
+
*/
|
|
118
|
+
export function resolveFilePath(filePath: string, cwd: string): string {
|
|
119
|
+
// Validate first
|
|
120
|
+
if (!isValidFilePath(filePath)) {
|
|
121
|
+
throw new Error(`Invalid file path: ${filePath}`);
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
// Resolve to absolute path
|
|
125
|
+
const absolutePath = path.resolve(cwd, filePath);
|
|
126
|
+
|
|
127
|
+
// Ensure the resolved path is still within the project directory
|
|
128
|
+
const normalizedCwd = path.resolve(cwd);
|
|
129
|
+
if (!absolutePath.startsWith(normalizedCwd)) {
|
|
130
|
+
throw new Error(
|
|
131
|
+
`File path escapes project directory: ${filePath} -> ${absolutePath}`,
|
|
132
|
+
);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
return absolutePath;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Parse line range from a string like "10-20" or "10"
|
|
140
|
+
*/
|
|
141
|
+
export function parseLineRange(
|
|
142
|
+
rangeStr: string,
|
|
143
|
+
): {start: number; end?: number} | null {
|
|
144
|
+
if (!rangeStr) {
|
|
145
|
+
return null;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
const parts = rangeStr.split('-');
|
|
149
|
+
|
|
150
|
+
if (parts.length === 1) {
|
|
151
|
+
// Single line: "10"
|
|
152
|
+
const line = parseInt(parts[0], 10);
|
|
153
|
+
if (isNaN(line) || line <= 0) {
|
|
154
|
+
return null;
|
|
155
|
+
}
|
|
156
|
+
return {start: line, end: undefined};
|
|
157
|
+
} else if (parts.length === 2) {
|
|
158
|
+
// Range: "10-20"
|
|
159
|
+
const start = parseInt(parts[0], 10);
|
|
160
|
+
const end = parseInt(parts[1], 10);
|
|
161
|
+
|
|
162
|
+
if (isNaN(start) || isNaN(end) || start <= 0 || end < start) {
|
|
163
|
+
return null;
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
return {start, end};
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
return null;
|
|
170
|
+
}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import test from 'ava';
|
|
2
|
+
import {fuzzyScore, fuzzyScoreFilePath} from './fuzzy-matching';
|
|
3
|
+
|
|
4
|
+
console.log(`\nfuzzy-matching.spec.ts`);
|
|
5
|
+
|
|
6
|
+
// Tests for fuzzyScore (general text matching)
|
|
7
|
+
test('fuzzyScore - exact match returns highest score', t => {
|
|
8
|
+
t.is(fuzzyScore('help', 'help'), 1000);
|
|
9
|
+
t.is(fuzzyScore('status', 'status'), 1000);
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
test('fuzzyScore - case insensitive matching', t => {
|
|
13
|
+
t.is(fuzzyScore('Help', 'help'), 1000);
|
|
14
|
+
t.is(fuzzyScore('STATUS', 'status'), 1000);
|
|
15
|
+
t.is(fuzzyScore('help', 'HELP'), 1000);
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
test('fuzzyScore - prefix match', t => {
|
|
19
|
+
const score = fuzzyScore('help', 'hel');
|
|
20
|
+
t.true(score > 0);
|
|
21
|
+
t.true(score >= 800);
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
test('fuzzyScore - substring match', t => {
|
|
25
|
+
const score = fuzzyScore('status', 'tat');
|
|
26
|
+
t.true(score > 0);
|
|
27
|
+
t.true(score >= 700);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
test('fuzzyScore - sequential character match', t => {
|
|
31
|
+
const score = fuzzyScore('provider', 'pvdr');
|
|
32
|
+
t.true(score > 0);
|
|
33
|
+
t.true(score < 700); // Should be lower than substring match
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
test('fuzzyScore - no match returns 0', t => {
|
|
37
|
+
t.is(fuzzyScore('help', 'xyz'), 0);
|
|
38
|
+
t.is(fuzzyScore('status', 'abc'), 0);
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
test('fuzzyScore - empty query returns 0', t => {
|
|
42
|
+
t.is(fuzzyScore('help', ''), 0);
|
|
43
|
+
t.is(fuzzyScore('anything', ''), 0);
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
test('fuzzyScore - ranks exact match higher than prefix', t => {
|
|
47
|
+
const exactScore = fuzzyScore('help', 'help');
|
|
48
|
+
const prefixScore = fuzzyScore('helper', 'help');
|
|
49
|
+
t.true(exactScore > prefixScore);
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
test('fuzzyScore - ranks prefix higher than substring', t => {
|
|
53
|
+
const prefixScore = fuzzyScore('helper', 'help');
|
|
54
|
+
const substringScore = fuzzyScore('unhelpful', 'help');
|
|
55
|
+
t.true(prefixScore > substringScore);
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
test('fuzzyScore - ranks substring higher than fuzzy', t => {
|
|
59
|
+
const substringScore = fuzzyScore('helper', 'elp');
|
|
60
|
+
const fuzzyScore1 = fuzzyScore('enable-lp', 'elp');
|
|
61
|
+
t.true(substringScore > fuzzyScore1);
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
// Tests for fuzzyScoreFilePath (file path specific matching)
|
|
65
|
+
test('fuzzyScoreFilePath - exact filename match scores high', t => {
|
|
66
|
+
const score = fuzzyScoreFilePath('src/utils/helper.ts', 'helper.ts');
|
|
67
|
+
t.true(score >= 900);
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
test('fuzzyScoreFilePath - filename prefix scores higher than path prefix', t => {
|
|
71
|
+
const filenameScore = fuzzyScoreFilePath('src/utils/helper.ts', 'help');
|
|
72
|
+
const pathScore = fuzzyScoreFilePath('helper/src/utils/file.ts', 'help');
|
|
73
|
+
t.true(filenameScore > pathScore);
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
test('fuzzyScoreFilePath - filename substring matches correctly', t => {
|
|
77
|
+
const filenameScore = fuzzyScoreFilePath('src/utils/my-helper.ts', 'help');
|
|
78
|
+
const pathScore = fuzzyScoreFilePath('helper/src/utils/file.ts', 'help');
|
|
79
|
+
// Both should have scores, but path prefix might score higher than filename substring
|
|
80
|
+
t.true(filenameScore > 0);
|
|
81
|
+
t.true(pathScore > 0);
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
test('fuzzyScoreFilePath - handles paths with multiple slashes', t => {
|
|
85
|
+
const score = fuzzyScoreFilePath('a/b/c/d/file.ts', 'file');
|
|
86
|
+
t.true(score > 0);
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
test('fuzzyScoreFilePath - case insensitive for paths', t => {
|
|
90
|
+
const score1 = fuzzyScoreFilePath('src/Utils/Helper.ts', 'helper');
|
|
91
|
+
const score2 = fuzzyScoreFilePath('src/utils/helper.ts', 'HELPER');
|
|
92
|
+
t.true(score1 > 0);
|
|
93
|
+
t.true(score2 > 0);
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
// Command-specific test scenarios
|
|
97
|
+
test('command matching - /h matches help, status does not', t => {
|
|
98
|
+
const helpScore = fuzzyScore('help', 'h');
|
|
99
|
+
const statusScore = fuzzyScore('status', 'h');
|
|
100
|
+
t.true(helpScore > 0);
|
|
101
|
+
t.is(statusScore, 0);
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
test('command matching - /st matches status', t => {
|
|
105
|
+
const score = fuzzyScore('status', 'st');
|
|
106
|
+
t.true(score > 0);
|
|
107
|
+
t.true(score >= 800); // Should be prefix match
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
test('command matching - /mod matches model', t => {
|
|
111
|
+
const modelScore = fuzzyScore('model', 'mod');
|
|
112
|
+
const recommendationsScore = fuzzyScore('recommendations', 'mod');
|
|
113
|
+
|
|
114
|
+
t.true(modelScore > 0);
|
|
115
|
+
// recommendations doesn't contain 'mod' as substring, so it should be 0 or fuzzy match
|
|
116
|
+
// model should score higher (prefix match)
|
|
117
|
+
t.true(modelScore > recommendationsScore);
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
test('command matching - fuzzy matching for /pvd matches provider', t => {
|
|
121
|
+
const score = fuzzyScore('provider', 'pvd');
|
|
122
|
+
t.true(score > 0);
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
test('command matching - /init matches init higher than initialization', t => {
|
|
126
|
+
const initScore = fuzzyScore('init', 'init');
|
|
127
|
+
const initializationScore = fuzzyScore('initialization', 'init');
|
|
128
|
+
|
|
129
|
+
t.true(initScore > initializationScore);
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
// Edge cases
|
|
133
|
+
test('fuzzyScore - single character queries work', t => {
|
|
134
|
+
const score1 = fuzzyScore('help', 'h');
|
|
135
|
+
const score2 = fuzzyScore('status', 's');
|
|
136
|
+
t.true(score1 > 0);
|
|
137
|
+
t.true(score2 > 0);
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
test('fuzzyScore - longer text with short query', t => {
|
|
141
|
+
const score = fuzzyScore('very-long-command-name-here', 'vlc');
|
|
142
|
+
t.true(score > 0);
|
|
143
|
+
});
|
|
144
|
+
|
|
145
|
+
test('fuzzyScoreFilePath - works with Windows-style paths', t => {
|
|
146
|
+
// While the function uses /, testing that it handles edge cases
|
|
147
|
+
const score = fuzzyScoreFilePath('src\\utils\\file.ts', 'file');
|
|
148
|
+
t.true(score > 0);
|
|
149
|
+
});
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Fuzzy match scoring algorithm
|
|
3
|
+
* Returns a score from 0 to 1000 (higher = better match)
|
|
4
|
+
*
|
|
5
|
+
* This can be used for matching file paths, command names, or any other strings.
|
|
6
|
+
*/
|
|
7
|
+
export function fuzzyScore(text: string, query: string): number {
|
|
8
|
+
if (!query) {
|
|
9
|
+
return 0;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
const lowerText = text.toLowerCase();
|
|
13
|
+
const lowerQuery = query.toLowerCase();
|
|
14
|
+
|
|
15
|
+
// Exact match (highest score)
|
|
16
|
+
if (lowerText === lowerQuery) {
|
|
17
|
+
return 1000;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
// Text ends with query
|
|
21
|
+
if (lowerText.endsWith(lowerQuery)) {
|
|
22
|
+
return 850;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
// Text starts with query
|
|
26
|
+
if (lowerText.startsWith(lowerQuery)) {
|
|
27
|
+
return 800;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
// Text contains query as substring
|
|
31
|
+
if (lowerText.includes(lowerQuery)) {
|
|
32
|
+
return 700;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// Sequential character match (fuzzy)
|
|
36
|
+
// All query characters appear in order in the text
|
|
37
|
+
let textIndex = 0;
|
|
38
|
+
let queryIndex = 0;
|
|
39
|
+
let lastMatchIndex = -1;
|
|
40
|
+
let consecutiveMatches = 0;
|
|
41
|
+
|
|
42
|
+
while (textIndex < lowerText.length && queryIndex < lowerQuery.length) {
|
|
43
|
+
if (lowerText[textIndex] === lowerQuery[queryIndex]) {
|
|
44
|
+
// Bonus for consecutive matches
|
|
45
|
+
if (textIndex === lastMatchIndex + 1) {
|
|
46
|
+
consecutiveMatches++;
|
|
47
|
+
} else {
|
|
48
|
+
consecutiveMatches = 1;
|
|
49
|
+
}
|
|
50
|
+
lastMatchIndex = textIndex;
|
|
51
|
+
queryIndex++;
|
|
52
|
+
}
|
|
53
|
+
textIndex++;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// If all query characters matched
|
|
57
|
+
if (queryIndex === lowerQuery.length) {
|
|
58
|
+
// Score based on match density and consecutive matches
|
|
59
|
+
const matchDensity = lowerQuery.length / lowerText.length;
|
|
60
|
+
const consecutiveBonus = consecutiveMatches * 50;
|
|
61
|
+
return Math.min(500 + matchDensity * 100 + consecutiveBonus, 699);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
// No match
|
|
65
|
+
return 0;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
/**
|
|
69
|
+
* Fuzzy score specifically for file paths
|
|
70
|
+
* Gives higher priority to filename matches over directory matches
|
|
71
|
+
*/
|
|
72
|
+
export function fuzzyScoreFilePath(filePath: string, query: string): number {
|
|
73
|
+
if (!query) {
|
|
74
|
+
return 0;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const lowerPath = filePath.toLowerCase();
|
|
78
|
+
const lowerQuery = query.toLowerCase();
|
|
79
|
+
|
|
80
|
+
// Exact match (highest score)
|
|
81
|
+
if (lowerPath === lowerQuery) {
|
|
82
|
+
return 1000;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Exact match of filename (without path)
|
|
86
|
+
const filename = filePath.split('/').pop() || '';
|
|
87
|
+
if (filename.toLowerCase() === lowerQuery) {
|
|
88
|
+
return 900;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
// Path ends with query
|
|
92
|
+
if (lowerPath.endsWith(lowerQuery)) {
|
|
93
|
+
return 850;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
// Filename starts with query
|
|
97
|
+
if (filename.toLowerCase().startsWith(lowerQuery)) {
|
|
98
|
+
return 800;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// Path starts with query
|
|
102
|
+
if (lowerPath.startsWith(lowerQuery)) {
|
|
103
|
+
return 750;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
// Filename contains query as substring
|
|
107
|
+
if (filename.toLowerCase().includes(lowerQuery)) {
|
|
108
|
+
return 700;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// Path contains query as substring
|
|
112
|
+
if (lowerPath.includes(lowerQuery)) {
|
|
113
|
+
return 600;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
// Sequential character match (fuzzy)
|
|
117
|
+
let pathIndex = 0;
|
|
118
|
+
let queryIndex = 0;
|
|
119
|
+
let lastMatchIndex = -1;
|
|
120
|
+
let consecutiveMatches = 0;
|
|
121
|
+
|
|
122
|
+
while (pathIndex < lowerPath.length && queryIndex < lowerQuery.length) {
|
|
123
|
+
if (lowerPath[pathIndex] === lowerQuery[queryIndex]) {
|
|
124
|
+
// Bonus for consecutive matches
|
|
125
|
+
if (pathIndex === lastMatchIndex + 1) {
|
|
126
|
+
consecutiveMatches++;
|
|
127
|
+
} else {
|
|
128
|
+
consecutiveMatches = 1;
|
|
129
|
+
}
|
|
130
|
+
lastMatchIndex = pathIndex;
|
|
131
|
+
queryIndex++;
|
|
132
|
+
}
|
|
133
|
+
pathIndex++;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// If all query characters matched
|
|
137
|
+
if (queryIndex === lowerQuery.length) {
|
|
138
|
+
// Score based on match density and consecutive matches
|
|
139
|
+
const matchDensity = lowerQuery.length / lowerPath.length;
|
|
140
|
+
const consecutiveBonus = consecutiveMatches * 50;
|
|
141
|
+
return Math.min(500 + matchDensity * 100 + consecutiveBonus, 599);
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// No match
|
|
145
|
+
return 0;
|
|
146
|
+
}
|