@gguf/coder 0.3.1 → 0.3.3
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/package.json +5 -2
- package/.editorconfig +0 -16
- package/.env.example +0 -63
- package/.gitattributes +0 -1
- package/.semgrepignore +0 -19
- package/coder-dummy-file.ts +0 -52
- package/coder.config.example.json +0 -59
- package/coder.config.json +0 -13
- package/color_picker.html +0 -36
- package/scripts/extract-changelog.js +0 -73
- package/scripts/fetch-models.js +0 -143
- package/scripts/test.sh +0 -40
- package/scripts/update-homebrew-formula.sh +0 -125
- package/scripts/update-nix-version.sh +0 -157
- package/source/ai-sdk-client/AISDKClient.spec.ts +0 -117
- package/source/ai-sdk-client/AISDKClient.ts +0 -155
- package/source/ai-sdk-client/chat/chat-handler.spec.ts +0 -121
- package/source/ai-sdk-client/chat/chat-handler.ts +0 -276
- package/source/ai-sdk-client/chat/streaming-handler.spec.ts +0 -173
- package/source/ai-sdk-client/chat/streaming-handler.ts +0 -110
- package/source/ai-sdk-client/chat/tool-processor.spec.ts +0 -92
- package/source/ai-sdk-client/chat/tool-processor.ts +0 -70
- package/source/ai-sdk-client/converters/message-converter.spec.ts +0 -220
- package/source/ai-sdk-client/converters/message-converter.ts +0 -113
- package/source/ai-sdk-client/converters/tool-converter.spec.ts +0 -90
- package/source/ai-sdk-client/converters/tool-converter.ts +0 -46
- package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +0 -55
- package/source/ai-sdk-client/error-handling/error-extractor.ts +0 -15
- package/source/ai-sdk-client/error-handling/error-parser.spec.ts +0 -169
- package/source/ai-sdk-client/error-handling/error-parser.ts +0 -161
- package/source/ai-sdk-client/index.ts +0 -7
- package/source/ai-sdk-client/providers/provider-factory.spec.ts +0 -71
- package/source/ai-sdk-client/providers/provider-factory.ts +0 -41
- package/source/ai-sdk-client/types.ts +0 -9
- package/source/ai-sdk-client-empty-message.spec.ts +0 -141
- package/source/ai-sdk-client-error-handling.spec.ts +0 -186
- package/source/ai-sdk-client-maxretries.spec.ts +0 -114
- package/source/ai-sdk-client-preparestep.spec.ts +0 -279
- package/source/app/App.spec.tsx +0 -32
- package/source/app/App.tsx +0 -480
- package/source/app/components/AppContainer.spec.tsx +0 -96
- package/source/app/components/AppContainer.tsx +0 -56
- package/source/app/components/ChatInterface.spec.tsx +0 -163
- package/source/app/components/ChatInterface.tsx +0 -144
- package/source/app/components/ModalSelectors.spec.tsx +0 -141
- package/source/app/components/ModalSelectors.tsx +0 -135
- package/source/app/helpers.spec.ts +0 -97
- package/source/app/helpers.ts +0 -63
- package/source/app/index.ts +0 -4
- package/source/app/types.ts +0 -39
- package/source/app/utils/appUtils.ts +0 -294
- package/source/app/utils/conversationState.ts +0 -310
- package/source/app.spec.tsx +0 -244
- package/source/cli.spec.ts +0 -73
- package/source/cli.tsx +0 -51
- package/source/client-factory.spec.ts +0 -48
- package/source/client-factory.ts +0 -178
- package/source/command-parser.spec.ts +0 -127
- package/source/command-parser.ts +0 -36
- package/source/commands/checkpoint.spec.tsx +0 -277
- package/source/commands/checkpoint.tsx +0 -366
- package/source/commands/clear.tsx +0 -22
- package/source/commands/custom-commands.tsx +0 -121
- package/source/commands/exit.ts +0 -21
- package/source/commands/export.spec.tsx +0 -131
- package/source/commands/export.tsx +0 -79
- package/source/commands/help.tsx +0 -120
- package/source/commands/index.ts +0 -17
- package/source/commands/init.tsx +0 -339
- package/source/commands/lsp-command.spec.tsx +0 -281
- package/source/commands/lsp.tsx +0 -120
- package/source/commands/mcp-command.spec.tsx +0 -313
- package/source/commands/mcp.tsx +0 -162
- package/source/commands/model-database.spec.tsx +0 -758
- package/source/commands/model-database.tsx +0 -418
- package/source/commands/model.ts +0 -12
- package/source/commands/provider.ts +0 -12
- package/source/commands/setup-config.tsx +0 -16
- package/source/commands/simple-commands.spec.tsx +0 -175
- package/source/commands/status.ts +0 -12
- package/source/commands/theme.ts +0 -12
- package/source/commands/update.spec.tsx +0 -261
- package/source/commands/update.tsx +0 -201
- package/source/commands/usage.spec.tsx +0 -495
- package/source/commands/usage.tsx +0 -100
- package/source/commands.spec.ts +0 -436
- package/source/commands.ts +0 -83
- package/source/components/assistant-message.spec.tsx +0 -796
- package/source/components/assistant-message.tsx +0 -34
- package/source/components/bash-execution-indicator.tsx +0 -21
- package/source/components/cancelling-indicator.tsx +0 -16
- package/source/components/chat-queue.spec.tsx +0 -83
- package/source/components/chat-queue.tsx +0 -36
- package/source/components/checkpoint-display.spec.tsx +0 -219
- package/source/components/checkpoint-display.tsx +0 -126
- package/source/components/checkpoint-selector.spec.tsx +0 -173
- package/source/components/checkpoint-selector.tsx +0 -173
- package/source/components/development-mode-indicator.spec.tsx +0 -268
- package/source/components/development-mode-indicator.tsx +0 -38
- package/source/components/message-box.spec.tsx +0 -427
- package/source/components/message-box.tsx +0 -87
- package/source/components/model-selector.tsx +0 -132
- package/source/components/provider-selector.tsx +0 -75
- package/source/components/random-spinner.tsx +0 -19
- package/source/components/security-disclaimer.tsx +0 -73
- package/source/components/status-connection-display.spec.tsx +0 -133
- package/source/components/status.tsx +0 -267
- package/source/components/theme-selector.tsx +0 -126
- package/source/components/tool-confirmation.tsx +0 -190
- package/source/components/tool-execution-indicator.tsx +0 -33
- package/source/components/tool-message.tsx +0 -85
- package/source/components/ui/titled-box.spec.tsx +0 -207
- package/source/components/ui/titled-box.tsx +0 -57
- package/source/components/usage/progress-bar.spec.tsx +0 -398
- package/source/components/usage/progress-bar.tsx +0 -30
- package/source/components/usage/usage-display.spec.tsx +0 -780
- package/source/components/usage/usage-display.tsx +0 -291
- package/source/components/user-input.spec.tsx +0 -327
- package/source/components/user-input.tsx +0 -533
- package/source/components/user-message.spec.tsx +0 -230
- package/source/components/user-message.tsx +0 -84
- package/source/components/welcome-message.tsx +0 -76
- package/source/config/env-substitution.ts +0 -65
- package/source/config/index.spec.ts +0 -171
- package/source/config/index.ts +0 -154
- package/source/config/paths.spec.ts +0 -241
- package/source/config/paths.ts +0 -55
- package/source/config/preferences.ts +0 -51
- package/source/config/themes.ts +0 -315
- package/source/constants.ts +0 -130
- package/source/context/mode-context.spec.ts +0 -79
- package/source/context/mode-context.ts +0 -24
- package/source/custom-commands/executor.spec.ts +0 -142
- package/source/custom-commands/executor.ts +0 -64
- package/source/custom-commands/loader.spec.ts +0 -314
- package/source/custom-commands/loader.ts +0 -153
- package/source/custom-commands/parser.ts +0 -196
- package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +0 -39
- package/source/hooks/chat-handler/conversation/conversation-loop.tsx +0 -511
- package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +0 -50
- package/source/hooks/chat-handler/conversation/tool-executor.tsx +0 -109
- package/source/hooks/chat-handler/index.ts +0 -12
- package/source/hooks/chat-handler/state/streaming-state.spec.ts +0 -26
- package/source/hooks/chat-handler/state/streaming-state.ts +0 -19
- package/source/hooks/chat-handler/types.ts +0 -38
- package/source/hooks/chat-handler/useChatHandler.spec.tsx +0 -321
- package/source/hooks/chat-handler/useChatHandler.tsx +0 -194
- package/source/hooks/chat-handler/utils/context-checker.spec.ts +0 -60
- package/source/hooks/chat-handler/utils/context-checker.tsx +0 -73
- package/source/hooks/chat-handler/utils/message-helpers.spec.ts +0 -42
- package/source/hooks/chat-handler/utils/message-helpers.tsx +0 -36
- package/source/hooks/chat-handler/utils/tool-filters.spec.ts +0 -109
- package/source/hooks/chat-handler/utils/tool-filters.ts +0 -64
- package/source/hooks/useAppHandlers.tsx +0 -291
- package/source/hooks/useAppInitialization.tsx +0 -422
- package/source/hooks/useAppState.tsx +0 -311
- package/source/hooks/useDirectoryTrust.tsx +0 -98
- package/source/hooks/useInputState.ts +0 -414
- package/source/hooks/useModeHandlers.tsx +0 -302
- package/source/hooks/useNonInteractiveMode.ts +0 -140
- package/source/hooks/useTerminalWidth.tsx +0 -81
- package/source/hooks/useTheme.ts +0 -18
- package/source/hooks/useToolHandler.tsx +0 -349
- package/source/hooks/useUIState.ts +0 -61
- package/source/init/agents-template-generator.ts +0 -421
- package/source/init/existing-rules-extractor.ts +0 -319
- package/source/init/file-scanner.spec.ts +0 -227
- package/source/init/file-scanner.ts +0 -238
- package/source/init/framework-detector.ts +0 -382
- package/source/init/language-detector.ts +0 -269
- package/source/init/project-analyzer.spec.ts +0 -231
- package/source/init/project-analyzer.ts +0 -458
- package/source/lsp/index.ts +0 -31
- package/source/lsp/lsp-client.spec.ts +0 -508
- package/source/lsp/lsp-client.ts +0 -487
- package/source/lsp/lsp-manager.spec.ts +0 -477
- package/source/lsp/lsp-manager.ts +0 -419
- package/source/lsp/protocol.spec.ts +0 -502
- package/source/lsp/protocol.ts +0 -360
- package/source/lsp/server-discovery.spec.ts +0 -654
- package/source/lsp/server-discovery.ts +0 -515
- package/source/markdown-parser/html-entities.spec.ts +0 -88
- package/source/markdown-parser/html-entities.ts +0 -45
- package/source/markdown-parser/index.spec.ts +0 -281
- package/source/markdown-parser/index.ts +0 -126
- package/source/markdown-parser/table-parser.spec.ts +0 -133
- package/source/markdown-parser/table-parser.ts +0 -114
- package/source/markdown-parser/utils.spec.ts +0 -70
- package/source/markdown-parser/utils.ts +0 -13
- package/source/mcp/mcp-client.spec.ts +0 -81
- package/source/mcp/mcp-client.ts +0 -625
- package/source/mcp/transport-factory.spec.ts +0 -406
- package/source/mcp/transport-factory.ts +0 -312
- package/source/message-handler.ts +0 -67
- package/source/model-database/database-engine.spec.ts +0 -494
- package/source/model-database/database-engine.ts +0 -50
- package/source/model-database/model-database.spec.ts +0 -363
- package/source/model-database/model-database.ts +0 -91
- package/source/model-database/model-engine.spec.ts +0 -447
- package/source/model-database/model-engine.ts +0 -65
- package/source/model-database/model-fetcher.spec.ts +0 -583
- package/source/model-database/model-fetcher.ts +0 -330
- package/source/models/index.ts +0 -1
- package/source/models/models-cache.spec.ts +0 -214
- package/source/models/models-cache.ts +0 -78
- package/source/models/models-dev-client.spec.ts +0 -379
- package/source/models/models-dev-client.ts +0 -329
- package/source/models/models-types.ts +0 -68
- package/source/prompt-history.ts +0 -155
- package/source/security/command-injection.spec.ts +0 -240
- package/source/services/checkpoint-manager.spec.ts +0 -523
- package/source/services/checkpoint-manager.ts +0 -466
- package/source/services/file-snapshot.spec.ts +0 -569
- package/source/services/file-snapshot.ts +0 -220
- package/source/test-utils/render-with-theme.tsx +0 -48
- package/source/tokenization/index.ts +0 -1
- package/source/tokenization/tokenizer-factory.spec.ts +0 -170
- package/source/tokenization/tokenizer-factory.ts +0 -125
- package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +0 -200
- package/source/tokenization/tokenizers/anthropic-tokenizer.ts +0 -43
- package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +0 -236
- package/source/tokenization/tokenizers/fallback-tokenizer.ts +0 -26
- package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +0 -224
- package/source/tokenization/tokenizers/llama-tokenizer.ts +0 -41
- package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +0 -184
- package/source/tokenization/tokenizers/openai-tokenizer.ts +0 -57
- package/source/tool-calling/index.ts +0 -5
- package/source/tool-calling/json-parser.spec.ts +0 -639
- package/source/tool-calling/json-parser.ts +0 -247
- package/source/tool-calling/tool-parser.spec.ts +0 -395
- package/source/tool-calling/tool-parser.ts +0 -120
- package/source/tool-calling/xml-parser.spec.ts +0 -662
- package/source/tool-calling/xml-parser.ts +0 -289
- package/source/tools/execute-bash.spec.tsx +0 -353
- package/source/tools/execute-bash.tsx +0 -219
- package/source/tools/execute-function.spec.ts +0 -130
- package/source/tools/fetch-url.spec.tsx +0 -342
- package/source/tools/fetch-url.tsx +0 -172
- package/source/tools/find-files.spec.tsx +0 -924
- package/source/tools/find-files.tsx +0 -293
- package/source/tools/index.ts +0 -102
- package/source/tools/lsp-get-diagnostics.tsx +0 -192
- package/source/tools/needs-approval.spec.ts +0 -282
- package/source/tools/read-file.spec.tsx +0 -801
- package/source/tools/read-file.tsx +0 -387
- package/source/tools/search-file-contents.spec.tsx +0 -1273
- package/source/tools/search-file-contents.tsx +0 -293
- package/source/tools/string-replace.spec.tsx +0 -730
- package/source/tools/string-replace.tsx +0 -548
- package/source/tools/tool-manager.ts +0 -210
- package/source/tools/tool-registry.spec.ts +0 -415
- package/source/tools/tool-registry.ts +0 -228
- package/source/tools/web-search.tsx +0 -223
- package/source/tools/write-file.spec.tsx +0 -559
- package/source/tools/write-file.tsx +0 -228
- package/source/types/app.ts +0 -37
- package/source/types/checkpoint.ts +0 -48
- package/source/types/commands.ts +0 -46
- package/source/types/components.ts +0 -27
- package/source/types/config.ts +0 -103
- package/source/types/core-connection-status.spec.ts +0 -67
- package/source/types/core.ts +0 -181
- package/source/types/hooks.ts +0 -50
- package/source/types/index.ts +0 -12
- package/source/types/markdown-parser.ts +0 -11
- package/source/types/mcp.ts +0 -52
- package/source/types/system.ts +0 -16
- package/source/types/tokenization.ts +0 -41
- package/source/types/ui.ts +0 -40
- package/source/types/usage.ts +0 -58
- package/source/types/utils.ts +0 -16
- package/source/usage/calculator.spec.ts +0 -385
- package/source/usage/calculator.ts +0 -104
- package/source/usage/storage.spec.ts +0 -703
- package/source/usage/storage.ts +0 -238
- package/source/usage/tracker.spec.ts +0 -456
- package/source/usage/tracker.ts +0 -102
- package/source/utils/atomic-deletion.spec.ts +0 -194
- package/source/utils/atomic-deletion.ts +0 -127
- package/source/utils/bounded-map.spec.ts +0 -300
- package/source/utils/bounded-map.ts +0 -193
- package/source/utils/checkpoint-utils.spec.ts +0 -222
- package/source/utils/checkpoint-utils.ts +0 -92
- package/source/utils/error-formatter.spec.ts +0 -169
- package/source/utils/error-formatter.ts +0 -194
- package/source/utils/file-autocomplete.spec.ts +0 -173
- package/source/utils/file-autocomplete.ts +0 -196
- package/source/utils/file-cache.spec.ts +0 -309
- package/source/utils/file-cache.ts +0 -195
- package/source/utils/file-content-loader.spec.ts +0 -180
- package/source/utils/file-content-loader.ts +0 -179
- package/source/utils/file-mention-handler.spec.ts +0 -261
- package/source/utils/file-mention-handler.ts +0 -84
- package/source/utils/file-mention-parser.spec.ts +0 -182
- package/source/utils/file-mention-parser.ts +0 -170
- package/source/utils/fuzzy-matching.spec.ts +0 -149
- package/source/utils/fuzzy-matching.ts +0 -146
- package/source/utils/indentation-normalizer.spec.ts +0 -216
- package/source/utils/indentation-normalizer.ts +0 -76
- package/source/utils/installation-detector.spec.ts +0 -178
- package/source/utils/installation-detector.ts +0 -153
- package/source/utils/logging/config.spec.ts +0 -311
- package/source/utils/logging/config.ts +0 -210
- package/source/utils/logging/console-facade.spec.ts +0 -184
- package/source/utils/logging/console-facade.ts +0 -384
- package/source/utils/logging/correlation.spec.ts +0 -679
- package/source/utils/logging/correlation.ts +0 -474
- package/source/utils/logging/formatters.spec.ts +0 -464
- package/source/utils/logging/formatters.ts +0 -207
- package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +0 -93
- package/source/utils/logging/health-monitor/alerts/alert-manager.ts +0 -79
- package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +0 -56
- package/source/utils/logging/health-monitor/checks/configuration-check.ts +0 -43
- package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +0 -56
- package/source/utils/logging/health-monitor/checks/logging-check.ts +0 -58
- package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +0 -100
- package/source/utils/logging/health-monitor/checks/memory-check.ts +0 -78
- package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +0 -56
- package/source/utils/logging/health-monitor/checks/performance-check.ts +0 -56
- package/source/utils/logging/health-monitor/checks/request-check.spec.ts +0 -56
- package/source/utils/logging/health-monitor/checks/request-check.ts +0 -76
- package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +0 -70
- package/source/utils/logging/health-monitor/core/health-check-runner.ts +0 -138
- package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +0 -58
- package/source/utils/logging/health-monitor/core/health-monitor.ts +0 -344
- package/source/utils/logging/health-monitor/core/scoring.spec.ts +0 -65
- package/source/utils/logging/health-monitor/core/scoring.ts +0 -91
- package/source/utils/logging/health-monitor/index.ts +0 -15
- package/source/utils/logging/health-monitor/instances.ts +0 -48
- package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +0 -141
- package/source/utils/logging/health-monitor/middleware/http-middleware.ts +0 -75
- package/source/utils/logging/health-monitor/types.ts +0 -126
- package/source/utils/logging/index.spec.ts +0 -284
- package/source/utils/logging/index.ts +0 -236
- package/source/utils/logging/integration.spec.ts +0 -441
- package/source/utils/logging/log-method-factory.spec.ts +0 -573
- package/source/utils/logging/log-method-factory.ts +0 -233
- package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +0 -277
- package/source/utils/logging/log-query/aggregation/aggregator.ts +0 -159
- package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +0 -159
- package/source/utils/logging/log-query/aggregation/facet-generator.ts +0 -47
- package/source/utils/logging/log-query/index.ts +0 -23
- package/source/utils/logging/log-query/query/filter-predicates.spec.ts +0 -247
- package/source/utils/logging/log-query/query/filter-predicates.ts +0 -154
- package/source/utils/logging/log-query/query/query-builder.spec.ts +0 -182
- package/source/utils/logging/log-query/query/query-builder.ts +0 -151
- package/source/utils/logging/log-query/query/query-engine.spec.ts +0 -214
- package/source/utils/logging/log-query/query/query-engine.ts +0 -45
- package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +0 -143
- package/source/utils/logging/log-query/storage/circular-buffer.ts +0 -75
- package/source/utils/logging/log-query/storage/index-manager.spec.ts +0 -150
- package/source/utils/logging/log-query/storage/index-manager.ts +0 -71
- package/source/utils/logging/log-query/storage/log-storage.spec.ts +0 -257
- package/source/utils/logging/log-query/storage/log-storage.ts +0 -80
- package/source/utils/logging/log-query/types.ts +0 -163
- package/source/utils/logging/log-query/utils/helpers.spec.ts +0 -263
- package/source/utils/logging/log-query/utils/helpers.ts +0 -72
- package/source/utils/logging/log-query/utils/sorting.spec.ts +0 -182
- package/source/utils/logging/log-query/utils/sorting.ts +0 -61
- package/source/utils/logging/logger-provider.spec.ts +0 -262
- package/source/utils/logging/logger-provider.ts +0 -362
- package/source/utils/logging/performance.spec.ts +0 -209
- package/source/utils/logging/performance.ts +0 -757
- package/source/utils/logging/pino-logger.spec.ts +0 -425
- package/source/utils/logging/pino-logger.ts +0 -514
- package/source/utils/logging/redaction.spec.ts +0 -490
- package/source/utils/logging/redaction.ts +0 -267
- package/source/utils/logging/request-tracker.spec.ts +0 -1198
- package/source/utils/logging/request-tracker.ts +0 -803
- package/source/utils/logging/transports.spec.ts +0 -505
- package/source/utils/logging/transports.ts +0 -305
- package/source/utils/logging/types.ts +0 -216
- package/source/utils/message-builder.spec.ts +0 -179
- package/source/utils/message-builder.ts +0 -101
- package/source/utils/message-queue.tsx +0 -486
- package/source/utils/paste-detection.spec.ts +0 -69
- package/source/utils/paste-detection.ts +0 -124
- package/source/utils/paste-roundtrip.spec.ts +0 -442
- package/source/utils/paste-utils.spec.ts +0 -128
- package/source/utils/paste-utils.ts +0 -52
- package/source/utils/programming-language-helper.spec.ts +0 -74
- package/source/utils/programming-language-helper.ts +0 -32
- package/source/utils/prompt-assembly.spec.ts +0 -221
- package/source/utils/prompt-processor.ts +0 -173
- package/source/utils/tool-args-parser.spec.ts +0 -136
- package/source/utils/tool-args-parser.ts +0 -54
- package/source/utils/tool-cancellation.spec.ts +0 -230
- package/source/utils/tool-cancellation.ts +0 -28
- package/source/utils/tool-result-display.spec.tsx +0 -469
- package/source/utils/tool-result-display.tsx +0 -90
- package/source/utils/update-checker.spec.ts +0 -383
- package/source/utils/update-checker.ts +0 -183
- package/source/wizard/config-wizard.spec.tsx +0 -103
- package/source/wizard/config-wizard.tsx +0 -382
- package/source/wizard/steps/location-step.spec.tsx +0 -186
- package/source/wizard/steps/location-step.tsx +0 -147
- package/source/wizard/steps/mcp-step.spec.tsx +0 -607
- package/source/wizard/steps/mcp-step.tsx +0 -632
- package/source/wizard/steps/provider-step.spec.tsx +0 -342
- package/source/wizard/steps/provider-step.tsx +0 -957
- package/source/wizard/steps/summary-step.spec.tsx +0 -749
- package/source/wizard/steps/summary-step.tsx +0 -228
- package/source/wizard/templates/mcp-templates.spec.ts +0 -613
- package/source/wizard/templates/mcp-templates.ts +0 -570
- package/source/wizard/templates/provider-templates.spec.ts +0 -152
- package/source/wizard/templates/provider-templates.ts +0 -485
- package/source/wizard/utils/fetch-cloud-models.spec.ts +0 -428
- package/source/wizard/utils/fetch-cloud-models.ts +0 -223
- package/source/wizard/utils/fetch-local-models.spec.ts +0 -297
- package/source/wizard/utils/fetch-local-models.ts +0 -192
- package/source/wizard/validation-array.spec.ts +0 -264
- package/source/wizard/validation.spec.ts +0 -373
- package/source/wizard/validation.ts +0 -232
|
@@ -1,330 +0,0 @@
|
|
|
1
|
-
import {CACHE_MODELS_EXPIRATION_MS} from '@/constants';
|
|
2
|
-
import {ModelEntry} from '@/types/index';
|
|
3
|
-
import {logError} from '@/utils/message-queue';
|
|
4
|
-
|
|
5
|
-
// Cache for fetched model data
|
|
6
|
-
interface ModelCache {
|
|
7
|
-
models: ModelEntry[];
|
|
8
|
-
timestamp: number;
|
|
9
|
-
}
|
|
10
|
-
|
|
11
|
-
// OpenRouter API response structure
|
|
12
|
-
interface OpenRouterModel {
|
|
13
|
-
id: string;
|
|
14
|
-
name: string;
|
|
15
|
-
description: string;
|
|
16
|
-
created: number;
|
|
17
|
-
context_length: number;
|
|
18
|
-
architecture: {
|
|
19
|
-
modality: string;
|
|
20
|
-
input_modalities: string[];
|
|
21
|
-
output_modalities: string[];
|
|
22
|
-
tokenizer: string;
|
|
23
|
-
};
|
|
24
|
-
pricing: {
|
|
25
|
-
prompt: string;
|
|
26
|
-
completion: string;
|
|
27
|
-
};
|
|
28
|
-
supported_parameters?: string[];
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
interface OpenRouterResponse {
|
|
32
|
-
data: OpenRouterModel[];
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
const OPENROUTER_API = 'https://openrouter.ai/api/v1/models';
|
|
36
|
-
|
|
37
|
-
let modelCache: ModelCache | null = null;
|
|
38
|
-
|
|
39
|
-
/**
|
|
40
|
-
* Known open-weight model prefixes/patterns
|
|
41
|
-
*/
|
|
42
|
-
const OPEN_WEIGHT_PATTERNS = [
|
|
43
|
-
'meta-llama/',
|
|
44
|
-
'mistralai/',
|
|
45
|
-
'qwen/',
|
|
46
|
-
'deepseek/',
|
|
47
|
-
'google/gemma',
|
|
48
|
-
'microsoft/phi',
|
|
49
|
-
'nvidia/',
|
|
50
|
-
'cohere/command-r',
|
|
51
|
-
'databricks/',
|
|
52
|
-
'allenai/',
|
|
53
|
-
'huggingfaceh4/',
|
|
54
|
-
'openchat/',
|
|
55
|
-
'teknium/',
|
|
56
|
-
'nousresearch/',
|
|
57
|
-
'cognitivecomputations/',
|
|
58
|
-
'thebloke/',
|
|
59
|
-
'codellama/',
|
|
60
|
-
'/llama',
|
|
61
|
-
'/mistral',
|
|
62
|
-
'/qwen',
|
|
63
|
-
'/deepseek',
|
|
64
|
-
'/gemma',
|
|
65
|
-
'/phi-',
|
|
66
|
-
'/wizardlm',
|
|
67
|
-
'/vicuna',
|
|
68
|
-
'/falcon',
|
|
69
|
-
'/starcoder',
|
|
70
|
-
'/codestral',
|
|
71
|
-
'/devstral',
|
|
72
|
-
];
|
|
73
|
-
|
|
74
|
-
/**
|
|
75
|
-
* Check if a model is open-weight based on ID
|
|
76
|
-
*/
|
|
77
|
-
function isOpenWeight(modelId: string): boolean {
|
|
78
|
-
const lowerId = modelId.toLowerCase();
|
|
79
|
-
return OPEN_WEIGHT_PATTERNS.some(pattern => lowerId.includes(pattern));
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
/**
|
|
83
|
-
* Extract author from model ID
|
|
84
|
-
*/
|
|
85
|
-
function extractAuthor(modelId: string): string {
|
|
86
|
-
const parts = modelId.split('/');
|
|
87
|
-
if (parts.length >= 2) {
|
|
88
|
-
const author = parts[0];
|
|
89
|
-
// Capitalize first letter
|
|
90
|
-
return author.charAt(0).toUpperCase() + author.slice(1);
|
|
91
|
-
}
|
|
92
|
-
return 'Unknown';
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
/**
|
|
96
|
-
* Check if model supports text input and output
|
|
97
|
-
*/
|
|
98
|
-
function supportsTextInTextOut(model: OpenRouterModel): boolean {
|
|
99
|
-
const inputMods = model.architecture?.input_modalities || [];
|
|
100
|
-
const outputMods = model.architecture?.output_modalities || [];
|
|
101
|
-
|
|
102
|
-
const hasTextInput = inputMods.length === 0 || inputMods.includes('text');
|
|
103
|
-
const hasTextOutput = outputMods.length === 0 || outputMods.includes('text');
|
|
104
|
-
|
|
105
|
-
return hasTextInput && hasTextOutput;
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
/**
|
|
109
|
-
* Check if model is relevant for coding (has tool support or is a known coding model)
|
|
110
|
-
*/
|
|
111
|
-
function isRelevantForCoding(model: OpenRouterModel): boolean {
|
|
112
|
-
const modelId = model.id.toLowerCase();
|
|
113
|
-
const modelName = model.name.toLowerCase();
|
|
114
|
-
const description = (model.description || '').toLowerCase();
|
|
115
|
-
|
|
116
|
-
// Must support text in/out
|
|
117
|
-
if (!supportsTextInTextOut(model)) return false;
|
|
118
|
-
|
|
119
|
-
// Skip embedding models
|
|
120
|
-
if (modelId.includes('embed') || modelName.includes('embed')) return false;
|
|
121
|
-
|
|
122
|
-
// Skip image/audio only models
|
|
123
|
-
if (
|
|
124
|
-
modelId.includes('dall-e') ||
|
|
125
|
-
modelId.includes('stable-diffusion') ||
|
|
126
|
-
modelId.includes('whisper') ||
|
|
127
|
-
modelId.includes('tts') ||
|
|
128
|
-
modelId.includes('imagen')
|
|
129
|
-
) {
|
|
130
|
-
return false;
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
// Has tool calling support
|
|
134
|
-
if (model.supported_parameters?.includes('tools')) return true;
|
|
135
|
-
|
|
136
|
-
// Known coding-capable models
|
|
137
|
-
const codingPatterns = [
|
|
138
|
-
'gpt',
|
|
139
|
-
'claude',
|
|
140
|
-
'gemini',
|
|
141
|
-
'deepseek',
|
|
142
|
-
'qwen',
|
|
143
|
-
'coder',
|
|
144
|
-
'codestral',
|
|
145
|
-
'devstral',
|
|
146
|
-
'llama',
|
|
147
|
-
'mistral',
|
|
148
|
-
'phi',
|
|
149
|
-
'command',
|
|
150
|
-
'grok',
|
|
151
|
-
'codex',
|
|
152
|
-
];
|
|
153
|
-
|
|
154
|
-
if (codingPatterns.some(p => modelId.includes(p) || modelName.includes(p))) {
|
|
155
|
-
return true;
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
// Description mentions coding
|
|
159
|
-
if (
|
|
160
|
-
description.includes('code') ||
|
|
161
|
-
description.includes('programming') ||
|
|
162
|
-
description.includes('developer')
|
|
163
|
-
) {
|
|
164
|
-
return true;
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
return false;
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
/**
|
|
171
|
-
* Calculate cost score (0-10 scale, 10 = free/cheapest)
|
|
172
|
-
*/
|
|
173
|
-
function calculateCostScore(
|
|
174
|
-
promptPrice: string,
|
|
175
|
-
completionPrice: string,
|
|
176
|
-
): number {
|
|
177
|
-
const input = parseFloat(promptPrice) || 0;
|
|
178
|
-
const output = parseFloat(completionPrice) || 0;
|
|
179
|
-
|
|
180
|
-
// Price per million tokens (prices are per token)
|
|
181
|
-
const inputPerMillion = input * 1_000_000;
|
|
182
|
-
const outputPerMillion = output * 1_000_000;
|
|
183
|
-
|
|
184
|
-
// Weighted average (output typically more important)
|
|
185
|
-
const avgCost = inputPerMillion * 0.3 + outputPerMillion * 0.7;
|
|
186
|
-
|
|
187
|
-
if (avgCost === 0) return 10;
|
|
188
|
-
if (avgCost < 0.1) return 9;
|
|
189
|
-
if (avgCost < 0.5) return 8;
|
|
190
|
-
if (avgCost < 1) return 7;
|
|
191
|
-
if (avgCost < 2) return 6;
|
|
192
|
-
if (avgCost < 5) return 5;
|
|
193
|
-
if (avgCost < 10) return 4;
|
|
194
|
-
if (avgCost < 20) return 3;
|
|
195
|
-
if (avgCost < 50) return 2;
|
|
196
|
-
return 1;
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
/**
|
|
200
|
-
* Format cost details string
|
|
201
|
-
*/
|
|
202
|
-
function formatCostDetails(
|
|
203
|
-
promptPrice: string,
|
|
204
|
-
completionPrice: string,
|
|
205
|
-
isLocal: boolean,
|
|
206
|
-
): string {
|
|
207
|
-
const input = parseFloat(promptPrice) || 0;
|
|
208
|
-
const output = parseFloat(completionPrice) || 0;
|
|
209
|
-
|
|
210
|
-
if (input === 0 && output === 0) {
|
|
211
|
-
return isLocal ? 'Free (open weights)' : 'Free';
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
// Convert to per million tokens
|
|
215
|
-
const inputPerMillion = input * 1_000_000;
|
|
216
|
-
const outputPerMillion = output * 1_000_000;
|
|
217
|
-
|
|
218
|
-
return `$${inputPerMillion.toFixed(2)}/M in, $${outputPerMillion.toFixed(
|
|
219
|
-
2,
|
|
220
|
-
)}/M out`;
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
/**
|
|
224
|
-
* Format context length for display
|
|
225
|
-
*/
|
|
226
|
-
function formatContextLength(contextLength: number): string {
|
|
227
|
-
if (contextLength >= 1_000_000) {
|
|
228
|
-
return `${(contextLength / 1_000_000).toFixed(1)}M`;
|
|
229
|
-
}
|
|
230
|
-
if (contextLength >= 1000) {
|
|
231
|
-
return `${Math.round(contextLength / 1000)}K`;
|
|
232
|
-
}
|
|
233
|
-
return `${contextLength}`;
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
/**
|
|
237
|
-
* Fetch models from OpenRouter API
|
|
238
|
-
*/
|
|
239
|
-
async function fetchOpenRouterModels(): Promise<OpenRouterModel[]> {
|
|
240
|
-
try {
|
|
241
|
-
const response = await fetch(OPENROUTER_API);
|
|
242
|
-
if (!response.ok) {
|
|
243
|
-
throw new Error(`OpenRouter API returned ${response.status}`);
|
|
244
|
-
}
|
|
245
|
-
|
|
246
|
-
const data = (await response.json()) as OpenRouterResponse;
|
|
247
|
-
return data.data || [];
|
|
248
|
-
} catch (error) {
|
|
249
|
-
logError('Failed to fetch OpenRouter models', true, {error});
|
|
250
|
-
return [];
|
|
251
|
-
}
|
|
252
|
-
}
|
|
253
|
-
|
|
254
|
-
/**
|
|
255
|
-
* Fetch and process models from OpenRouter
|
|
256
|
-
*/
|
|
257
|
-
export async function fetchModels(): Promise<ModelEntry[]> {
|
|
258
|
-
// Check cache first
|
|
259
|
-
if (
|
|
260
|
-
modelCache &&
|
|
261
|
-
Date.now() - modelCache.timestamp < CACHE_MODELS_EXPIRATION_MS
|
|
262
|
-
) {
|
|
263
|
-
return modelCache.models;
|
|
264
|
-
}
|
|
265
|
-
|
|
266
|
-
const openRouterModels = await fetchOpenRouterModels();
|
|
267
|
-
const models: ModelEntry[] = [];
|
|
268
|
-
|
|
269
|
-
for (const model of openRouterModels) {
|
|
270
|
-
// Skip if not relevant for coding
|
|
271
|
-
if (!isRelevantForCoding(model)) continue;
|
|
272
|
-
|
|
273
|
-
const isLocal = isOpenWeight(model.id);
|
|
274
|
-
const costScore = calculateCostScore(
|
|
275
|
-
model.pricing.prompt,
|
|
276
|
-
model.pricing.completion,
|
|
277
|
-
);
|
|
278
|
-
|
|
279
|
-
const entry: ModelEntry = {
|
|
280
|
-
id: model.id,
|
|
281
|
-
name: model.name,
|
|
282
|
-
author: extractAuthor(model.id),
|
|
283
|
-
size: formatContextLength(model.context_length),
|
|
284
|
-
local: isLocal,
|
|
285
|
-
api: true,
|
|
286
|
-
contextLength: model.context_length,
|
|
287
|
-
created: model.created,
|
|
288
|
-
quality: {
|
|
289
|
-
cost: costScore,
|
|
290
|
-
},
|
|
291
|
-
costType: costScore >= 9 ? ('free' as const) : ('paid' as const),
|
|
292
|
-
costDetails: formatCostDetails(
|
|
293
|
-
model.pricing.prompt,
|
|
294
|
-
model.pricing.completion,
|
|
295
|
-
isLocal,
|
|
296
|
-
),
|
|
297
|
-
hasToolSupport: model.supported_parameters?.includes('tools') || false,
|
|
298
|
-
};
|
|
299
|
-
|
|
300
|
-
models.push(entry);
|
|
301
|
-
}
|
|
302
|
-
|
|
303
|
-
// Sort by created date (newest first)
|
|
304
|
-
models.sort((a, b) => (b.created || 0) - (a.created || 0));
|
|
305
|
-
|
|
306
|
-
// Update cache
|
|
307
|
-
modelCache = {
|
|
308
|
-
models,
|
|
309
|
-
timestamp: Date.now(),
|
|
310
|
-
};
|
|
311
|
-
|
|
312
|
-
return models;
|
|
313
|
-
}
|
|
314
|
-
|
|
315
|
-
/**
|
|
316
|
-
* Clear the model cache (useful for forcing a refresh)
|
|
317
|
-
*/
|
|
318
|
-
export function clearModelCache(): void {
|
|
319
|
-
modelCache = null;
|
|
320
|
-
}
|
|
321
|
-
|
|
322
|
-
/**
|
|
323
|
-
* Check if models are currently cached
|
|
324
|
-
*/
|
|
325
|
-
export function isModelsCached(): boolean {
|
|
326
|
-
return (
|
|
327
|
-
modelCache !== null &&
|
|
328
|
-
Date.now() - modelCache.timestamp < CACHE_MODELS_EXPIRATION_MS
|
|
329
|
-
);
|
|
330
|
-
}
|
package/source/models/index.ts
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export {getModelContextLimit} from './models-dev-client.js';
|
|
@@ -1,214 +0,0 @@
|
|
|
1
|
-
import test from 'ava';
|
|
2
|
-
import { unlink, writeFile, readFile, mkdir } from 'node:fs/promises';
|
|
3
|
-
import { existsSync } from 'node:fs';
|
|
4
|
-
import { join } from 'node:path';
|
|
5
|
-
import { tmpdir } from 'node:os';
|
|
6
|
-
import type { CachedModelsData } from './models-types.js';
|
|
7
|
-
import {
|
|
8
|
-
readCache,
|
|
9
|
-
writeCache,
|
|
10
|
-
} from './models-cache.js';
|
|
11
|
-
|
|
12
|
-
// Get the actual cache file path that will be used
|
|
13
|
-
// We need to import dynamically to get the actual path at runtime
|
|
14
|
-
let actualCachePath: string;
|
|
15
|
-
|
|
16
|
-
async function getCacheFilePath(): Promise<string> {
|
|
17
|
-
// Import dynamically to avoid module resolution issues during testing
|
|
18
|
-
const { xdgCache } = await import('xdg-basedir');
|
|
19
|
-
const path = await import('node:path');
|
|
20
|
-
|
|
21
|
-
const DEFAULT_CACHE_DIR =
|
|
22
|
-
process.platform === 'darwin'
|
|
23
|
-
? path.join(process.env.HOME || '~', 'Library', 'Caches')
|
|
24
|
-
: path.join(process.env.HOME || '~', '.cache');
|
|
25
|
-
|
|
26
|
-
const cacheBase = xdgCache || DEFAULT_CACHE_DIR;
|
|
27
|
-
return path.join(cacheBase, 'coder', 'models.json');
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
// ============================================================================
|
|
31
|
-
// Setup and Teardown
|
|
32
|
-
// ============================================================================
|
|
33
|
-
|
|
34
|
-
test.before(async () => {
|
|
35
|
-
actualCachePath = await getCacheFilePath();
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
test.afterEach.always(async () => {
|
|
39
|
-
// Clean up test cache file after each test
|
|
40
|
-
if (actualCachePath && existsSync(actualCachePath)) {
|
|
41
|
-
try {
|
|
42
|
-
await unlink(actualCachePath);
|
|
43
|
-
} catch {
|
|
44
|
-
// Ignore cleanup errors
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
// ============================================================================
|
|
50
|
-
// Tests for readCache
|
|
51
|
-
// ============================================================================
|
|
52
|
-
|
|
53
|
-
test('readCache returns null when cache file does not exist', async t => {
|
|
54
|
-
// Ensure cache file doesn't exist
|
|
55
|
-
if (existsSync(actualCachePath)) {
|
|
56
|
-
await unlink(actualCachePath);
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
const result = await readCache();
|
|
60
|
-
t.is(result, null);
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
test('readCache returns valid cached data', async t => {
|
|
64
|
-
const mockData: CachedModelsData = {
|
|
65
|
-
data: { models: [] } as any,
|
|
66
|
-
fetchedAt: Date.now() - 1000,
|
|
67
|
-
expiresAt: Date.now() + 1000000, // Not expired
|
|
68
|
-
};
|
|
69
|
-
|
|
70
|
-
// Ensure directory exists
|
|
71
|
-
const dir = join(actualCachePath, '..');
|
|
72
|
-
try {
|
|
73
|
-
await mkdir(dir, { recursive: true });
|
|
74
|
-
} catch {
|
|
75
|
-
// Directory may already exist
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
await writeFile(actualCachePath, JSON.stringify(mockData), 'utf-8');
|
|
79
|
-
|
|
80
|
-
const result = await readCache();
|
|
81
|
-
|
|
82
|
-
t.truthy(result);
|
|
83
|
-
t.deepEqual(result?.data, mockData.data);
|
|
84
|
-
t.is(result?.fetchedAt, mockData.fetchedAt);
|
|
85
|
-
t.is(result?.expiresAt, mockData.expiresAt);
|
|
86
|
-
});
|
|
87
|
-
|
|
88
|
-
test('readCache returns null when cache is expired', async t => {
|
|
89
|
-
const expiredData: CachedModelsData = {
|
|
90
|
-
data: { models: [] } as any,
|
|
91
|
-
fetchedAt: Date.now() - 1000000,
|
|
92
|
-
expiresAt: Date.now() - 1000, // Expired
|
|
93
|
-
};
|
|
94
|
-
|
|
95
|
-
// Ensure directory exists
|
|
96
|
-
const dir = join(actualCachePath, '..');
|
|
97
|
-
try {
|
|
98
|
-
await mkdir(dir, { recursive: true });
|
|
99
|
-
} catch {
|
|
100
|
-
// Directory may already exist
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
await writeFile(actualCachePath, JSON.stringify(expiredData), 'utf-8');
|
|
104
|
-
|
|
105
|
-
const result = await readCache();
|
|
106
|
-
|
|
107
|
-
t.is(result, null);
|
|
108
|
-
});
|
|
109
|
-
|
|
110
|
-
test('readCache returns null on JSON parse error', async t => {
|
|
111
|
-
// Ensure directory exists
|
|
112
|
-
const dir = join(actualCachePath, '..');
|
|
113
|
-
try {
|
|
114
|
-
await mkdir(dir, { recursive: true });
|
|
115
|
-
} catch {
|
|
116
|
-
// Directory may already exist
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
await writeFile(actualCachePath, 'invalid json{{{', 'utf-8');
|
|
120
|
-
|
|
121
|
-
const result = await readCache();
|
|
122
|
-
|
|
123
|
-
t.is(result, null);
|
|
124
|
-
});
|
|
125
|
-
|
|
126
|
-
// ============================================================================
|
|
127
|
-
// Tests for writeCache
|
|
128
|
-
// ============================================================================
|
|
129
|
-
|
|
130
|
-
test('writeCache writes cache data correctly', async t => {
|
|
131
|
-
const mockData = { models: [{ id: 'test-model' }] } as any;
|
|
132
|
-
|
|
133
|
-
await writeCache(mockData);
|
|
134
|
-
|
|
135
|
-
// Verify the file was written
|
|
136
|
-
const exists = existsSync(actualCachePath);
|
|
137
|
-
t.true(exists);
|
|
138
|
-
|
|
139
|
-
// Verify the content
|
|
140
|
-
const content = await readFile(actualCachePath, 'utf-8');
|
|
141
|
-
const writtenData = JSON.parse(content) as CachedModelsData;
|
|
142
|
-
|
|
143
|
-
t.truthy(writtenData.data);
|
|
144
|
-
t.truthy(writtenData.fetchedAt);
|
|
145
|
-
t.truthy(writtenData.expiresAt);
|
|
146
|
-
t.true(writtenData.expiresAt > Date.now());
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
test('writeCache creates directory if it does not exist', async t => {
|
|
150
|
-
const mockData = { models: [] } as any;
|
|
151
|
-
|
|
152
|
-
// This should not throw even if directory doesn't exist
|
|
153
|
-
await t.notThrowsAsync(async () => {
|
|
154
|
-
await writeCache(mockData);
|
|
155
|
-
});
|
|
156
|
-
|
|
157
|
-
// Verify the file was created
|
|
158
|
-
t.true(existsSync(actualCachePath));
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
// ============================================================================
|
|
162
|
-
// Integration-like tests
|
|
163
|
-
// ============================================================================
|
|
164
|
-
|
|
165
|
-
test('cache expiration is calculated correctly', async t => {
|
|
166
|
-
const mockData = { models: [] } as any;
|
|
167
|
-
const now = Date.now();
|
|
168
|
-
|
|
169
|
-
await writeCache(mockData);
|
|
170
|
-
|
|
171
|
-
const content = await readFile(actualCachePath, 'utf-8');
|
|
172
|
-
const writtenData = JSON.parse(content) as CachedModelsData;
|
|
173
|
-
|
|
174
|
-
// Get the expiration constant
|
|
175
|
-
const { CACHE_MODELS_EXPIRATION_MS } = await import('@/constants');
|
|
176
|
-
|
|
177
|
-
// Expiration should be roughly CACHE_MODELS_EXPIRATION_MS in the future
|
|
178
|
-
t.true(writtenData.expiresAt >= now + CACHE_MODELS_EXPIRATION_MS - 100);
|
|
179
|
-
t.true(writtenData.expiresAt <= now + CACHE_MODELS_EXPIRATION_MS + 100);
|
|
180
|
-
});
|
|
181
|
-
|
|
182
|
-
test('readCache and writeCache round-trip correctly', async t => {
|
|
183
|
-
const originalData = {
|
|
184
|
-
models: [
|
|
185
|
-
{ id: 'model1', name: 'Test Model 1' },
|
|
186
|
-
{ id: 'model2', name: 'Test Model 2' },
|
|
187
|
-
],
|
|
188
|
-
} as any;
|
|
189
|
-
|
|
190
|
-
// Write the cache
|
|
191
|
-
await writeCache(originalData);
|
|
192
|
-
|
|
193
|
-
// Read it back
|
|
194
|
-
const readResult = await readCache();
|
|
195
|
-
|
|
196
|
-
t.truthy(readResult);
|
|
197
|
-
t.deepEqual(readResult?.data, originalData);
|
|
198
|
-
});
|
|
199
|
-
|
|
200
|
-
test('writeCache overwrites existing cache', async t => {
|
|
201
|
-
// Write first cache
|
|
202
|
-
const firstData = { models: [{ id: 'model1' }] } as any;
|
|
203
|
-
await writeCache(firstData);
|
|
204
|
-
|
|
205
|
-
// Write second cache (should overwrite)
|
|
206
|
-
const secondData = { models: [{ id: 'model2' }, { id: 'model3' }] } as any;
|
|
207
|
-
await writeCache(secondData);
|
|
208
|
-
|
|
209
|
-
// Read back
|
|
210
|
-
const readResult = await readCache();
|
|
211
|
-
|
|
212
|
-
t.truthy(readResult);
|
|
213
|
-
t.deepEqual(readResult?.data, secondData);
|
|
214
|
-
});
|
|
@@ -1,78 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Cache management for models.dev data
|
|
3
|
-
* Stores model database in XDG_CACHE_HOME for fast lookup
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import {constants} from 'node:fs';
|
|
7
|
-
import {access, mkdir, readFile, writeFile} from 'node:fs/promises';
|
|
8
|
-
import * as path from 'node:path';
|
|
9
|
-
import {CACHE_MODELS_EXPIRATION_MS} from '@/constants';
|
|
10
|
-
import {formatError} from '@/utils/error-formatter';
|
|
11
|
-
import {getLogger} from '@/utils/logging';
|
|
12
|
-
import {xdgCache} from 'xdg-basedir';
|
|
13
|
-
import type {CachedModelsData, ModelsDevDatabase} from './models-types.js';
|
|
14
|
-
|
|
15
|
-
const DEFAULT_CACHE_DIR =
|
|
16
|
-
process.platform === 'darwin'
|
|
17
|
-
? path.join(process.env.HOME || '~', 'Library', 'Caches')
|
|
18
|
-
: path.join(process.env.HOME || '~', '.cache');
|
|
19
|
-
|
|
20
|
-
function getCacheDir(): string {
|
|
21
|
-
const cacheBase = xdgCache || DEFAULT_CACHE_DIR;
|
|
22
|
-
return path.join(cacheBase, 'coder');
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
function getCacheFilePath(): string {
|
|
26
|
-
return path.join(getCacheDir(), 'models.json');
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
async function ensureCacheDir(): Promise<void> {
|
|
30
|
-
const dir = getCacheDir();
|
|
31
|
-
|
|
32
|
-
try {
|
|
33
|
-
await access(dir, constants.F_OK);
|
|
34
|
-
} catch {
|
|
35
|
-
await mkdir(dir, {recursive: true});
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
export async function readCache(): Promise<CachedModelsData | null> {
|
|
40
|
-
try {
|
|
41
|
-
const cachePath = getCacheFilePath();
|
|
42
|
-
|
|
43
|
-
await access(cachePath, constants.F_OK);
|
|
44
|
-
|
|
45
|
-
const content = await readFile(cachePath, 'utf-8');
|
|
46
|
-
const cached = JSON.parse(content) as CachedModelsData;
|
|
47
|
-
|
|
48
|
-
// Check if cache is expired
|
|
49
|
-
if (Date.now() > cached.expiresAt) {
|
|
50
|
-
return null;
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
return cached;
|
|
54
|
-
} catch (error) {
|
|
55
|
-
// If there's any error reading cache, return null to trigger fresh fetch
|
|
56
|
-
const logger = getLogger();
|
|
57
|
-
logger.warn({error: formatError(error)}, 'Failed to read models cache');
|
|
58
|
-
return null;
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
export async function writeCache(data: ModelsDevDatabase): Promise<void> {
|
|
63
|
-
try {
|
|
64
|
-
await ensureCacheDir();
|
|
65
|
-
|
|
66
|
-
const cached: CachedModelsData = {
|
|
67
|
-
data,
|
|
68
|
-
fetchedAt: Date.now(),
|
|
69
|
-
expiresAt: Date.now() + CACHE_MODELS_EXPIRATION_MS,
|
|
70
|
-
};
|
|
71
|
-
|
|
72
|
-
const cachePath = getCacheFilePath();
|
|
73
|
-
await writeFile(cachePath, JSON.stringify(cached, null, 2), 'utf-8');
|
|
74
|
-
} catch (error) {
|
|
75
|
-
const logger = getLogger();
|
|
76
|
-
logger.warn({error: formatError(error)}, 'Failed to write models cache');
|
|
77
|
-
}
|
|
78
|
-
}
|