fss-link 1.0.49 → 1.0.51
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/dist/index.js +0 -0
- package/dist/package.json +2 -2
- package/dist/src/config/auth.js +8 -5
- package/dist/src/config/auth.js.map +1 -1
- package/dist/src/config/database.d.ts +103 -11
- package/dist/src/config/database.js +301 -59
- package/dist/src/config/database.js.map +1 -1
- package/dist/src/config/databaseBackup.d.ts +114 -0
- package/dist/src/config/databaseBackup.js +334 -0
- package/dist/src/config/databaseBackup.js.map +1 -0
- package/dist/src/config/databaseMigrations.d.ts +63 -0
- package/dist/src/config/databaseMigrations.js +379 -0
- package/dist/src/config/databaseMigrations.js.map +1 -0
- package/dist/src/config/databasePool.d.ts +70 -0
- package/dist/src/config/databasePool.js +193 -0
- package/dist/src/config/databasePool.js.map +1 -0
- package/dist/src/config/queryOptimizer.d.ts +127 -0
- package/dist/src/config/queryOptimizer.js +309 -0
- package/dist/src/config/queryOptimizer.js.map +1 -0
- package/dist/src/utils/sandbox.js +2 -8
- package/dist/src/utils/sandbox.js.map +1 -1
- package/dist/src/validateNonInterActiveAuth.js +3 -7
- package/dist/src/validateNonInterActiveAuth.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +2 -2
- package/dist/commands/mcp/add.test.ts +0 -122
- package/dist/commands/mcp/add.ts +0 -222
- package/dist/commands/mcp/list.test.ts +0 -154
- package/dist/commands/mcp/list.ts +0 -139
- package/dist/commands/mcp/remove.test.ts +0 -69
- package/dist/commands/mcp/remove.ts +0 -60
- package/dist/commands/mcp.test.ts +0 -55
- package/dist/commands/mcp.ts +0 -27
- package/dist/config/apiValidation.test.ts +0 -118
- package/dist/config/auth.test.ts +0 -79
- package/dist/config/auth.ts +0 -100
- package/dist/config/config.integration.test.ts +0 -407
- package/dist/config/config.test.ts +0 -1952
- package/dist/config/config.ts +0 -690
- package/dist/config/database.test.ts +0 -96
- package/dist/config/database.ts +0 -824
- package/dist/config/extension.test.ts +0 -236
- package/dist/config/extension.ts +0 -180
- package/dist/config/keyBindings.test.ts +0 -62
- package/dist/config/keyBindings.ts +0 -184
- package/dist/config/modelManager.ts +0 -326
- package/dist/config/providerManager.ts +0 -244
- package/dist/config/providerPersistence.test.ts +0 -377
- package/dist/config/providerPersistence.ts +0 -105
- package/dist/config/sandboxConfig.ts +0 -107
- package/dist/config/settings.test.ts +0 -1424
- package/dist/config/settings.ts +0 -517
- package/dist/config/settingsSchema.test.ts +0 -252
- package/dist/config/settingsSchema.ts +0 -728
- package/dist/config/trustedFolders.test.ts +0 -208
- package/dist/config/trustedFolders.ts +0 -167
- package/dist/gemini.test.tsx +0 -252
- package/dist/gemini.tsx +0 -357
- package/dist/generated/git-commit.ts +0 -10
- package/dist/index.ts +0 -21
- package/dist/nonInteractiveCli.test.ts +0 -276
- package/dist/nonInteractiveCli.ts +0 -143
- package/dist/patches/is-in-ci.ts +0 -17
- package/dist/services/BuiltinCommandLoader.test.ts +0 -127
- package/dist/services/BuiltinCommandLoader.ts +0 -95
- package/dist/services/CommandService.test.ts +0 -352
- package/dist/services/CommandService.ts +0 -103
- package/dist/services/FileCommandLoader.test.ts +0 -1002
- package/dist/services/FileCommandLoader.ts +0 -289
- package/dist/services/McpPromptLoader.ts +0 -231
- package/dist/services/SearchEngineConfigProvider.ts +0 -100
- package/dist/services/prompt-processors/argumentProcessor.test.ts +0 -41
- package/dist/services/prompt-processors/argumentProcessor.ts +0 -23
- package/dist/services/prompt-processors/shellProcessor.test.ts +0 -709
- package/dist/services/prompt-processors/shellProcessor.ts +0 -248
- package/dist/services/prompt-processors/types.ts +0 -44
- package/dist/services/types.ts +0 -24
- package/dist/src/config/apiValidation.test.d.ts +0 -6
- package/dist/src/config/apiValidation.test.js +0 -99
- package/dist/src/config/apiValidation.test.js.map +0 -1
- package/dist/src/config/database.test.d.ts +0 -6
- package/dist/src/config/database.test.js +0 -80
- package/dist/src/config/database.test.js.map +0 -1
- package/dist/src/config/providerManager.d.ts +0 -74
- package/dist/src/config/providerManager.js +0 -203
- package/dist/src/config/providerManager.js.map +0 -1
- package/dist/src/config/providerPersistence.test.d.ts +0 -6
- package/dist/src/config/providerPersistence.test.js +0 -283
- package/dist/src/config/providerPersistence.test.js.map +0 -1
- package/dist/src/ui/components/GeminiKeyDialog.d.ts +0 -11
- package/dist/src/ui/components/GeminiKeyDialog.js +0 -156
- package/dist/src/ui/components/GeminiKeyDialog.js.map +0 -1
- package/dist/src/ui/components/OpenAIEndpointDialog.d.ts +0 -19
- package/dist/src/ui/components/OpenAIEndpointDialog.js +0 -163
- package/dist/src/ui/components/OpenAIEndpointDialog.js.map +0 -1
- package/dist/test-setup.ts +0 -12
- package/dist/test-utils/customMatchers.ts +0 -65
- package/dist/test-utils/mockCommandContext.test.ts +0 -62
- package/dist/test-utils/mockCommandContext.ts +0 -105
- package/dist/test-utils/render.tsx +0 -18
- package/dist/ui/App.test.tsx +0 -2181
- package/dist/ui/App.tsx +0 -1344
- package/dist/ui/IdeIntegrationNudge.tsx +0 -98
- package/dist/ui/__snapshots__/App.test.tsx.snap +0 -124
- package/dist/ui/colors.ts +0 -56
- package/dist/ui/commands/aboutCommand.test.ts +0 -153
- package/dist/ui/commands/aboutCommand.ts +0 -49
- package/dist/ui/commands/authCommand.test.ts +0 -36
- package/dist/ui/commands/authCommand.ts +0 -17
- package/dist/ui/commands/bugCommand.test.ts +0 -114
- package/dist/ui/commands/bugCommand.ts +0 -92
- package/dist/ui/commands/chatCommand.test.ts +0 -414
- package/dist/ui/commands/chatCommand.ts +0 -280
- package/dist/ui/commands/clearCommand.test.ts +0 -100
- package/dist/ui/commands/clearCommand.ts +0 -29
- package/dist/ui/commands/compressCommand.test.ts +0 -129
- package/dist/ui/commands/compressCommand.ts +0 -78
- package/dist/ui/commands/contextCommand.ts +0 -132
- package/dist/ui/commands/copyCommand.test.ts +0 -296
- package/dist/ui/commands/copyCommand.ts +0 -67
- package/dist/ui/commands/corgiCommand.test.ts +0 -34
- package/dist/ui/commands/corgiCommand.ts +0 -16
- package/dist/ui/commands/directoryCommand.test.tsx +0 -185
- package/dist/ui/commands/directoryCommand.tsx +0 -179
- package/dist/ui/commands/docsCommand.test.ts +0 -99
- package/dist/ui/commands/docsCommand.ts +0 -42
- package/dist/ui/commands/editorCommand.test.ts +0 -30
- package/dist/ui/commands/editorCommand.ts +0 -21
- package/dist/ui/commands/extensionsCommand.test.ts +0 -67
- package/dist/ui/commands/extensionsCommand.ts +0 -46
- package/dist/ui/commands/helpCommand.test.ts +0 -52
- package/dist/ui/commands/helpCommand.ts +0 -23
- package/dist/ui/commands/ideCommand.test.ts +0 -255
- package/dist/ui/commands/ideCommand.ts +0 -283
- package/dist/ui/commands/initCommand.test.ts +0 -127
- package/dist/ui/commands/initCommand.ts +0 -117
- package/dist/ui/commands/mcpCommand.test.ts +0 -1057
- package/dist/ui/commands/mcpCommand.ts +0 -531
- package/dist/ui/commands/memoryCommand.test.ts +0 -344
- package/dist/ui/commands/memoryCommand.ts +0 -305
- package/dist/ui/commands/privacyCommand.test.ts +0 -38
- package/dist/ui/commands/privacyCommand.ts +0 -17
- package/dist/ui/commands/quitCommand.test.ts +0 -55
- package/dist/ui/commands/quitCommand.ts +0 -36
- package/dist/ui/commands/restoreCommand.test.ts +0 -250
- package/dist/ui/commands/restoreCommand.ts +0 -157
- package/dist/ui/commands/searchEngineSetupCommand.ts +0 -18
- package/dist/ui/commands/settingsCommand.test.ts +0 -36
- package/dist/ui/commands/settingsCommand.ts +0 -17
- package/dist/ui/commands/setupGithubCommand.test.ts +0 -238
- package/dist/ui/commands/setupGithubCommand.ts +0 -212
- package/dist/ui/commands/speakCommand.ts +0 -175
- package/dist/ui/commands/statsCommand.test.ts +0 -78
- package/dist/ui/commands/statsCommand.ts +0 -70
- package/dist/ui/commands/terminalSetupCommand.test.ts +0 -85
- package/dist/ui/commands/terminalSetupCommand.ts +0 -45
- package/dist/ui/commands/themeCommand.test.ts +0 -38
- package/dist/ui/commands/themeCommand.ts +0 -17
- package/dist/ui/commands/toolsCommand.test.ts +0 -105
- package/dist/ui/commands/toolsCommand.ts +0 -71
- package/dist/ui/commands/ttsCommand.ts +0 -143
- package/dist/ui/commands/types.ts +0 -204
- package/dist/ui/commands/vimCommand.ts +0 -25
- package/dist/ui/commands/voiceCommand.ts +0 -125
- package/dist/ui/components/AboutBox.tsx +0 -133
- package/dist/ui/components/AsciiArt.ts +0 -54
- package/dist/ui/components/AuthDialog.test.tsx +0 -334
- package/dist/ui/components/AuthDialog.tsx +0 -289
- package/dist/ui/components/AuthInProgress.tsx +0 -62
- package/dist/ui/components/AutoAcceptIndicator.tsx +0 -47
- package/dist/ui/components/ConsoleSummaryDisplay.tsx +0 -35
- package/dist/ui/components/ContextSummaryDisplay.test.tsx +0 -85
- package/dist/ui/components/ContextSummaryDisplay.tsx +0 -120
- package/dist/ui/components/ContextUsageDisplay.tsx +0 -77
- package/dist/ui/components/DebugProfiler.tsx +0 -36
- package/dist/ui/components/DetailedMessagesDisplay.tsx +0 -82
- package/dist/ui/components/EditorSettingsDialog.tsx +0 -172
- package/dist/ui/components/FolderTrustDialog.test.tsx +0 -36
- package/dist/ui/components/FolderTrustDialog.tsx +0 -74
- package/dist/ui/components/Footer.test.tsx +0 -159
- package/dist/ui/components/Footer.tsx +0 -158
- package/dist/ui/components/GeminiKeyDialog.tsx +0 -252
- package/dist/ui/components/GeminiRespondingSpinner.tsx +0 -34
- package/dist/ui/components/Header.test.tsx +0 -44
- package/dist/ui/components/Header.tsx +0 -70
- package/dist/ui/components/Help.tsx +0 -174
- package/dist/ui/components/HistoryItemDisplay.test.tsx +0 -125
- package/dist/ui/components/HistoryItemDisplay.tsx +0 -98
- package/dist/ui/components/InputPrompt.test.tsx +0 -1467
- package/dist/ui/components/InputPrompt.tsx +0 -641
- package/dist/ui/components/LMStudioModelPrompt.tsx +0 -215
- package/dist/ui/components/LoadingIndicator.test.tsx +0 -296
- package/dist/ui/components/LoadingIndicator.tsx +0 -82
- package/dist/ui/components/MemoryUsageDisplay.tsx +0 -36
- package/dist/ui/components/ModelStatsDisplay.test.tsx +0 -252
- package/dist/ui/components/ModelStatsDisplay.tsx +0 -197
- package/dist/ui/components/OllamaModelPrompt.tsx +0 -206
- package/dist/ui/components/OpenAIEndpointDialog.tsx +0 -261
- package/dist/ui/components/OpenAIKeyPrompt.test.tsx +0 -64
- package/dist/ui/components/OpenAIKeyPrompt.tsx +0 -197
- package/dist/ui/components/PrepareLabel.tsx +0 -48
- package/dist/ui/components/SearchEngineConfigDialog.tsx +0 -280
- package/dist/ui/components/SessionSummaryDisplay.test.tsx +0 -75
- package/dist/ui/components/SessionSummaryDisplay.tsx +0 -18
- package/dist/ui/components/SettingsDialog.test.tsx +0 -865
- package/dist/ui/components/SettingsDialog.tsx +0 -753
- package/dist/ui/components/ShellConfirmationDialog.test.tsx +0 -53
- package/dist/ui/components/ShellConfirmationDialog.tsx +0 -103
- package/dist/ui/components/ShellModeIndicator.tsx +0 -18
- package/dist/ui/components/ShowMoreLines.tsx +0 -40
- package/dist/ui/components/StatsDisplay.test.tsx +0 -401
- package/dist/ui/components/StatsDisplay.tsx +0 -273
- package/dist/ui/components/SuggestionsDisplay.tsx +0 -102
- package/dist/ui/components/ThemeDialog.tsx +0 -310
- package/dist/ui/components/Tips.tsx +0 -45
- package/dist/ui/components/TodoDisplay.test.tsx +0 -97
- package/dist/ui/components/TodoDisplay.tsx +0 -72
- package/dist/ui/components/ToolStatsDisplay.test.tsx +0 -180
- package/dist/ui/components/ToolStatsDisplay.tsx +0 -208
- package/dist/ui/components/UpdateNotification.tsx +0 -23
- package/dist/ui/components/WelcomeBackDialog.tsx +0 -290
- package/dist/ui/components/__snapshots__/IDEContextDetailDisplay.test.tsx.snap +0 -24
- package/dist/ui/components/__snapshots__/ModelStatsDisplay.test.tsx.snap +0 -121
- package/dist/ui/components/__snapshots__/SessionSummaryDisplay.test.tsx.snap +0 -30
- package/dist/ui/components/__snapshots__/ShellConfirmationDialog.test.tsx.snap +0 -21
- package/dist/ui/components/__snapshots__/StatsDisplay.test.tsx.snap +0 -264
- package/dist/ui/components/__snapshots__/ToolStatsDisplay.test.tsx.snap +0 -91
- package/dist/ui/components/messages/CompressionMessage.tsx +0 -49
- package/dist/ui/components/messages/DiffRenderer.test.tsx +0 -365
- package/dist/ui/components/messages/DiffRenderer.tsx +0 -358
- package/dist/ui/components/messages/ErrorMessage.tsx +0 -31
- package/dist/ui/components/messages/GeminiMessage.tsx +0 -43
- package/dist/ui/components/messages/GeminiMessageContent.tsx +0 -43
- package/dist/ui/components/messages/InfoMessage.tsx +0 -32
- package/dist/ui/components/messages/ToolConfirmationMessage.test.tsx +0 -58
- package/dist/ui/components/messages/ToolConfirmationMessage.tsx +0 -297
- package/dist/ui/components/messages/ToolGroupMessage.tsx +0 -126
- package/dist/ui/components/messages/ToolMessage.test.tsx +0 -183
- package/dist/ui/components/messages/ToolMessage.tsx +0 -296
- package/dist/ui/components/messages/UserMessage.tsx +0 -43
- package/dist/ui/components/messages/UserShellMessage.tsx +0 -25
- package/dist/ui/components/shared/MaxSizedBox.test.tsx +0 -425
- package/dist/ui/components/shared/MaxSizedBox.tsx +0 -624
- package/dist/ui/components/shared/RadioButtonSelect.test.tsx +0 -181
- package/dist/ui/components/shared/RadioButtonSelect.tsx +0 -234
- package/dist/ui/components/shared/__snapshots__/RadioButtonSelect.test.tsx.snap +0 -47
- package/dist/ui/components/shared/text-buffer.test.ts +0 -1728
- package/dist/ui/components/shared/text-buffer.ts +0 -2227
- package/dist/ui/components/shared/vim-buffer-actions.test.ts +0 -1119
- package/dist/ui/components/shared/vim-buffer-actions.ts +0 -814
- package/dist/ui/constants.ts +0 -17
- package/dist/ui/contexts/KeypressContext.test.tsx +0 -391
- package/dist/ui/contexts/KeypressContext.tsx +0 -440
- package/dist/ui/contexts/OverflowContext.tsx +0 -87
- package/dist/ui/contexts/SessionContext.test.tsx +0 -132
- package/dist/ui/contexts/SessionContext.tsx +0 -143
- package/dist/ui/contexts/SettingsContext.tsx +0 -20
- package/dist/ui/contexts/StreamingContext.tsx +0 -22
- package/dist/ui/contexts/VimModeContext.tsx +0 -79
- package/dist/ui/editors/editorSettingsManager.ts +0 -66
- package/dist/ui/hooks/atCommandProcessor.test.ts +0 -1102
- package/dist/ui/hooks/atCommandProcessor.ts +0 -485
- package/dist/ui/hooks/shellCommandProcessor.test.ts +0 -481
- package/dist/ui/hooks/shellCommandProcessor.ts +0 -314
- package/dist/ui/hooks/slashCommandProcessor.test.ts +0 -1044
- package/dist/ui/hooks/slashCommandProcessor.ts +0 -595
- package/dist/ui/hooks/useAtCompletion.test.ts +0 -497
- package/dist/ui/hooks/useAtCompletion.ts +0 -244
- package/dist/ui/hooks/useAuthCommand.ts +0 -129
- package/dist/ui/hooks/useAutoAcceptIndicator.test.ts +0 -300
- package/dist/ui/hooks/useAutoAcceptIndicator.ts +0 -52
- package/dist/ui/hooks/useBracketedPaste.ts +0 -37
- package/dist/ui/hooks/useCommandCompletion.test.ts +0 -518
- package/dist/ui/hooks/useCommandCompletion.tsx +0 -238
- package/dist/ui/hooks/useCompletion.ts +0 -128
- package/dist/ui/hooks/useConsoleMessages.test.ts +0 -147
- package/dist/ui/hooks/useConsoleMessages.ts +0 -110
- package/dist/ui/hooks/useEditorSettings.test.ts +0 -283
- package/dist/ui/hooks/useEditorSettings.ts +0 -75
- package/dist/ui/hooks/useFocus.test.ts +0 -119
- package/dist/ui/hooks/useFocus.ts +0 -48
- package/dist/ui/hooks/useFolderTrust.test.ts +0 -159
- package/dist/ui/hooks/useFolderTrust.ts +0 -72
- package/dist/ui/hooks/useGeminiStream.test.tsx +0 -1998
- package/dist/ui/hooks/useGeminiStream.ts +0 -1017
- package/dist/ui/hooks/useGitBranchName.test.ts +0 -280
- package/dist/ui/hooks/useGitBranchName.ts +0 -79
- package/dist/ui/hooks/useHistoryManager.test.ts +0 -202
- package/dist/ui/hooks/useHistoryManager.ts +0 -111
- package/dist/ui/hooks/useInputHistory.test.ts +0 -261
- package/dist/ui/hooks/useInputHistory.ts +0 -111
- package/dist/ui/hooks/useKeypress.test.ts +0 -280
- package/dist/ui/hooks/useKeypress.ts +0 -39
- package/dist/ui/hooks/useKittyKeyboardProtocol.ts +0 -31
- package/dist/ui/hooks/useLoadingIndicator.test.ts +0 -139
- package/dist/ui/hooks/useLoadingIndicator.ts +0 -57
- package/dist/ui/hooks/useLogger.ts +0 -32
- package/dist/ui/hooks/useMessageQueue.test.ts +0 -226
- package/dist/ui/hooks/useMessageQueue.ts +0 -69
- package/dist/ui/hooks/usePhraseCycler.test.ts +0 -145
- package/dist/ui/hooks/usePhraseCycler.ts +0 -198
- package/dist/ui/hooks/usePrivacySettings.test.ts +0 -242
- package/dist/ui/hooks/usePrivacySettings.ts +0 -150
- package/dist/ui/hooks/useReactToolScheduler.ts +0 -309
- package/dist/ui/hooks/useRefreshMemoryCommand.ts +0 -7
- package/dist/ui/hooks/useReverseSearchCompletion.test.tsx +0 -260
- package/dist/ui/hooks/useReverseSearchCompletion.tsx +0 -95
- package/dist/ui/hooks/useSettingsCommand.ts +0 -25
- package/dist/ui/hooks/useShellHistory.test.ts +0 -219
- package/dist/ui/hooks/useShellHistory.ts +0 -133
- package/dist/ui/hooks/useShowMemoryCommand.ts +0 -75
- package/dist/ui/hooks/useSlashCompletion.test.ts +0 -434
- package/dist/ui/hooks/useSlashCompletion.ts +0 -187
- package/dist/ui/hooks/useStateAndRef.ts +0 -36
- package/dist/ui/hooks/useTerminalSize.ts +0 -32
- package/dist/ui/hooks/useThemeCommand.ts +0 -110
- package/dist/ui/hooks/useTimer.test.ts +0 -120
- package/dist/ui/hooks/useTimer.ts +0 -65
- package/dist/ui/hooks/useToolScheduler.test.ts +0 -1123
- package/dist/ui/hooks/useWelcomeBack.ts +0 -253
- package/dist/ui/hooks/vim.test.ts +0 -1691
- package/dist/ui/hooks/vim.ts +0 -784
- package/dist/ui/keyMatchers.test.ts +0 -337
- package/dist/ui/keyMatchers.ts +0 -105
- package/dist/ui/privacy/CloudFreePrivacyNotice.tsx +0 -117
- package/dist/ui/privacy/CloudPaidPrivacyNotice.tsx +0 -59
- package/dist/ui/privacy/GeminiPrivacyNotice.tsx +0 -62
- package/dist/ui/privacy/PrivacyNotice.tsx +0 -42
- package/dist/ui/semantic-colors.ts +0 -26
- package/dist/ui/themes/ansi-light.ts +0 -150
- package/dist/ui/themes/ansi.ts +0 -159
- package/dist/ui/themes/atom-one-dark.ts +0 -147
- package/dist/ui/themes/ayu-light.ts +0 -139
- package/dist/ui/themes/ayu.ts +0 -113
- package/dist/ui/themes/color-utils.test.ts +0 -221
- package/dist/ui/themes/color-utils.ts +0 -231
- package/dist/ui/themes/default-light.ts +0 -108
- package/dist/ui/themes/default.ts +0 -151
- package/dist/ui/themes/dracula.ts +0 -124
- package/dist/ui/themes/fss-code-dark.ts +0 -156
- package/dist/ui/themes/fss-dark.ts +0 -113
- package/dist/ui/themes/fss-light.ts +0 -139
- package/dist/ui/themes/github-dark.ts +0 -147
- package/dist/ui/themes/github-light.ts +0 -149
- package/dist/ui/themes/googlecode.ts +0 -146
- package/dist/ui/themes/no-color.ts +0 -125
- package/dist/ui/themes/qwen-dark.ts +0 -118
- package/dist/ui/themes/qwen-light.ts +0 -144
- package/dist/ui/themes/semantic-tokens.ts +0 -127
- package/dist/ui/themes/shades-of-purple.ts +0 -352
- package/dist/ui/themes/theme-manager.test.ts +0 -99
- package/dist/ui/themes/theme-manager.ts +0 -257
- package/dist/ui/themes/theme.test.ts +0 -97
- package/dist/ui/themes/theme.ts +0 -451
- package/dist/ui/themes/xcode.ts +0 -154
- package/dist/ui/types.ts +0 -255
- package/dist/ui/utils/CodeColorizer.tsx +0 -217
- package/dist/ui/utils/ConsolePatcher.ts +0 -71
- package/dist/ui/utils/InlineMarkdownRenderer.tsx +0 -173
- package/dist/ui/utils/MarkdownDisplay.test.tsx +0 -244
- package/dist/ui/utils/MarkdownDisplay.tsx +0 -415
- package/dist/ui/utils/TableRenderer.tsx +0 -159
- package/dist/ui/utils/__snapshots__/MarkdownDisplay.test.tsx.snap +0 -93
- package/dist/ui/utils/clipboardUtils.test.ts +0 -76
- package/dist/ui/utils/clipboardUtils.ts +0 -149
- package/dist/ui/utils/commandUtils.test.ts +0 -384
- package/dist/ui/utils/commandUtils.ts +0 -106
- package/dist/ui/utils/computeStats.test.ts +0 -292
- package/dist/ui/utils/computeStats.ts +0 -86
- package/dist/ui/utils/displayUtils.test.ts +0 -58
- package/dist/ui/utils/displayUtils.ts +0 -32
- package/dist/ui/utils/formatters.test.ts +0 -72
- package/dist/ui/utils/formatters.ts +0 -63
- package/dist/ui/utils/isNarrowWidth.ts +0 -9
- package/dist/ui/utils/kittyProtocolDetector.ts +0 -105
- package/dist/ui/utils/markdownUtilities.test.ts +0 -50
- package/dist/ui/utils/markdownUtilities.ts +0 -125
- package/dist/ui/utils/platformConstants.ts +0 -52
- package/dist/ui/utils/terminalSetup.ts +0 -342
- package/dist/ui/utils/textUtils.ts +0 -40
- package/dist/ui/utils/updateCheck.test.ts +0 -163
- package/dist/ui/utils/updateCheck.ts +0 -100
- package/dist/utils/checks.ts +0 -28
- package/dist/utils/cleanup.test.ts +0 -68
- package/dist/utils/cleanup.ts +0 -36
- package/dist/utils/dialogScopeUtils.ts +0 -64
- package/dist/utils/events.ts +0 -14
- package/dist/utils/gitUtils.test.ts +0 -149
- package/dist/utils/gitUtils.ts +0 -116
- package/dist/utils/handleAutoUpdate.test.ts +0 -272
- package/dist/utils/handleAutoUpdate.ts +0 -145
- package/dist/utils/installationInfo.test.ts +0 -315
- package/dist/utils/installationInfo.ts +0 -176
- package/dist/utils/package.ts +0 -38
- package/dist/utils/readStdin.ts +0 -51
- package/dist/utils/resolvePath.ts +0 -21
- package/dist/utils/sandbox-macos-permissive-closed.sb +0 -32
- package/dist/utils/sandbox-macos-permissive-open.sb +0 -25
- package/dist/utils/sandbox-macos-permissive-proxied.sb +0 -37
- package/dist/utils/sandbox-macos-restrictive-closed.sb +0 -93
- package/dist/utils/sandbox-macos-restrictive-open.sb +0 -96
- package/dist/utils/sandbox-macos-restrictive-proxied.sb +0 -98
- package/dist/utils/sandbox.ts +0 -962
- package/dist/utils/settingsUtils.test.ts +0 -797
- package/dist/utils/settingsUtils.ts +0 -489
- package/dist/utils/spawnWrapper.ts +0 -9
- package/dist/utils/startupWarnings.test.ts +0 -83
- package/dist/utils/startupWarnings.ts +0 -40
- package/dist/utils/updateEventEmitter.ts +0 -13
- package/dist/utils/userStartupWarnings.test.ts +0 -87
- package/dist/utils/userStartupWarnings.ts +0 -69
- package/dist/utils/version.ts +0 -12
- package/dist/validateNonInterActiveAuth.test.ts +0 -260
- package/dist/validateNonInterActiveAuth.ts +0 -51
- package/dist/vitest.config.ts +0 -37
- package/dist/zed-integration/acp.ts +0 -366
- package/dist/zed-integration/fileSystemService.ts +0 -47
- package/dist/zed-integration/schema.ts +0 -466
- package/dist/zed-integration/zedIntegration.ts +0 -944
|
@@ -1,1424 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
/// <reference types="vitest/globals" />
|
|
8
|
-
|
|
9
|
-
// Mock 'os' first.
|
|
10
|
-
import * as osActual from 'os'; // Import for type info for the mock factory
|
|
11
|
-
vi.mock('os', async (importOriginal) => {
|
|
12
|
-
const actualOs = await importOriginal<typeof osActual>();
|
|
13
|
-
return {
|
|
14
|
-
...actualOs,
|
|
15
|
-
homedir: vi.fn(() => '/mock/home/user'),
|
|
16
|
-
platform: vi.fn(() => 'linux'),
|
|
17
|
-
};
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
// Mock './settings.js' to ensure it uses the mocked 'os.homedir()' for its internal constants.
|
|
21
|
-
vi.mock('./settings.js', async (importActual) => {
|
|
22
|
-
const originalModule = await importActual<typeof import('./settings.js')>();
|
|
23
|
-
return {
|
|
24
|
-
__esModule: true, // Ensure correct module shape
|
|
25
|
-
...originalModule, // Re-export all original members
|
|
26
|
-
// We are relying on originalModule's USER_SETTINGS_PATH being constructed with mocked os.homedir()
|
|
27
|
-
};
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
// NOW import everything else, including the (now effectively re-exported) settings.js
|
|
31
|
-
import * as pathActual from 'path'; // Restored for MOCK_WORKSPACE_SETTINGS_PATH
|
|
32
|
-
import {
|
|
33
|
-
describe,
|
|
34
|
-
it,
|
|
35
|
-
expect,
|
|
36
|
-
vi,
|
|
37
|
-
beforeEach,
|
|
38
|
-
afterEach,
|
|
39
|
-
type Mocked,
|
|
40
|
-
type Mock,
|
|
41
|
-
} from 'vitest';
|
|
42
|
-
import * as fs from 'fs'; // fs will be mocked separately
|
|
43
|
-
import stripJsonComments from 'strip-json-comments'; // Will be mocked separately
|
|
44
|
-
|
|
45
|
-
// These imports will get the versions from the vi.mock('./settings.js', ...) factory.
|
|
46
|
-
import {
|
|
47
|
-
loadSettings,
|
|
48
|
-
USER_SETTINGS_PATH, // This IS the mocked path.
|
|
49
|
-
getSystemSettingsPath,
|
|
50
|
-
SETTINGS_DIRECTORY_NAME, // This is from the original module, but used by the mock.
|
|
51
|
-
SettingScope,
|
|
52
|
-
} from './settings.js';
|
|
53
|
-
|
|
54
|
-
const MOCK_WORKSPACE_DIR = '/mock/workspace';
|
|
55
|
-
// Use the (mocked) SETTINGS_DIRECTORY_NAME for consistency
|
|
56
|
-
const MOCK_WORKSPACE_SETTINGS_PATH = pathActual.join(
|
|
57
|
-
MOCK_WORKSPACE_DIR,
|
|
58
|
-
SETTINGS_DIRECTORY_NAME,
|
|
59
|
-
'settings.json',
|
|
60
|
-
);
|
|
61
|
-
|
|
62
|
-
vi.mock('fs', async (importOriginal) => {
|
|
63
|
-
// Get all the functions from the real 'fs' module
|
|
64
|
-
const actualFs = await importOriginal<typeof fs>();
|
|
65
|
-
|
|
66
|
-
return {
|
|
67
|
-
...actualFs, // Keep all the real functions
|
|
68
|
-
// Now, just override the ones we need for the test
|
|
69
|
-
existsSync: vi.fn(),
|
|
70
|
-
readFileSync: vi.fn(),
|
|
71
|
-
writeFileSync: vi.fn(),
|
|
72
|
-
mkdirSync: vi.fn(),
|
|
73
|
-
realpathSync: (p: string) => p,
|
|
74
|
-
};
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
vi.mock('strip-json-comments', () => ({
|
|
78
|
-
default: vi.fn((content) => content),
|
|
79
|
-
}));
|
|
80
|
-
|
|
81
|
-
describe('Settings Loading and Merging', () => {
|
|
82
|
-
let mockFsExistsSync: Mocked<typeof fs.existsSync>;
|
|
83
|
-
let mockStripJsonComments: Mocked<typeof stripJsonComments>;
|
|
84
|
-
let mockFsMkdirSync: Mocked<typeof fs.mkdirSync>;
|
|
85
|
-
|
|
86
|
-
beforeEach(() => {
|
|
87
|
-
vi.resetAllMocks();
|
|
88
|
-
|
|
89
|
-
mockFsExistsSync = vi.mocked(fs.existsSync);
|
|
90
|
-
mockFsMkdirSync = vi.mocked(fs.mkdirSync);
|
|
91
|
-
mockStripJsonComments = vi.mocked(stripJsonComments);
|
|
92
|
-
|
|
93
|
-
vi.mocked(osActual.homedir).mockReturnValue('/mock/home/user');
|
|
94
|
-
(mockStripJsonComments as unknown as Mock).mockImplementation(
|
|
95
|
-
(jsonString: string) => jsonString,
|
|
96
|
-
);
|
|
97
|
-
(mockFsExistsSync as Mock).mockReturnValue(false);
|
|
98
|
-
(fs.readFileSync as Mock).mockReturnValue('{}'); // Return valid empty JSON
|
|
99
|
-
(mockFsMkdirSync as Mock).mockImplementation(() => undefined);
|
|
100
|
-
});
|
|
101
|
-
|
|
102
|
-
afterEach(() => {
|
|
103
|
-
vi.restoreAllMocks();
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
describe('loadSettings', () => {
|
|
107
|
-
it('should load empty settings if no files exist', () => {
|
|
108
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
109
|
-
expect(settings.system.settings).toEqual({});
|
|
110
|
-
expect(settings.user.settings).toEqual({});
|
|
111
|
-
expect(settings.workspace.settings).toEqual({});
|
|
112
|
-
expect(settings.merged).toEqual({
|
|
113
|
-
customThemes: {},
|
|
114
|
-
mcpServers: {},
|
|
115
|
-
includeDirectories: [],
|
|
116
|
-
chatCompression: {},
|
|
117
|
-
});
|
|
118
|
-
expect(settings.errors.length).toBe(0);
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
it('should load system settings if only system file exists', () => {
|
|
122
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
123
|
-
(p: fs.PathLike) => p === getSystemSettingsPath(),
|
|
124
|
-
);
|
|
125
|
-
const systemSettingsContent = {
|
|
126
|
-
theme: 'system-default',
|
|
127
|
-
sandbox: false,
|
|
128
|
-
};
|
|
129
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
130
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
131
|
-
if (p === getSystemSettingsPath())
|
|
132
|
-
return JSON.stringify(systemSettingsContent);
|
|
133
|
-
return '{}';
|
|
134
|
-
},
|
|
135
|
-
);
|
|
136
|
-
|
|
137
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
138
|
-
|
|
139
|
-
expect(fs.readFileSync).toHaveBeenCalledWith(
|
|
140
|
-
getSystemSettingsPath(),
|
|
141
|
-
'utf-8',
|
|
142
|
-
);
|
|
143
|
-
expect(settings.system.settings).toEqual(systemSettingsContent);
|
|
144
|
-
expect(settings.user.settings).toEqual({});
|
|
145
|
-
expect(settings.workspace.settings).toEqual({});
|
|
146
|
-
expect(settings.merged).toEqual({
|
|
147
|
-
...systemSettingsContent,
|
|
148
|
-
customThemes: {},
|
|
149
|
-
mcpServers: {},
|
|
150
|
-
includeDirectories: [],
|
|
151
|
-
chatCompression: {},
|
|
152
|
-
});
|
|
153
|
-
});
|
|
154
|
-
|
|
155
|
-
it('should load user settings if only user file exists', () => {
|
|
156
|
-
const expectedUserSettingsPath = USER_SETTINGS_PATH; // Use the path actually resolved by the (mocked) module
|
|
157
|
-
|
|
158
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
159
|
-
(p: fs.PathLike) => p === expectedUserSettingsPath,
|
|
160
|
-
);
|
|
161
|
-
const userSettingsContent = {
|
|
162
|
-
theme: 'dark',
|
|
163
|
-
contextFileName: 'USER_CONTEXT.md',
|
|
164
|
-
};
|
|
165
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
166
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
167
|
-
if (p === expectedUserSettingsPath)
|
|
168
|
-
return JSON.stringify(userSettingsContent);
|
|
169
|
-
return '{}';
|
|
170
|
-
},
|
|
171
|
-
);
|
|
172
|
-
|
|
173
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
174
|
-
|
|
175
|
-
expect(fs.readFileSync).toHaveBeenCalledWith(
|
|
176
|
-
expectedUserSettingsPath,
|
|
177
|
-
'utf-8',
|
|
178
|
-
);
|
|
179
|
-
expect(settings.user.settings).toEqual(userSettingsContent);
|
|
180
|
-
expect(settings.workspace.settings).toEqual({});
|
|
181
|
-
expect(settings.merged).toEqual({
|
|
182
|
-
...userSettingsContent,
|
|
183
|
-
customThemes: {},
|
|
184
|
-
mcpServers: {},
|
|
185
|
-
includeDirectories: [],
|
|
186
|
-
chatCompression: {},
|
|
187
|
-
});
|
|
188
|
-
});
|
|
189
|
-
|
|
190
|
-
it('should load workspace settings if only workspace file exists', () => {
|
|
191
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
192
|
-
(p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
193
|
-
);
|
|
194
|
-
const workspaceSettingsContent = {
|
|
195
|
-
sandbox: true,
|
|
196
|
-
contextFileName: 'WORKSPACE_CONTEXT.md',
|
|
197
|
-
};
|
|
198
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
199
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
200
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
201
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
202
|
-
return '';
|
|
203
|
-
},
|
|
204
|
-
);
|
|
205
|
-
|
|
206
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
207
|
-
|
|
208
|
-
expect(fs.readFileSync).toHaveBeenCalledWith(
|
|
209
|
-
MOCK_WORKSPACE_SETTINGS_PATH,
|
|
210
|
-
'utf-8',
|
|
211
|
-
);
|
|
212
|
-
expect(settings.user.settings).toEqual({});
|
|
213
|
-
expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
|
|
214
|
-
expect(settings.merged).toEqual({
|
|
215
|
-
...workspaceSettingsContent,
|
|
216
|
-
customThemes: {},
|
|
217
|
-
mcpServers: {},
|
|
218
|
-
includeDirectories: [],
|
|
219
|
-
chatCompression: {},
|
|
220
|
-
});
|
|
221
|
-
});
|
|
222
|
-
|
|
223
|
-
it('should merge user and workspace settings, with workspace taking precedence', () => {
|
|
224
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
225
|
-
const userSettingsContent = {
|
|
226
|
-
theme: 'dark',
|
|
227
|
-
sandbox: false,
|
|
228
|
-
contextFileName: 'USER_CONTEXT.md',
|
|
229
|
-
};
|
|
230
|
-
const workspaceSettingsContent = {
|
|
231
|
-
sandbox: true,
|
|
232
|
-
coreTools: ['tool1'],
|
|
233
|
-
contextFileName: 'WORKSPACE_CONTEXT.md',
|
|
234
|
-
};
|
|
235
|
-
|
|
236
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
237
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
238
|
-
if (p === USER_SETTINGS_PATH)
|
|
239
|
-
return JSON.stringify(userSettingsContent);
|
|
240
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
241
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
242
|
-
return '';
|
|
243
|
-
},
|
|
244
|
-
);
|
|
245
|
-
|
|
246
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
247
|
-
|
|
248
|
-
expect(settings.user.settings).toEqual(userSettingsContent);
|
|
249
|
-
expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
|
|
250
|
-
expect(settings.merged).toEqual({
|
|
251
|
-
theme: 'dark',
|
|
252
|
-
sandbox: true,
|
|
253
|
-
coreTools: ['tool1'],
|
|
254
|
-
contextFileName: 'WORKSPACE_CONTEXT.md',
|
|
255
|
-
customThemes: {},
|
|
256
|
-
mcpServers: {},
|
|
257
|
-
includeDirectories: [],
|
|
258
|
-
chatCompression: {},
|
|
259
|
-
});
|
|
260
|
-
});
|
|
261
|
-
|
|
262
|
-
it('should merge system, user and workspace settings, with system taking precedence over workspace, and workspace over user', () => {
|
|
263
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
264
|
-
const systemSettingsContent = {
|
|
265
|
-
theme: 'system-theme',
|
|
266
|
-
sandbox: false,
|
|
267
|
-
allowMCPServers: ['server1', 'server2'],
|
|
268
|
-
telemetry: { enabled: false },
|
|
269
|
-
};
|
|
270
|
-
const userSettingsContent = {
|
|
271
|
-
theme: 'dark',
|
|
272
|
-
sandbox: true,
|
|
273
|
-
contextFileName: 'USER_CONTEXT.md',
|
|
274
|
-
};
|
|
275
|
-
const workspaceSettingsContent = {
|
|
276
|
-
sandbox: false,
|
|
277
|
-
coreTools: ['tool1'],
|
|
278
|
-
contextFileName: 'WORKSPACE_CONTEXT.md',
|
|
279
|
-
allowMCPServers: ['server1', 'server2', 'server3'],
|
|
280
|
-
};
|
|
281
|
-
|
|
282
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
283
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
284
|
-
if (p === getSystemSettingsPath())
|
|
285
|
-
return JSON.stringify(systemSettingsContent);
|
|
286
|
-
if (p === USER_SETTINGS_PATH)
|
|
287
|
-
return JSON.stringify(userSettingsContent);
|
|
288
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
289
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
290
|
-
return '';
|
|
291
|
-
},
|
|
292
|
-
);
|
|
293
|
-
|
|
294
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
295
|
-
|
|
296
|
-
expect(settings.system.settings).toEqual(systemSettingsContent);
|
|
297
|
-
expect(settings.user.settings).toEqual(userSettingsContent);
|
|
298
|
-
expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
|
|
299
|
-
expect(settings.merged).toEqual({
|
|
300
|
-
theme: 'system-theme',
|
|
301
|
-
sandbox: false,
|
|
302
|
-
telemetry: { enabled: false },
|
|
303
|
-
coreTools: ['tool1'],
|
|
304
|
-
contextFileName: 'WORKSPACE_CONTEXT.md',
|
|
305
|
-
allowMCPServers: ['server1', 'server2'],
|
|
306
|
-
customThemes: {},
|
|
307
|
-
mcpServers: {},
|
|
308
|
-
includeDirectories: [],
|
|
309
|
-
chatCompression: {},
|
|
310
|
-
});
|
|
311
|
-
});
|
|
312
|
-
|
|
313
|
-
it('should ignore folderTrust from workspace settings', () => {
|
|
314
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
315
|
-
const userSettingsContent = {
|
|
316
|
-
folderTrust: true,
|
|
317
|
-
};
|
|
318
|
-
const workspaceSettingsContent = {
|
|
319
|
-
folderTrust: false, // This should be ignored
|
|
320
|
-
};
|
|
321
|
-
const systemSettingsContent = {
|
|
322
|
-
// No folderTrust here
|
|
323
|
-
};
|
|
324
|
-
|
|
325
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
326
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
327
|
-
if (p === getSystemSettingsPath())
|
|
328
|
-
return JSON.stringify(systemSettingsContent);
|
|
329
|
-
if (p === USER_SETTINGS_PATH)
|
|
330
|
-
return JSON.stringify(userSettingsContent);
|
|
331
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
332
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
333
|
-
return '{}';
|
|
334
|
-
},
|
|
335
|
-
);
|
|
336
|
-
|
|
337
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
338
|
-
expect(settings.merged.folderTrust).toBe(true); // User setting should be used
|
|
339
|
-
});
|
|
340
|
-
|
|
341
|
-
it('should use system folderTrust over user setting', () => {
|
|
342
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
343
|
-
const userSettingsContent = {
|
|
344
|
-
folderTrust: false,
|
|
345
|
-
};
|
|
346
|
-
const workspaceSettingsContent = {
|
|
347
|
-
folderTrust: true, // This should be ignored
|
|
348
|
-
};
|
|
349
|
-
const systemSettingsContent = {
|
|
350
|
-
folderTrust: true,
|
|
351
|
-
};
|
|
352
|
-
|
|
353
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
354
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
355
|
-
if (p === getSystemSettingsPath())
|
|
356
|
-
return JSON.stringify(systemSettingsContent);
|
|
357
|
-
if (p === USER_SETTINGS_PATH)
|
|
358
|
-
return JSON.stringify(userSettingsContent);
|
|
359
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
360
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
361
|
-
return '{}';
|
|
362
|
-
},
|
|
363
|
-
);
|
|
364
|
-
|
|
365
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
366
|
-
expect(settings.merged.folderTrust).toBe(true); // System setting should be used
|
|
367
|
-
});
|
|
368
|
-
|
|
369
|
-
it('should handle contextFileName correctly when only in user settings', () => {
|
|
370
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
371
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
372
|
-
);
|
|
373
|
-
const userSettingsContent = { contextFileName: 'CUSTOM.md' };
|
|
374
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
375
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
376
|
-
if (p === USER_SETTINGS_PATH)
|
|
377
|
-
return JSON.stringify(userSettingsContent);
|
|
378
|
-
return '';
|
|
379
|
-
},
|
|
380
|
-
);
|
|
381
|
-
|
|
382
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
383
|
-
expect(settings.merged.contextFileName).toBe('CUSTOM.md');
|
|
384
|
-
});
|
|
385
|
-
|
|
386
|
-
it('should handle contextFileName correctly when only in workspace settings', () => {
|
|
387
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
388
|
-
(p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
389
|
-
);
|
|
390
|
-
const workspaceSettingsContent = {
|
|
391
|
-
contextFileName: 'PROJECT_SPECIFIC.md',
|
|
392
|
-
};
|
|
393
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
394
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
395
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
396
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
397
|
-
return '';
|
|
398
|
-
},
|
|
399
|
-
);
|
|
400
|
-
|
|
401
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
402
|
-
expect(settings.merged.contextFileName).toBe('PROJECT_SPECIFIC.md');
|
|
403
|
-
});
|
|
404
|
-
|
|
405
|
-
it('should handle excludedProjectEnvVars correctly when only in user settings', () => {
|
|
406
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
407
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
408
|
-
);
|
|
409
|
-
const userSettingsContent = {
|
|
410
|
-
excludedProjectEnvVars: ['DEBUG', 'NODE_ENV', 'CUSTOM_VAR'],
|
|
411
|
-
};
|
|
412
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
413
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
414
|
-
if (p === USER_SETTINGS_PATH)
|
|
415
|
-
return JSON.stringify(userSettingsContent);
|
|
416
|
-
return '';
|
|
417
|
-
},
|
|
418
|
-
);
|
|
419
|
-
|
|
420
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
421
|
-
expect(settings.merged.excludedProjectEnvVars).toEqual([
|
|
422
|
-
'DEBUG',
|
|
423
|
-
'NODE_ENV',
|
|
424
|
-
'CUSTOM_VAR',
|
|
425
|
-
]);
|
|
426
|
-
});
|
|
427
|
-
|
|
428
|
-
it('should handle excludedProjectEnvVars correctly when only in workspace settings', () => {
|
|
429
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
430
|
-
(p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
431
|
-
);
|
|
432
|
-
const workspaceSettingsContent = {
|
|
433
|
-
excludedProjectEnvVars: ['WORKSPACE_DEBUG', 'WORKSPACE_VAR'],
|
|
434
|
-
};
|
|
435
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
436
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
437
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
438
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
439
|
-
return '';
|
|
440
|
-
},
|
|
441
|
-
);
|
|
442
|
-
|
|
443
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
444
|
-
expect(settings.merged.excludedProjectEnvVars).toEqual([
|
|
445
|
-
'WORKSPACE_DEBUG',
|
|
446
|
-
'WORKSPACE_VAR',
|
|
447
|
-
]);
|
|
448
|
-
});
|
|
449
|
-
|
|
450
|
-
it('should merge excludedProjectEnvVars with workspace taking precedence over user', () => {
|
|
451
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
452
|
-
const userSettingsContent = {
|
|
453
|
-
excludedProjectEnvVars: ['DEBUG', 'NODE_ENV', 'USER_VAR'],
|
|
454
|
-
};
|
|
455
|
-
const workspaceSettingsContent = {
|
|
456
|
-
excludedProjectEnvVars: ['WORKSPACE_DEBUG', 'WORKSPACE_VAR'],
|
|
457
|
-
};
|
|
458
|
-
|
|
459
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
460
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
461
|
-
if (p === USER_SETTINGS_PATH)
|
|
462
|
-
return JSON.stringify(userSettingsContent);
|
|
463
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
464
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
465
|
-
return '';
|
|
466
|
-
},
|
|
467
|
-
);
|
|
468
|
-
|
|
469
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
470
|
-
expect(settings.user.settings.excludedProjectEnvVars).toEqual([
|
|
471
|
-
'DEBUG',
|
|
472
|
-
'NODE_ENV',
|
|
473
|
-
'USER_VAR',
|
|
474
|
-
]);
|
|
475
|
-
expect(settings.workspace.settings.excludedProjectEnvVars).toEqual([
|
|
476
|
-
'WORKSPACE_DEBUG',
|
|
477
|
-
'WORKSPACE_VAR',
|
|
478
|
-
]);
|
|
479
|
-
expect(settings.merged.excludedProjectEnvVars).toEqual([
|
|
480
|
-
'WORKSPACE_DEBUG',
|
|
481
|
-
'WORKSPACE_VAR',
|
|
482
|
-
]);
|
|
483
|
-
});
|
|
484
|
-
|
|
485
|
-
it('should default contextFileName to undefined if not in any settings file', () => {
|
|
486
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
487
|
-
const userSettingsContent = { theme: 'dark' };
|
|
488
|
-
const workspaceSettingsContent = { sandbox: true };
|
|
489
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
490
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
491
|
-
if (p === USER_SETTINGS_PATH)
|
|
492
|
-
return JSON.stringify(userSettingsContent);
|
|
493
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
494
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
495
|
-
return '';
|
|
496
|
-
},
|
|
497
|
-
);
|
|
498
|
-
|
|
499
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
500
|
-
expect(settings.merged.contextFileName).toBeUndefined();
|
|
501
|
-
});
|
|
502
|
-
|
|
503
|
-
it('should load telemetry setting from user settings', () => {
|
|
504
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
505
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
506
|
-
);
|
|
507
|
-
const userSettingsContent = { telemetry: true };
|
|
508
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
509
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
510
|
-
if (p === USER_SETTINGS_PATH)
|
|
511
|
-
return JSON.stringify(userSettingsContent);
|
|
512
|
-
return '{}';
|
|
513
|
-
},
|
|
514
|
-
);
|
|
515
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
516
|
-
expect(settings.merged.telemetry).toBe(true);
|
|
517
|
-
});
|
|
518
|
-
|
|
519
|
-
it('should load telemetry setting from workspace settings', () => {
|
|
520
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
521
|
-
(p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
522
|
-
);
|
|
523
|
-
const workspaceSettingsContent = { telemetry: false };
|
|
524
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
525
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
526
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
527
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
528
|
-
return '{}';
|
|
529
|
-
},
|
|
530
|
-
);
|
|
531
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
532
|
-
expect(settings.merged.telemetry).toBe(false);
|
|
533
|
-
});
|
|
534
|
-
|
|
535
|
-
it('should prioritize workspace telemetry setting over user setting', () => {
|
|
536
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
537
|
-
const userSettingsContent = { telemetry: true };
|
|
538
|
-
const workspaceSettingsContent = { telemetry: false };
|
|
539
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
540
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
541
|
-
if (p === USER_SETTINGS_PATH)
|
|
542
|
-
return JSON.stringify(userSettingsContent);
|
|
543
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
544
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
545
|
-
return '{}';
|
|
546
|
-
},
|
|
547
|
-
);
|
|
548
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
549
|
-
expect(settings.merged.telemetry).toBe(false);
|
|
550
|
-
});
|
|
551
|
-
|
|
552
|
-
it('should have telemetry as undefined if not in any settings file', () => {
|
|
553
|
-
(mockFsExistsSync as Mock).mockReturnValue(false); // No settings files exist
|
|
554
|
-
(fs.readFileSync as Mock).mockReturnValue('{}');
|
|
555
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
556
|
-
expect(settings.merged.telemetry).toBeUndefined();
|
|
557
|
-
expect(settings.merged.customThemes).toEqual({});
|
|
558
|
-
expect(settings.merged.mcpServers).toEqual({});
|
|
559
|
-
});
|
|
560
|
-
|
|
561
|
-
it('should merge MCP servers correctly, with workspace taking precedence', () => {
|
|
562
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
563
|
-
const userSettingsContent = {
|
|
564
|
-
mcpServers: {
|
|
565
|
-
'user-server': {
|
|
566
|
-
command: 'user-command',
|
|
567
|
-
args: ['--user-arg'],
|
|
568
|
-
description: 'User MCP server',
|
|
569
|
-
},
|
|
570
|
-
'shared-server': {
|
|
571
|
-
command: 'user-shared-command',
|
|
572
|
-
description: 'User shared server config',
|
|
573
|
-
},
|
|
574
|
-
},
|
|
575
|
-
};
|
|
576
|
-
const workspaceSettingsContent = {
|
|
577
|
-
mcpServers: {
|
|
578
|
-
'workspace-server': {
|
|
579
|
-
command: 'workspace-command',
|
|
580
|
-
args: ['--workspace-arg'],
|
|
581
|
-
description: 'Workspace MCP server',
|
|
582
|
-
},
|
|
583
|
-
'shared-server': {
|
|
584
|
-
command: 'workspace-shared-command',
|
|
585
|
-
description: 'Workspace shared server config',
|
|
586
|
-
},
|
|
587
|
-
},
|
|
588
|
-
};
|
|
589
|
-
|
|
590
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
591
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
592
|
-
if (p === USER_SETTINGS_PATH)
|
|
593
|
-
return JSON.stringify(userSettingsContent);
|
|
594
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
595
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
596
|
-
return '';
|
|
597
|
-
},
|
|
598
|
-
);
|
|
599
|
-
|
|
600
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
601
|
-
|
|
602
|
-
expect(settings.user.settings).toEqual(userSettingsContent);
|
|
603
|
-
expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
|
|
604
|
-
expect(settings.merged.mcpServers).toEqual({
|
|
605
|
-
'user-server': {
|
|
606
|
-
command: 'user-command',
|
|
607
|
-
args: ['--user-arg'],
|
|
608
|
-
description: 'User MCP server',
|
|
609
|
-
},
|
|
610
|
-
'workspace-server': {
|
|
611
|
-
command: 'workspace-command',
|
|
612
|
-
args: ['--workspace-arg'],
|
|
613
|
-
description: 'Workspace MCP server',
|
|
614
|
-
},
|
|
615
|
-
'shared-server': {
|
|
616
|
-
command: 'workspace-shared-command',
|
|
617
|
-
description: 'Workspace shared server config',
|
|
618
|
-
},
|
|
619
|
-
});
|
|
620
|
-
});
|
|
621
|
-
|
|
622
|
-
it('should handle MCP servers when only in user settings', () => {
|
|
623
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
624
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
625
|
-
);
|
|
626
|
-
const userSettingsContent = {
|
|
627
|
-
mcpServers: {
|
|
628
|
-
'user-only-server': {
|
|
629
|
-
command: 'user-only-command',
|
|
630
|
-
description: 'User only server',
|
|
631
|
-
},
|
|
632
|
-
},
|
|
633
|
-
};
|
|
634
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
635
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
636
|
-
if (p === USER_SETTINGS_PATH)
|
|
637
|
-
return JSON.stringify(userSettingsContent);
|
|
638
|
-
return '';
|
|
639
|
-
},
|
|
640
|
-
);
|
|
641
|
-
|
|
642
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
643
|
-
expect(settings.merged.mcpServers).toEqual({
|
|
644
|
-
'user-only-server': {
|
|
645
|
-
command: 'user-only-command',
|
|
646
|
-
description: 'User only server',
|
|
647
|
-
},
|
|
648
|
-
});
|
|
649
|
-
});
|
|
650
|
-
|
|
651
|
-
it('should handle MCP servers when only in workspace settings', () => {
|
|
652
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
653
|
-
(p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
654
|
-
);
|
|
655
|
-
const workspaceSettingsContent = {
|
|
656
|
-
mcpServers: {
|
|
657
|
-
'workspace-only-server': {
|
|
658
|
-
command: 'workspace-only-command',
|
|
659
|
-
description: 'Workspace only server',
|
|
660
|
-
},
|
|
661
|
-
},
|
|
662
|
-
};
|
|
663
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
664
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
665
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
666
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
667
|
-
return '';
|
|
668
|
-
},
|
|
669
|
-
);
|
|
670
|
-
|
|
671
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
672
|
-
expect(settings.merged.mcpServers).toEqual({
|
|
673
|
-
'workspace-only-server': {
|
|
674
|
-
command: 'workspace-only-command',
|
|
675
|
-
description: 'Workspace only server',
|
|
676
|
-
},
|
|
677
|
-
});
|
|
678
|
-
});
|
|
679
|
-
|
|
680
|
-
it('should have mcpServers as empty object if not in any settings file', () => {
|
|
681
|
-
(mockFsExistsSync as Mock).mockReturnValue(false); // No settings files exist
|
|
682
|
-
(fs.readFileSync as Mock).mockReturnValue('{}');
|
|
683
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
684
|
-
expect(settings.merged.mcpServers).toEqual({});
|
|
685
|
-
});
|
|
686
|
-
|
|
687
|
-
it('should merge chatCompression settings, with workspace taking precedence', () => {
|
|
688
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
689
|
-
const userSettingsContent = {
|
|
690
|
-
chatCompression: { contextPercentageThreshold: 0.5 },
|
|
691
|
-
};
|
|
692
|
-
const workspaceSettingsContent = {
|
|
693
|
-
chatCompression: { contextPercentageThreshold: 0.8 },
|
|
694
|
-
};
|
|
695
|
-
|
|
696
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
697
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
698
|
-
if (p === USER_SETTINGS_PATH)
|
|
699
|
-
return JSON.stringify(userSettingsContent);
|
|
700
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
701
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
702
|
-
return '{}';
|
|
703
|
-
},
|
|
704
|
-
);
|
|
705
|
-
|
|
706
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
707
|
-
|
|
708
|
-
expect(settings.user.settings.chatCompression).toEqual({
|
|
709
|
-
contextPercentageThreshold: 0.5,
|
|
710
|
-
});
|
|
711
|
-
expect(settings.workspace.settings.chatCompression).toEqual({
|
|
712
|
-
contextPercentageThreshold: 0.8,
|
|
713
|
-
});
|
|
714
|
-
expect(settings.merged.chatCompression).toEqual({
|
|
715
|
-
contextPercentageThreshold: 0.8,
|
|
716
|
-
});
|
|
717
|
-
});
|
|
718
|
-
|
|
719
|
-
it('should handle chatCompression when only in user settings', () => {
|
|
720
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
721
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
722
|
-
);
|
|
723
|
-
const userSettingsContent = {
|
|
724
|
-
chatCompression: { contextPercentageThreshold: 0.5 },
|
|
725
|
-
};
|
|
726
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
727
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
728
|
-
if (p === USER_SETTINGS_PATH)
|
|
729
|
-
return JSON.stringify(userSettingsContent);
|
|
730
|
-
return '{}';
|
|
731
|
-
},
|
|
732
|
-
);
|
|
733
|
-
|
|
734
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
735
|
-
expect(settings.merged.chatCompression).toEqual({
|
|
736
|
-
contextPercentageThreshold: 0.5,
|
|
737
|
-
});
|
|
738
|
-
});
|
|
739
|
-
|
|
740
|
-
it('should have chatCompression as an empty object if not in any settings file', () => {
|
|
741
|
-
(mockFsExistsSync as Mock).mockReturnValue(false); // No settings files exist
|
|
742
|
-
(fs.readFileSync as Mock).mockReturnValue('{}');
|
|
743
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
744
|
-
expect(settings.merged.chatCompression).toEqual({});
|
|
745
|
-
});
|
|
746
|
-
|
|
747
|
-
it('should ignore chatCompression if contextPercentageThreshold is invalid', () => {
|
|
748
|
-
const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => {});
|
|
749
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
750
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
751
|
-
);
|
|
752
|
-
const userSettingsContent = {
|
|
753
|
-
chatCompression: { contextPercentageThreshold: 1.5 },
|
|
754
|
-
};
|
|
755
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
756
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
757
|
-
if (p === USER_SETTINGS_PATH)
|
|
758
|
-
return JSON.stringify(userSettingsContent);
|
|
759
|
-
return '{}';
|
|
760
|
-
},
|
|
761
|
-
);
|
|
762
|
-
|
|
763
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
764
|
-
expect(settings.merged.chatCompression).toBeUndefined();
|
|
765
|
-
expect(warnSpy).toHaveBeenCalledWith(
|
|
766
|
-
'Invalid value for chatCompression.contextPercentageThreshold: "1.5". Please use a value between 0 and 1. Using default compression settings.',
|
|
767
|
-
);
|
|
768
|
-
warnSpy.mockRestore();
|
|
769
|
-
});
|
|
770
|
-
|
|
771
|
-
it('should deep merge chatCompression settings', () => {
|
|
772
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
773
|
-
const userSettingsContent = {
|
|
774
|
-
chatCompression: { contextPercentageThreshold: 0.5 },
|
|
775
|
-
};
|
|
776
|
-
const workspaceSettingsContent = {
|
|
777
|
-
chatCompression: {},
|
|
778
|
-
};
|
|
779
|
-
|
|
780
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
781
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
782
|
-
if (p === USER_SETTINGS_PATH)
|
|
783
|
-
return JSON.stringify(userSettingsContent);
|
|
784
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
785
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
786
|
-
return '{}';
|
|
787
|
-
},
|
|
788
|
-
);
|
|
789
|
-
|
|
790
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
791
|
-
|
|
792
|
-
expect(settings.merged.chatCompression).toEqual({
|
|
793
|
-
contextPercentageThreshold: 0.5,
|
|
794
|
-
});
|
|
795
|
-
});
|
|
796
|
-
|
|
797
|
-
it('should merge includeDirectories from all scopes', () => {
|
|
798
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
799
|
-
const systemSettingsContent = {
|
|
800
|
-
includeDirectories: ['/system/dir'],
|
|
801
|
-
};
|
|
802
|
-
const userSettingsContent = {
|
|
803
|
-
includeDirectories: ['/user/dir1', '/user/dir2'],
|
|
804
|
-
};
|
|
805
|
-
const workspaceSettingsContent = {
|
|
806
|
-
includeDirectories: ['/workspace/dir'],
|
|
807
|
-
};
|
|
808
|
-
|
|
809
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
810
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
811
|
-
if (p === getSystemSettingsPath())
|
|
812
|
-
return JSON.stringify(systemSettingsContent);
|
|
813
|
-
if (p === USER_SETTINGS_PATH)
|
|
814
|
-
return JSON.stringify(userSettingsContent);
|
|
815
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
816
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
817
|
-
return '{}';
|
|
818
|
-
},
|
|
819
|
-
);
|
|
820
|
-
|
|
821
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
822
|
-
|
|
823
|
-
expect(settings.merged.includeDirectories).toEqual([
|
|
824
|
-
'/system/dir',
|
|
825
|
-
'/user/dir1',
|
|
826
|
-
'/user/dir2',
|
|
827
|
-
'/workspace/dir',
|
|
828
|
-
]);
|
|
829
|
-
});
|
|
830
|
-
|
|
831
|
-
it('should handle JSON parsing errors gracefully', () => {
|
|
832
|
-
(mockFsExistsSync as Mock).mockReturnValue(true); // Both files "exist"
|
|
833
|
-
const invalidJsonContent = 'invalid json';
|
|
834
|
-
const userReadError = new SyntaxError(
|
|
835
|
-
"Expected ',' or '}' after property value in JSON at position 10",
|
|
836
|
-
);
|
|
837
|
-
const workspaceReadError = new SyntaxError(
|
|
838
|
-
'Unexpected token i in JSON at position 0',
|
|
839
|
-
);
|
|
840
|
-
|
|
841
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
842
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
843
|
-
if (p === USER_SETTINGS_PATH) {
|
|
844
|
-
// Simulate JSON.parse throwing for user settings
|
|
845
|
-
vi.spyOn(JSON, 'parse').mockImplementationOnce(() => {
|
|
846
|
-
throw userReadError;
|
|
847
|
-
});
|
|
848
|
-
return invalidJsonContent; // Content that would cause JSON.parse to throw
|
|
849
|
-
}
|
|
850
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH) {
|
|
851
|
-
// Simulate JSON.parse throwing for workspace settings
|
|
852
|
-
vi.spyOn(JSON, 'parse').mockImplementationOnce(() => {
|
|
853
|
-
throw workspaceReadError;
|
|
854
|
-
});
|
|
855
|
-
return invalidJsonContent;
|
|
856
|
-
}
|
|
857
|
-
return '{}'; // Default for other reads
|
|
858
|
-
},
|
|
859
|
-
);
|
|
860
|
-
|
|
861
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
862
|
-
|
|
863
|
-
// Check that settings are empty due to parsing errors
|
|
864
|
-
expect(settings.user.settings).toEqual({});
|
|
865
|
-
expect(settings.workspace.settings).toEqual({});
|
|
866
|
-
expect(settings.merged).toEqual({
|
|
867
|
-
customThemes: {},
|
|
868
|
-
mcpServers: {},
|
|
869
|
-
includeDirectories: [],
|
|
870
|
-
chatCompression: {},
|
|
871
|
-
});
|
|
872
|
-
|
|
873
|
-
// Check that error objects are populated in settings.errors
|
|
874
|
-
expect(settings.errors).toBeDefined();
|
|
875
|
-
// Assuming both user and workspace files cause errors and are added in order
|
|
876
|
-
expect(settings.errors.length).toEqual(2);
|
|
877
|
-
|
|
878
|
-
const userError = settings.errors.find(
|
|
879
|
-
(e) => e.path === USER_SETTINGS_PATH,
|
|
880
|
-
);
|
|
881
|
-
expect(userError).toBeDefined();
|
|
882
|
-
expect(userError?.message).toBe(userReadError.message);
|
|
883
|
-
|
|
884
|
-
const workspaceError = settings.errors.find(
|
|
885
|
-
(e) => e.path === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
886
|
-
);
|
|
887
|
-
expect(workspaceError).toBeDefined();
|
|
888
|
-
expect(workspaceError?.message).toBe(workspaceReadError.message);
|
|
889
|
-
|
|
890
|
-
// Restore JSON.parse mock if it was spied on specifically for this test
|
|
891
|
-
vi.restoreAllMocks(); // Or more targeted restore if needed
|
|
892
|
-
});
|
|
893
|
-
|
|
894
|
-
it('should resolve environment variables in user settings', () => {
|
|
895
|
-
process.env['TEST_API_KEY'] = 'user_api_key_from_env';
|
|
896
|
-
const userSettingsContent = {
|
|
897
|
-
apiKey: '$TEST_API_KEY',
|
|
898
|
-
someUrl: 'https://test.com/${TEST_API_KEY}',
|
|
899
|
-
};
|
|
900
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
901
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
902
|
-
);
|
|
903
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
904
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
905
|
-
if (p === USER_SETTINGS_PATH)
|
|
906
|
-
return JSON.stringify(userSettingsContent);
|
|
907
|
-
return '{}';
|
|
908
|
-
},
|
|
909
|
-
);
|
|
910
|
-
|
|
911
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
912
|
-
// @ts-expect-error: dynamic property for test
|
|
913
|
-
expect(settings.user.settings.apiKey).toBe('user_api_key_from_env');
|
|
914
|
-
// @ts-expect-error: dynamic property for test
|
|
915
|
-
expect(settings.user.settings.someUrl).toBe(
|
|
916
|
-
'https://test.com/user_api_key_from_env',
|
|
917
|
-
);
|
|
918
|
-
// @ts-expect-error: dynamic property for test
|
|
919
|
-
expect(settings.merged.apiKey).toBe('user_api_key_from_env');
|
|
920
|
-
delete process.env['TEST_API_KEY'];
|
|
921
|
-
});
|
|
922
|
-
|
|
923
|
-
it('should resolve environment variables in workspace settings', () => {
|
|
924
|
-
process.env['WORKSPACE_ENDPOINT'] = 'workspace_endpoint_from_env';
|
|
925
|
-
const workspaceSettingsContent = {
|
|
926
|
-
endpoint: '${WORKSPACE_ENDPOINT}/api',
|
|
927
|
-
nested: { value: '$WORKSPACE_ENDPOINT' },
|
|
928
|
-
};
|
|
929
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
930
|
-
(p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
931
|
-
);
|
|
932
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
933
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
934
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
935
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
936
|
-
return '{}';
|
|
937
|
-
},
|
|
938
|
-
);
|
|
939
|
-
|
|
940
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
941
|
-
expect(settings.workspace.settings.endpoint).toBe(
|
|
942
|
-
'workspace_endpoint_from_env/api',
|
|
943
|
-
);
|
|
944
|
-
expect(settings.workspace.settings.nested.value).toBe(
|
|
945
|
-
'workspace_endpoint_from_env',
|
|
946
|
-
);
|
|
947
|
-
// @ts-expect-error: dynamic property for test
|
|
948
|
-
expect(settings.merged.endpoint).toBe('workspace_endpoint_from_env/api');
|
|
949
|
-
delete process.env['WORKSPACE_ENDPOINT'];
|
|
950
|
-
});
|
|
951
|
-
|
|
952
|
-
it('should correctly resolve and merge env variables from different scopes', () => {
|
|
953
|
-
process.env['SYSTEM_VAR'] = 'system_value';
|
|
954
|
-
process.env['USER_VAR'] = 'user_value';
|
|
955
|
-
process.env['WORKSPACE_VAR'] = 'workspace_value';
|
|
956
|
-
process.env['SHARED_VAR'] = 'final_value';
|
|
957
|
-
|
|
958
|
-
const systemSettingsContent = {
|
|
959
|
-
configValue: '$SHARED_VAR',
|
|
960
|
-
systemOnly: '$SYSTEM_VAR',
|
|
961
|
-
};
|
|
962
|
-
const userSettingsContent = {
|
|
963
|
-
configValue: '$SHARED_VAR',
|
|
964
|
-
userOnly: '$USER_VAR',
|
|
965
|
-
theme: 'dark',
|
|
966
|
-
};
|
|
967
|
-
const workspaceSettingsContent = {
|
|
968
|
-
configValue: '$SHARED_VAR',
|
|
969
|
-
workspaceOnly: '$WORKSPACE_VAR',
|
|
970
|
-
theme: 'light',
|
|
971
|
-
};
|
|
972
|
-
|
|
973
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
974
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
975
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
976
|
-
if (p === getSystemSettingsPath()) {
|
|
977
|
-
return JSON.stringify(systemSettingsContent);
|
|
978
|
-
}
|
|
979
|
-
if (p === USER_SETTINGS_PATH) {
|
|
980
|
-
return JSON.stringify(userSettingsContent);
|
|
981
|
-
}
|
|
982
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH) {
|
|
983
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
984
|
-
}
|
|
985
|
-
return '{}';
|
|
986
|
-
},
|
|
987
|
-
);
|
|
988
|
-
|
|
989
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
990
|
-
|
|
991
|
-
// Check resolved values in individual scopes
|
|
992
|
-
// @ts-expect-error: dynamic property for test
|
|
993
|
-
expect(settings.system.settings.configValue).toBe('final_value');
|
|
994
|
-
// @ts-expect-error: dynamic property for test
|
|
995
|
-
expect(settings.system.settings.systemOnly).toBe('system_value');
|
|
996
|
-
// @ts-expect-error: dynamic property for test
|
|
997
|
-
expect(settings.user.settings.configValue).toBe('final_value');
|
|
998
|
-
// @ts-expect-error: dynamic property for test
|
|
999
|
-
expect(settings.user.settings.userOnly).toBe('user_value');
|
|
1000
|
-
// @ts-expect-error: dynamic property for test
|
|
1001
|
-
expect(settings.workspace.settings.configValue).toBe('final_value');
|
|
1002
|
-
// @ts-expect-error: dynamic property for test
|
|
1003
|
-
expect(settings.workspace.settings.workspaceOnly).toBe('workspace_value');
|
|
1004
|
-
|
|
1005
|
-
// Check merged values (system > workspace > user)
|
|
1006
|
-
// @ts-expect-error: dynamic property for test
|
|
1007
|
-
expect(settings.merged.configValue).toBe('final_value');
|
|
1008
|
-
// @ts-expect-error: dynamic property for test
|
|
1009
|
-
expect(settings.merged.systemOnly).toBe('system_value');
|
|
1010
|
-
// @ts-expect-error: dynamic property for test
|
|
1011
|
-
expect(settings.merged.userOnly).toBe('user_value');
|
|
1012
|
-
// @ts-expect-error: dynamic property for test
|
|
1013
|
-
expect(settings.merged.workspaceOnly).toBe('workspace_value');
|
|
1014
|
-
expect(settings.merged.theme).toBe('light'); // workspace overrides user
|
|
1015
|
-
|
|
1016
|
-
delete process.env['SYSTEM_VAR'];
|
|
1017
|
-
delete process.env['USER_VAR'];
|
|
1018
|
-
delete process.env['WORKSPACE_VAR'];
|
|
1019
|
-
delete process.env['SHARED_VAR'];
|
|
1020
|
-
});
|
|
1021
|
-
|
|
1022
|
-
it('should correctly merge dnsResolutionOrder with workspace taking precedence', () => {
|
|
1023
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
1024
|
-
const userSettingsContent = {
|
|
1025
|
-
dnsResolutionOrder: 'ipv4first',
|
|
1026
|
-
};
|
|
1027
|
-
const workspaceSettingsContent = {
|
|
1028
|
-
dnsResolutionOrder: 'verbatim',
|
|
1029
|
-
};
|
|
1030
|
-
|
|
1031
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1032
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1033
|
-
if (p === USER_SETTINGS_PATH)
|
|
1034
|
-
return JSON.stringify(userSettingsContent);
|
|
1035
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
1036
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
1037
|
-
return '{}';
|
|
1038
|
-
},
|
|
1039
|
-
);
|
|
1040
|
-
|
|
1041
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1042
|
-
expect(settings.merged.dnsResolutionOrder).toBe('verbatim');
|
|
1043
|
-
});
|
|
1044
|
-
|
|
1045
|
-
it('should use user dnsResolutionOrder if workspace is not defined', () => {
|
|
1046
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1047
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
1048
|
-
);
|
|
1049
|
-
const userSettingsContent = {
|
|
1050
|
-
dnsResolutionOrder: 'verbatim',
|
|
1051
|
-
};
|
|
1052
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1053
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1054
|
-
if (p === USER_SETTINGS_PATH)
|
|
1055
|
-
return JSON.stringify(userSettingsContent);
|
|
1056
|
-
return '{}';
|
|
1057
|
-
},
|
|
1058
|
-
);
|
|
1059
|
-
|
|
1060
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1061
|
-
expect(settings.merged.dnsResolutionOrder).toBe('verbatim');
|
|
1062
|
-
});
|
|
1063
|
-
|
|
1064
|
-
it('should leave unresolved environment variables as is', () => {
|
|
1065
|
-
const userSettingsContent = { apiKey: '$UNDEFINED_VAR' };
|
|
1066
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1067
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
1068
|
-
);
|
|
1069
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1070
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1071
|
-
if (p === USER_SETTINGS_PATH)
|
|
1072
|
-
return JSON.stringify(userSettingsContent);
|
|
1073
|
-
return '{}';
|
|
1074
|
-
},
|
|
1075
|
-
);
|
|
1076
|
-
|
|
1077
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1078
|
-
expect(settings.user.settings.apiKey).toBe('$UNDEFINED_VAR');
|
|
1079
|
-
expect(settings.merged.apiKey).toBe('$UNDEFINED_VAR');
|
|
1080
|
-
});
|
|
1081
|
-
|
|
1082
|
-
it('should resolve multiple environment variables in a single string', () => {
|
|
1083
|
-
process.env['VAR_A'] = 'valueA';
|
|
1084
|
-
process.env['VAR_B'] = 'valueB';
|
|
1085
|
-
const userSettingsContent = { path: '/path/$VAR_A/${VAR_B}/end' };
|
|
1086
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1087
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
1088
|
-
);
|
|
1089
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1090
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1091
|
-
if (p === USER_SETTINGS_PATH)
|
|
1092
|
-
return JSON.stringify(userSettingsContent);
|
|
1093
|
-
return '{}';
|
|
1094
|
-
},
|
|
1095
|
-
);
|
|
1096
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1097
|
-
expect(settings.user.settings.path).toBe('/path/valueA/valueB/end');
|
|
1098
|
-
delete process.env['VAR_A'];
|
|
1099
|
-
delete process.env['VAR_B'];
|
|
1100
|
-
});
|
|
1101
|
-
|
|
1102
|
-
it('should resolve environment variables in arrays', () => {
|
|
1103
|
-
process.env['ITEM_1'] = 'item1_env';
|
|
1104
|
-
process.env['ITEM_2'] = 'item2_env';
|
|
1105
|
-
const userSettingsContent = { list: ['$ITEM_1', '${ITEM_2}', 'literal'] };
|
|
1106
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1107
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
1108
|
-
);
|
|
1109
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1110
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1111
|
-
if (p === USER_SETTINGS_PATH)
|
|
1112
|
-
return JSON.stringify(userSettingsContent);
|
|
1113
|
-
return '{}';
|
|
1114
|
-
},
|
|
1115
|
-
);
|
|
1116
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1117
|
-
expect(settings.user.settings.list).toEqual([
|
|
1118
|
-
'item1_env',
|
|
1119
|
-
'item2_env',
|
|
1120
|
-
'literal',
|
|
1121
|
-
]);
|
|
1122
|
-
delete process.env['ITEM_1'];
|
|
1123
|
-
delete process.env['ITEM_2'];
|
|
1124
|
-
});
|
|
1125
|
-
|
|
1126
|
-
it('should correctly pass through null, boolean, and number types, and handle undefined properties', () => {
|
|
1127
|
-
process.env['MY_ENV_STRING'] = 'env_string_value';
|
|
1128
|
-
process.env['MY_ENV_STRING_NESTED'] = 'env_string_nested_value';
|
|
1129
|
-
|
|
1130
|
-
const userSettingsContent = {
|
|
1131
|
-
nullVal: null,
|
|
1132
|
-
trueVal: true,
|
|
1133
|
-
falseVal: false,
|
|
1134
|
-
numberVal: 123.45,
|
|
1135
|
-
stringVal: '$MY_ENV_STRING',
|
|
1136
|
-
nestedObj: {
|
|
1137
|
-
nestedNull: null,
|
|
1138
|
-
nestedBool: true,
|
|
1139
|
-
nestedNum: 0,
|
|
1140
|
-
nestedString: 'literal',
|
|
1141
|
-
anotherEnv: '${MY_ENV_STRING_NESTED}',
|
|
1142
|
-
},
|
|
1143
|
-
};
|
|
1144
|
-
|
|
1145
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1146
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
1147
|
-
);
|
|
1148
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1149
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1150
|
-
if (p === USER_SETTINGS_PATH)
|
|
1151
|
-
return JSON.stringify(userSettingsContent);
|
|
1152
|
-
return '{}';
|
|
1153
|
-
},
|
|
1154
|
-
);
|
|
1155
|
-
|
|
1156
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1157
|
-
|
|
1158
|
-
expect(settings.user.settings.nullVal).toBeNull();
|
|
1159
|
-
expect(settings.user.settings.trueVal).toBe(true);
|
|
1160
|
-
expect(settings.user.settings.falseVal).toBe(false);
|
|
1161
|
-
expect(settings.user.settings.numberVal).toBe(123.45);
|
|
1162
|
-
expect(settings.user.settings.stringVal).toBe('env_string_value');
|
|
1163
|
-
expect(settings.user.settings.undefinedVal).toBeUndefined();
|
|
1164
|
-
|
|
1165
|
-
expect(settings.user.settings.nestedObj.nestedNull).toBeNull();
|
|
1166
|
-
expect(settings.user.settings.nestedObj.nestedBool).toBe(true);
|
|
1167
|
-
expect(settings.user.settings.nestedObj.nestedNum).toBe(0);
|
|
1168
|
-
expect(settings.user.settings.nestedObj.nestedString).toBe('literal');
|
|
1169
|
-
expect(settings.user.settings.nestedObj.anotherEnv).toBe(
|
|
1170
|
-
'env_string_nested_value',
|
|
1171
|
-
);
|
|
1172
|
-
|
|
1173
|
-
delete process.env['MY_ENV_STRING'];
|
|
1174
|
-
delete process.env['MY_ENV_STRING_NESTED'];
|
|
1175
|
-
});
|
|
1176
|
-
|
|
1177
|
-
it('should resolve multiple concatenated environment variables in a single string value', () => {
|
|
1178
|
-
process.env['TEST_HOST'] = 'myhost';
|
|
1179
|
-
process.env['TEST_PORT'] = '9090';
|
|
1180
|
-
const userSettingsContent = {
|
|
1181
|
-
serverAddress: '${TEST_HOST}:${TEST_PORT}/api',
|
|
1182
|
-
};
|
|
1183
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1184
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
1185
|
-
);
|
|
1186
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1187
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1188
|
-
if (p === USER_SETTINGS_PATH)
|
|
1189
|
-
return JSON.stringify(userSettingsContent);
|
|
1190
|
-
return '{}';
|
|
1191
|
-
},
|
|
1192
|
-
);
|
|
1193
|
-
|
|
1194
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1195
|
-
expect(settings.user.settings.serverAddress).toBe('myhost:9090/api');
|
|
1196
|
-
|
|
1197
|
-
delete process.env['TEST_HOST'];
|
|
1198
|
-
delete process.env['TEST_PORT'];
|
|
1199
|
-
});
|
|
1200
|
-
|
|
1201
|
-
describe('when FSS_LINK_CLI_SYSTEM_SETTINGS_PATH is set', () => {
|
|
1202
|
-
const MOCK_ENV_SYSTEM_SETTINGS_PATH = '/mock/env/system/settings.json';
|
|
1203
|
-
|
|
1204
|
-
beforeEach(() => {
|
|
1205
|
-
process.env['FSS_LINK_CLI_SYSTEM_SETTINGS_PATH'] =
|
|
1206
|
-
MOCK_ENV_SYSTEM_SETTINGS_PATH;
|
|
1207
|
-
});
|
|
1208
|
-
|
|
1209
|
-
afterEach(() => {
|
|
1210
|
-
delete process.env['FSS_LINK_CLI_SYSTEM_SETTINGS_PATH'];
|
|
1211
|
-
});
|
|
1212
|
-
|
|
1213
|
-
it('should load system settings from the path specified in the environment variable', () => {
|
|
1214
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1215
|
-
(p: fs.PathLike) => p === MOCK_ENV_SYSTEM_SETTINGS_PATH,
|
|
1216
|
-
);
|
|
1217
|
-
const systemSettingsContent = {
|
|
1218
|
-
theme: 'env-var-theme',
|
|
1219
|
-
sandbox: true,
|
|
1220
|
-
};
|
|
1221
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1222
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1223
|
-
if (p === MOCK_ENV_SYSTEM_SETTINGS_PATH)
|
|
1224
|
-
return JSON.stringify(systemSettingsContent);
|
|
1225
|
-
return '{}';
|
|
1226
|
-
},
|
|
1227
|
-
);
|
|
1228
|
-
|
|
1229
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1230
|
-
|
|
1231
|
-
expect(fs.readFileSync).toHaveBeenCalledWith(
|
|
1232
|
-
MOCK_ENV_SYSTEM_SETTINGS_PATH,
|
|
1233
|
-
'utf-8',
|
|
1234
|
-
);
|
|
1235
|
-
expect(settings.system.path).toBe(MOCK_ENV_SYSTEM_SETTINGS_PATH);
|
|
1236
|
-
expect(settings.system.settings).toEqual(systemSettingsContent);
|
|
1237
|
-
expect(settings.merged).toEqual({
|
|
1238
|
-
...systemSettingsContent,
|
|
1239
|
-
customThemes: {},
|
|
1240
|
-
mcpServers: {},
|
|
1241
|
-
includeDirectories: [],
|
|
1242
|
-
chatCompression: {},
|
|
1243
|
-
});
|
|
1244
|
-
});
|
|
1245
|
-
});
|
|
1246
|
-
});
|
|
1247
|
-
|
|
1248
|
-
describe('LoadedSettings class', () => {
|
|
1249
|
-
it('setValue should update the correct scope and recompute merged settings', () => {
|
|
1250
|
-
(mockFsExistsSync as Mock).mockReturnValue(false);
|
|
1251
|
-
const loadedSettings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1252
|
-
|
|
1253
|
-
vi.mocked(fs.writeFileSync).mockImplementation(() => {});
|
|
1254
|
-
// mkdirSync is mocked in beforeEach to return undefined, which is fine for void usage
|
|
1255
|
-
|
|
1256
|
-
loadedSettings.setValue(SettingScope.User, 'theme', 'matrix');
|
|
1257
|
-
expect(loadedSettings.user.settings.theme).toBe('matrix');
|
|
1258
|
-
expect(loadedSettings.merged.theme).toBe('matrix');
|
|
1259
|
-
expect(fs.writeFileSync).toHaveBeenCalledWith(
|
|
1260
|
-
USER_SETTINGS_PATH,
|
|
1261
|
-
JSON.stringify({ theme: 'matrix' }, null, 2),
|
|
1262
|
-
'utf-8',
|
|
1263
|
-
);
|
|
1264
|
-
|
|
1265
|
-
loadedSettings.setValue(
|
|
1266
|
-
SettingScope.Workspace,
|
|
1267
|
-
'contextFileName',
|
|
1268
|
-
'MY_AGENTS.md',
|
|
1269
|
-
);
|
|
1270
|
-
expect(loadedSettings.workspace.settings.contextFileName).toBe(
|
|
1271
|
-
'MY_AGENTS.md',
|
|
1272
|
-
);
|
|
1273
|
-
expect(loadedSettings.merged.contextFileName).toBe('MY_AGENTS.md');
|
|
1274
|
-
expect(loadedSettings.merged.theme).toBe('matrix'); // User setting should still be there
|
|
1275
|
-
expect(fs.writeFileSync).toHaveBeenCalledWith(
|
|
1276
|
-
MOCK_WORKSPACE_SETTINGS_PATH,
|
|
1277
|
-
JSON.stringify({ contextFileName: 'MY_AGENTS.md' }, null, 2),
|
|
1278
|
-
'utf-8',
|
|
1279
|
-
);
|
|
1280
|
-
|
|
1281
|
-
// System theme overrides user and workspace themes
|
|
1282
|
-
loadedSettings.setValue(SettingScope.System, 'theme', 'ocean');
|
|
1283
|
-
|
|
1284
|
-
expect(loadedSettings.system.settings.theme).toBe('ocean');
|
|
1285
|
-
expect(loadedSettings.merged.theme).toBe('ocean');
|
|
1286
|
-
});
|
|
1287
|
-
});
|
|
1288
|
-
|
|
1289
|
-
describe('excludedProjectEnvVars integration', () => {
|
|
1290
|
-
const originalEnv = { ...process.env };
|
|
1291
|
-
|
|
1292
|
-
beforeEach(() => {
|
|
1293
|
-
process.env = { ...originalEnv };
|
|
1294
|
-
});
|
|
1295
|
-
|
|
1296
|
-
afterEach(() => {
|
|
1297
|
-
process.env = originalEnv;
|
|
1298
|
-
});
|
|
1299
|
-
|
|
1300
|
-
it('should exclude DEBUG and DEBUG_MODE from project .env files by default', () => {
|
|
1301
|
-
// Create a workspace settings file with excludedProjectEnvVars
|
|
1302
|
-
const workspaceSettingsContent = {
|
|
1303
|
-
excludedProjectEnvVars: ['DEBUG', 'DEBUG_MODE'],
|
|
1304
|
-
};
|
|
1305
|
-
|
|
1306
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1307
|
-
(p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
|
|
1308
|
-
);
|
|
1309
|
-
|
|
1310
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1311
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1312
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
1313
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
1314
|
-
return '{}';
|
|
1315
|
-
},
|
|
1316
|
-
);
|
|
1317
|
-
|
|
1318
|
-
// Mock findEnvFile to return a project .env file
|
|
1319
|
-
const originalFindEnvFile = (
|
|
1320
|
-
loadSettings as unknown as { findEnvFile: () => string }
|
|
1321
|
-
).findEnvFile;
|
|
1322
|
-
(loadSettings as unknown as { findEnvFile: () => string }).findEnvFile =
|
|
1323
|
-
() => '/mock/project/.env';
|
|
1324
|
-
|
|
1325
|
-
// Mock fs.readFileSync for .env file content
|
|
1326
|
-
const originalReadFileSync = fs.readFileSync;
|
|
1327
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1328
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1329
|
-
if (p === '/mock/project/.env') {
|
|
1330
|
-
return 'DEBUG=true\nDEBUG_MODE=1\nGEMINI_API_KEY=test-key';
|
|
1331
|
-
}
|
|
1332
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH) {
|
|
1333
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
1334
|
-
}
|
|
1335
|
-
return '{}';
|
|
1336
|
-
},
|
|
1337
|
-
);
|
|
1338
|
-
|
|
1339
|
-
try {
|
|
1340
|
-
// This will call loadEnvironment internally with the merged settings
|
|
1341
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1342
|
-
|
|
1343
|
-
// Verify the settings were loaded correctly
|
|
1344
|
-
expect(settings.merged.excludedProjectEnvVars).toEqual([
|
|
1345
|
-
'DEBUG',
|
|
1346
|
-
'DEBUG_MODE',
|
|
1347
|
-
]);
|
|
1348
|
-
|
|
1349
|
-
// Note: We can't directly test process.env changes here because the mocking
|
|
1350
|
-
// prevents the actual file system operations, but we can verify the settings
|
|
1351
|
-
// are correctly merged and passed to loadEnvironment
|
|
1352
|
-
} finally {
|
|
1353
|
-
(loadSettings as unknown as { findEnvFile: () => string }).findEnvFile =
|
|
1354
|
-
originalFindEnvFile;
|
|
1355
|
-
(fs.readFileSync as Mock).mockImplementation(originalReadFileSync);
|
|
1356
|
-
}
|
|
1357
|
-
});
|
|
1358
|
-
|
|
1359
|
-
it('should respect custom excludedProjectEnvVars from user settings', () => {
|
|
1360
|
-
const userSettingsContent = {
|
|
1361
|
-
excludedProjectEnvVars: ['NODE_ENV', 'DEBUG'],
|
|
1362
|
-
};
|
|
1363
|
-
|
|
1364
|
-
(mockFsExistsSync as Mock).mockImplementation(
|
|
1365
|
-
(p: fs.PathLike) => p === USER_SETTINGS_PATH,
|
|
1366
|
-
);
|
|
1367
|
-
|
|
1368
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1369
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1370
|
-
if (p === USER_SETTINGS_PATH)
|
|
1371
|
-
return JSON.stringify(userSettingsContent);
|
|
1372
|
-
return '{}';
|
|
1373
|
-
},
|
|
1374
|
-
);
|
|
1375
|
-
|
|
1376
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1377
|
-
expect(settings.user.settings.excludedProjectEnvVars).toEqual([
|
|
1378
|
-
'NODE_ENV',
|
|
1379
|
-
'DEBUG',
|
|
1380
|
-
]);
|
|
1381
|
-
expect(settings.merged.excludedProjectEnvVars).toEqual([
|
|
1382
|
-
'NODE_ENV',
|
|
1383
|
-
'DEBUG',
|
|
1384
|
-
]);
|
|
1385
|
-
});
|
|
1386
|
-
|
|
1387
|
-
it('should merge excludedProjectEnvVars with workspace taking precedence', () => {
|
|
1388
|
-
const userSettingsContent = {
|
|
1389
|
-
excludedProjectEnvVars: ['DEBUG', 'NODE_ENV', 'USER_VAR'],
|
|
1390
|
-
};
|
|
1391
|
-
const workspaceSettingsContent = {
|
|
1392
|
-
excludedProjectEnvVars: ['WORKSPACE_DEBUG', 'WORKSPACE_VAR'],
|
|
1393
|
-
};
|
|
1394
|
-
|
|
1395
|
-
(mockFsExistsSync as Mock).mockReturnValue(true);
|
|
1396
|
-
|
|
1397
|
-
(fs.readFileSync as Mock).mockImplementation(
|
|
1398
|
-
(p: fs.PathOrFileDescriptor) => {
|
|
1399
|
-
if (p === USER_SETTINGS_PATH)
|
|
1400
|
-
return JSON.stringify(userSettingsContent);
|
|
1401
|
-
if (p === MOCK_WORKSPACE_SETTINGS_PATH)
|
|
1402
|
-
return JSON.stringify(workspaceSettingsContent);
|
|
1403
|
-
return '{}';
|
|
1404
|
-
},
|
|
1405
|
-
);
|
|
1406
|
-
|
|
1407
|
-
const settings = loadSettings(MOCK_WORKSPACE_DIR);
|
|
1408
|
-
|
|
1409
|
-
expect(settings.user.settings.excludedProjectEnvVars).toEqual([
|
|
1410
|
-
'DEBUG',
|
|
1411
|
-
'NODE_ENV',
|
|
1412
|
-
'USER_VAR',
|
|
1413
|
-
]);
|
|
1414
|
-
expect(settings.workspace.settings.excludedProjectEnvVars).toEqual([
|
|
1415
|
-
'WORKSPACE_DEBUG',
|
|
1416
|
-
'WORKSPACE_VAR',
|
|
1417
|
-
]);
|
|
1418
|
-
expect(settings.merged.excludedProjectEnvVars).toEqual([
|
|
1419
|
-
'WORKSPACE_DEBUG',
|
|
1420
|
-
'WORKSPACE_VAR',
|
|
1421
|
-
]);
|
|
1422
|
-
});
|
|
1423
|
-
});
|
|
1424
|
-
});
|