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,289 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { promises as fs } from 'fs';
|
|
8
|
-
import path from 'path';
|
|
9
|
-
import toml from '@iarna/toml';
|
|
10
|
-
import { glob } from 'glob';
|
|
11
|
-
import { z } from 'zod';
|
|
12
|
-
import {
|
|
13
|
-
Config,
|
|
14
|
-
getProjectCommandsDir,
|
|
15
|
-
getUserCommandsDir,
|
|
16
|
-
} from 'fss-link-core';
|
|
17
|
-
import { ICommandLoader } from './types.js';
|
|
18
|
-
import {
|
|
19
|
-
CommandContext,
|
|
20
|
-
CommandKind,
|
|
21
|
-
SlashCommand,
|
|
22
|
-
SlashCommandActionReturn,
|
|
23
|
-
} from '../ui/commands/types.js';
|
|
24
|
-
import { DefaultArgumentProcessor } from './prompt-processors/argumentProcessor.js';
|
|
25
|
-
import {
|
|
26
|
-
IPromptProcessor,
|
|
27
|
-
SHORTHAND_ARGS_PLACEHOLDER,
|
|
28
|
-
SHELL_INJECTION_TRIGGER,
|
|
29
|
-
} from './prompt-processors/types.js';
|
|
30
|
-
import {
|
|
31
|
-
ConfirmationRequiredError,
|
|
32
|
-
ShellProcessor,
|
|
33
|
-
} from './prompt-processors/shellProcessor.js';
|
|
34
|
-
|
|
35
|
-
interface CommandDirectory {
|
|
36
|
-
path: string;
|
|
37
|
-
extensionName?: string;
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
/**
|
|
41
|
-
* Defines the Zod schema for a command definition file. This serves as the
|
|
42
|
-
* single source of truth for both validation and type inference.
|
|
43
|
-
*/
|
|
44
|
-
const TomlCommandDefSchema = z.object({
|
|
45
|
-
prompt: z.string({
|
|
46
|
-
required_error: "The 'prompt' field is required.",
|
|
47
|
-
invalid_type_error: "The 'prompt' field must be a string.",
|
|
48
|
-
}),
|
|
49
|
-
description: z.string().optional(),
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
/**
|
|
53
|
-
* Discovers and loads custom slash commands from .toml files in both the
|
|
54
|
-
* user's global config directory and the current project's directory.
|
|
55
|
-
*
|
|
56
|
-
* This loader is responsible for:
|
|
57
|
-
* - Recursively scanning command directories.
|
|
58
|
-
* - Parsing and validating TOML files.
|
|
59
|
-
* - Adapting valid definitions into executable SlashCommand objects.
|
|
60
|
-
* - Handling file system errors and malformed files gracefully.
|
|
61
|
-
*/
|
|
62
|
-
export class FileCommandLoader implements ICommandLoader {
|
|
63
|
-
private readonly projectRoot: string;
|
|
64
|
-
|
|
65
|
-
constructor(private readonly config: Config | null) {
|
|
66
|
-
this.projectRoot = config?.getProjectRoot() || process.cwd();
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
/**
|
|
70
|
-
* Loads all commands from user, project, and extension directories.
|
|
71
|
-
* Returns commands in order: user → project → extensions (alphabetically).
|
|
72
|
-
*
|
|
73
|
-
* Order is important for conflict resolution in CommandService:
|
|
74
|
-
* - User/project commands (without extensionName) use "last wins" strategy
|
|
75
|
-
* - Extension commands (with extensionName) get renamed if conflicts exist
|
|
76
|
-
*
|
|
77
|
-
* @param signal An AbortSignal to cancel the loading process.
|
|
78
|
-
* @returns A promise that resolves to an array of all loaded SlashCommands.
|
|
79
|
-
*/
|
|
80
|
-
async loadCommands(signal: AbortSignal): Promise<SlashCommand[]> {
|
|
81
|
-
const allCommands: SlashCommand[] = [];
|
|
82
|
-
const globOptions = {
|
|
83
|
-
nodir: true,
|
|
84
|
-
dot: true,
|
|
85
|
-
signal,
|
|
86
|
-
follow: true,
|
|
87
|
-
};
|
|
88
|
-
|
|
89
|
-
// Load commands from each directory
|
|
90
|
-
const commandDirs = this.getCommandDirectories();
|
|
91
|
-
for (const dirInfo of commandDirs) {
|
|
92
|
-
try {
|
|
93
|
-
const files = await glob('**/*.toml', {
|
|
94
|
-
...globOptions,
|
|
95
|
-
cwd: dirInfo.path,
|
|
96
|
-
});
|
|
97
|
-
|
|
98
|
-
const commandPromises = files.map((file) =>
|
|
99
|
-
this.parseAndAdaptFile(
|
|
100
|
-
path.join(dirInfo.path, file),
|
|
101
|
-
dirInfo.path,
|
|
102
|
-
dirInfo.extensionName,
|
|
103
|
-
),
|
|
104
|
-
);
|
|
105
|
-
|
|
106
|
-
const commands = (await Promise.all(commandPromises)).filter(
|
|
107
|
-
(cmd): cmd is SlashCommand => cmd !== null,
|
|
108
|
-
);
|
|
109
|
-
|
|
110
|
-
// Add all commands without deduplication
|
|
111
|
-
allCommands.push(...commands);
|
|
112
|
-
} catch (error) {
|
|
113
|
-
if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {
|
|
114
|
-
console.error(
|
|
115
|
-
`[FileCommandLoader] Error loading commands from ${dirInfo.path}:`,
|
|
116
|
-
error,
|
|
117
|
-
);
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
return allCommands;
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
/**
|
|
126
|
-
* Get all command directories in order for loading.
|
|
127
|
-
* User commands → Project commands → Extension commands
|
|
128
|
-
* This order ensures extension commands can detect all conflicts.
|
|
129
|
-
*/
|
|
130
|
-
private getCommandDirectories(): CommandDirectory[] {
|
|
131
|
-
const dirs: CommandDirectory[] = [];
|
|
132
|
-
|
|
133
|
-
// 1. User commands
|
|
134
|
-
dirs.push({ path: getUserCommandsDir() });
|
|
135
|
-
|
|
136
|
-
// 2. Project commands (override user commands)
|
|
137
|
-
dirs.push({ path: getProjectCommandsDir(this.projectRoot) });
|
|
138
|
-
|
|
139
|
-
// 3. Extension commands (processed last to detect all conflicts)
|
|
140
|
-
if (this.config) {
|
|
141
|
-
const activeExtensions = this.config
|
|
142
|
-
.getExtensions()
|
|
143
|
-
.filter((ext) => ext.isActive)
|
|
144
|
-
.sort((a, b) => a.name.localeCompare(b.name)); // Sort alphabetically for deterministic loading
|
|
145
|
-
|
|
146
|
-
const extensionCommandDirs = activeExtensions.map((ext) => ({
|
|
147
|
-
path: path.join(ext.path, 'commands'),
|
|
148
|
-
extensionName: ext.name,
|
|
149
|
-
}));
|
|
150
|
-
|
|
151
|
-
dirs.push(...extensionCommandDirs);
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
return dirs;
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
/**
|
|
158
|
-
* Parses a single .toml file and transforms it into a SlashCommand object.
|
|
159
|
-
* @param filePath The absolute path to the .toml file.
|
|
160
|
-
* @param baseDir The root command directory for name calculation.
|
|
161
|
-
* @param extensionName Optional extension name to prefix commands with.
|
|
162
|
-
* @returns A promise resolving to a SlashCommand, or null if the file is invalid.
|
|
163
|
-
*/
|
|
164
|
-
private async parseAndAdaptFile(
|
|
165
|
-
filePath: string,
|
|
166
|
-
baseDir: string,
|
|
167
|
-
extensionName?: string,
|
|
168
|
-
): Promise<SlashCommand | null> {
|
|
169
|
-
let fileContent: string;
|
|
170
|
-
try {
|
|
171
|
-
fileContent = await fs.readFile(filePath, 'utf-8');
|
|
172
|
-
} catch (error: unknown) {
|
|
173
|
-
console.error(
|
|
174
|
-
`[FileCommandLoader] Failed to read file ${filePath}:`,
|
|
175
|
-
error instanceof Error ? error.message : String(error),
|
|
176
|
-
);
|
|
177
|
-
return null;
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
let parsed: unknown;
|
|
181
|
-
try {
|
|
182
|
-
parsed = toml.parse(fileContent);
|
|
183
|
-
} catch (error: unknown) {
|
|
184
|
-
console.error(
|
|
185
|
-
`[FileCommandLoader] Failed to parse TOML file ${filePath}:`,
|
|
186
|
-
error instanceof Error ? error.message : String(error),
|
|
187
|
-
);
|
|
188
|
-
return null;
|
|
189
|
-
}
|
|
190
|
-
|
|
191
|
-
const validationResult = TomlCommandDefSchema.safeParse(parsed);
|
|
192
|
-
|
|
193
|
-
if (!validationResult.success) {
|
|
194
|
-
console.error(
|
|
195
|
-
`[FileCommandLoader] Skipping invalid command file: ${filePath}. Validation errors:`,
|
|
196
|
-
validationResult.error.flatten(),
|
|
197
|
-
);
|
|
198
|
-
return null;
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
const validDef = validationResult.data;
|
|
202
|
-
|
|
203
|
-
const relativePathWithExt = path.relative(baseDir, filePath);
|
|
204
|
-
const relativePath = relativePathWithExt.substring(
|
|
205
|
-
0,
|
|
206
|
-
relativePathWithExt.length - 5, // length of '.toml'
|
|
207
|
-
);
|
|
208
|
-
const baseCommandName = relativePath
|
|
209
|
-
.split(path.sep)
|
|
210
|
-
// Sanitize each path segment to prevent ambiguity. Since ':' is our
|
|
211
|
-
// namespace separator, we replace any literal colons in filenames
|
|
212
|
-
// with underscores to avoid naming conflicts.
|
|
213
|
-
.map((segment) => segment.replaceAll(':', '_'))
|
|
214
|
-
.join(':');
|
|
215
|
-
|
|
216
|
-
// Add extension name tag for extension commands
|
|
217
|
-
const defaultDescription = `Custom command from ${path.basename(filePath)}`;
|
|
218
|
-
let description = validDef.description || defaultDescription;
|
|
219
|
-
if (extensionName) {
|
|
220
|
-
description = `[${extensionName}] ${description}`;
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
const processors: IPromptProcessor[] = [];
|
|
224
|
-
const usesArgs = validDef.prompt.includes(SHORTHAND_ARGS_PLACEHOLDER);
|
|
225
|
-
const usesShellInjection = validDef.prompt.includes(
|
|
226
|
-
SHELL_INJECTION_TRIGGER,
|
|
227
|
-
);
|
|
228
|
-
|
|
229
|
-
// Interpolation (Shell Execution and Argument Injection)
|
|
230
|
-
// If the prompt uses either shell injection OR argument placeholders,
|
|
231
|
-
// we must use the ShellProcessor.
|
|
232
|
-
if (usesShellInjection || usesArgs) {
|
|
233
|
-
processors.push(new ShellProcessor(baseCommandName));
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
// Default Argument Handling
|
|
237
|
-
// If NO explicit argument injection ({{args}}) was used, we append the raw invocation.
|
|
238
|
-
if (!usesArgs) {
|
|
239
|
-
processors.push(new DefaultArgumentProcessor());
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
return {
|
|
243
|
-
name: baseCommandName,
|
|
244
|
-
description,
|
|
245
|
-
kind: CommandKind.FILE,
|
|
246
|
-
extensionName,
|
|
247
|
-
action: async (
|
|
248
|
-
context: CommandContext,
|
|
249
|
-
_args: string,
|
|
250
|
-
): Promise<SlashCommandActionReturn> => {
|
|
251
|
-
if (!context.invocation) {
|
|
252
|
-
console.error(
|
|
253
|
-
`[FileCommandLoader] Critical error: Command '${baseCommandName}' was executed without invocation context.`,
|
|
254
|
-
);
|
|
255
|
-
return {
|
|
256
|
-
type: 'submit_prompt',
|
|
257
|
-
content: validDef.prompt, // Fallback to unprocessed prompt
|
|
258
|
-
};
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
try {
|
|
262
|
-
let processedPrompt = validDef.prompt;
|
|
263
|
-
for (const processor of processors) {
|
|
264
|
-
processedPrompt = await processor.process(processedPrompt, context);
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
return {
|
|
268
|
-
type: 'submit_prompt',
|
|
269
|
-
content: processedPrompt,
|
|
270
|
-
};
|
|
271
|
-
} catch (e) {
|
|
272
|
-
// Check if it's our specific error type
|
|
273
|
-
if (e instanceof ConfirmationRequiredError) {
|
|
274
|
-
// Halt and request confirmation from the UI layer.
|
|
275
|
-
return {
|
|
276
|
-
type: 'confirm_shell_commands',
|
|
277
|
-
commandsToConfirm: e.commandsToConfirm,
|
|
278
|
-
originalInvocation: {
|
|
279
|
-
raw: context.invocation.raw,
|
|
280
|
-
},
|
|
281
|
-
};
|
|
282
|
-
}
|
|
283
|
-
// Re-throw other errors to be handled by the global error handler.
|
|
284
|
-
throw e;
|
|
285
|
-
}
|
|
286
|
-
},
|
|
287
|
-
};
|
|
288
|
-
}
|
|
289
|
-
}
|
|
@@ -1,231 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import {
|
|
8
|
-
Config,
|
|
9
|
-
getErrorMessage,
|
|
10
|
-
getMCPServerPrompts,
|
|
11
|
-
} from 'fss-link-core';
|
|
12
|
-
import {
|
|
13
|
-
CommandContext,
|
|
14
|
-
CommandKind,
|
|
15
|
-
SlashCommand,
|
|
16
|
-
SlashCommandActionReturn,
|
|
17
|
-
} from '../ui/commands/types.js';
|
|
18
|
-
import { ICommandLoader } from './types.js';
|
|
19
|
-
import { PromptArgument } from '@modelcontextprotocol/sdk/types.js';
|
|
20
|
-
|
|
21
|
-
/**
|
|
22
|
-
* Discovers and loads executable slash commands from prompts exposed by
|
|
23
|
-
* Model-Context-Protocol (MCP) servers.
|
|
24
|
-
*/
|
|
25
|
-
export class McpPromptLoader implements ICommandLoader {
|
|
26
|
-
constructor(private readonly config: Config | null) {}
|
|
27
|
-
|
|
28
|
-
/**
|
|
29
|
-
* Loads all available prompts from all configured MCP servers and adapts
|
|
30
|
-
* them into executable SlashCommand objects.
|
|
31
|
-
*
|
|
32
|
-
* @param _signal An AbortSignal (unused for this synchronous loader).
|
|
33
|
-
* @returns A promise that resolves to an array of loaded SlashCommands.
|
|
34
|
-
*/
|
|
35
|
-
loadCommands(_signal: AbortSignal): Promise<SlashCommand[]> {
|
|
36
|
-
const promptCommands: SlashCommand[] = [];
|
|
37
|
-
if (!this.config) {
|
|
38
|
-
return Promise.resolve([]);
|
|
39
|
-
}
|
|
40
|
-
const mcpServers = this.config.getMcpServers() || {};
|
|
41
|
-
for (const serverName in mcpServers) {
|
|
42
|
-
const prompts = getMCPServerPrompts(this.config, serverName) || [];
|
|
43
|
-
for (const prompt of prompts) {
|
|
44
|
-
const commandName = `${prompt.name}`;
|
|
45
|
-
const newPromptCommand: SlashCommand = {
|
|
46
|
-
name: commandName,
|
|
47
|
-
description: prompt.description || `Invoke prompt ${prompt.name}`,
|
|
48
|
-
kind: CommandKind.MCP_PROMPT,
|
|
49
|
-
subCommands: [
|
|
50
|
-
{
|
|
51
|
-
name: 'help',
|
|
52
|
-
description: 'Show help for this prompt',
|
|
53
|
-
kind: CommandKind.MCP_PROMPT,
|
|
54
|
-
action: async (): Promise<SlashCommandActionReturn> => {
|
|
55
|
-
if (!prompt.arguments || prompt.arguments.length === 0) {
|
|
56
|
-
return {
|
|
57
|
-
type: 'message',
|
|
58
|
-
messageType: 'info',
|
|
59
|
-
content: `Prompt "${prompt.name}" has no arguments.`,
|
|
60
|
-
};
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
let helpMessage = `Arguments for "${prompt.name}":\n\n`;
|
|
64
|
-
if (prompt.arguments && prompt.arguments.length > 0) {
|
|
65
|
-
helpMessage += `You can provide arguments by name (e.g., --argName="value") or by position.\n\n`;
|
|
66
|
-
helpMessage += `e.g., ${prompt.name} ${prompt.arguments?.map((_) => `"foo"`)} is equivalent to ${prompt.name} ${prompt.arguments?.map((arg) => `--${arg.name}="foo"`)}\n\n`;
|
|
67
|
-
}
|
|
68
|
-
for (const arg of prompt.arguments) {
|
|
69
|
-
helpMessage += ` --${arg.name}\n`;
|
|
70
|
-
if (arg.description) {
|
|
71
|
-
helpMessage += ` ${arg.description}\n`;
|
|
72
|
-
}
|
|
73
|
-
helpMessage += ` (required: ${
|
|
74
|
-
arg.required ? 'yes' : 'no'
|
|
75
|
-
})\n\n`;
|
|
76
|
-
}
|
|
77
|
-
return {
|
|
78
|
-
type: 'message',
|
|
79
|
-
messageType: 'info',
|
|
80
|
-
content: helpMessage,
|
|
81
|
-
};
|
|
82
|
-
},
|
|
83
|
-
},
|
|
84
|
-
],
|
|
85
|
-
action: async (
|
|
86
|
-
context: CommandContext,
|
|
87
|
-
args: string,
|
|
88
|
-
): Promise<SlashCommandActionReturn> => {
|
|
89
|
-
if (!this.config) {
|
|
90
|
-
return {
|
|
91
|
-
type: 'message',
|
|
92
|
-
messageType: 'error',
|
|
93
|
-
content: 'Config not loaded.',
|
|
94
|
-
};
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
const promptInputs = this.parseArgs(args, prompt.arguments);
|
|
98
|
-
if (promptInputs instanceof Error) {
|
|
99
|
-
return {
|
|
100
|
-
type: 'message',
|
|
101
|
-
messageType: 'error',
|
|
102
|
-
content: promptInputs.message,
|
|
103
|
-
};
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
try {
|
|
107
|
-
const mcpServers = this.config.getMcpServers() || {};
|
|
108
|
-
const mcpServerConfig = mcpServers[serverName];
|
|
109
|
-
if (!mcpServerConfig) {
|
|
110
|
-
return {
|
|
111
|
-
type: 'message',
|
|
112
|
-
messageType: 'error',
|
|
113
|
-
content: `MCP server config not found for '${serverName}'.`,
|
|
114
|
-
};
|
|
115
|
-
}
|
|
116
|
-
const result = await prompt.invoke(promptInputs);
|
|
117
|
-
|
|
118
|
-
if (result['error']) {
|
|
119
|
-
return {
|
|
120
|
-
type: 'message',
|
|
121
|
-
messageType: 'error',
|
|
122
|
-
content: `Error invoking prompt: ${result['error']}`,
|
|
123
|
-
};
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
if (!result.messages?.[0]?.content?.['text']) {
|
|
127
|
-
return {
|
|
128
|
-
type: 'message',
|
|
129
|
-
messageType: 'error',
|
|
130
|
-
content:
|
|
131
|
-
'Received an empty or invalid prompt response from the server.',
|
|
132
|
-
};
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
return {
|
|
136
|
-
type: 'submit_prompt',
|
|
137
|
-
content: JSON.stringify(result.messages[0].content.text),
|
|
138
|
-
};
|
|
139
|
-
} catch (error) {
|
|
140
|
-
return {
|
|
141
|
-
type: 'message',
|
|
142
|
-
messageType: 'error',
|
|
143
|
-
content: `Error: ${getErrorMessage(error)}`,
|
|
144
|
-
};
|
|
145
|
-
}
|
|
146
|
-
},
|
|
147
|
-
completion: async (_: CommandContext, partialArg: string) => {
|
|
148
|
-
if (!prompt || !prompt.arguments) {
|
|
149
|
-
return [];
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
const suggestions: string[] = [];
|
|
153
|
-
const usedArgNames = new Set(
|
|
154
|
-
(partialArg.match(/--([^=]+)/g) || []).map((s) => s.substring(2)),
|
|
155
|
-
);
|
|
156
|
-
|
|
157
|
-
for (const arg of prompt.arguments) {
|
|
158
|
-
if (!usedArgNames.has(arg.name)) {
|
|
159
|
-
suggestions.push(`--${arg.name}=""`);
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
return suggestions;
|
|
164
|
-
},
|
|
165
|
-
};
|
|
166
|
-
promptCommands.push(newPromptCommand);
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
return Promise.resolve(promptCommands);
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
private parseArgs(
|
|
173
|
-
userArgs: string,
|
|
174
|
-
promptArgs: PromptArgument[] | undefined,
|
|
175
|
-
): Record<string, unknown> | Error {
|
|
176
|
-
const argValues: { [key: string]: string } = {};
|
|
177
|
-
const promptInputs: Record<string, unknown> = {};
|
|
178
|
-
|
|
179
|
-
// arg parsing: --key="value" or --key=value
|
|
180
|
-
const namedArgRegex = /--([^=]+)=(?:"((?:\\.|[^"\\])*)"|([^ ]*))/g;
|
|
181
|
-
let match;
|
|
182
|
-
const remainingArgs: string[] = [];
|
|
183
|
-
let lastIndex = 0;
|
|
184
|
-
|
|
185
|
-
while ((match = namedArgRegex.exec(userArgs)) !== null) {
|
|
186
|
-
const key = match[1];
|
|
187
|
-
const value = match[2] ?? match[3]; // Quoted or unquoted value
|
|
188
|
-
argValues[key] = value;
|
|
189
|
-
// Capture text between matches as potential positional args
|
|
190
|
-
if (match.index > lastIndex) {
|
|
191
|
-
remainingArgs.push(userArgs.substring(lastIndex, match.index).trim());
|
|
192
|
-
}
|
|
193
|
-
lastIndex = namedArgRegex.lastIndex;
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
// Capture any remaining text after the last named arg
|
|
197
|
-
if (lastIndex < userArgs.length) {
|
|
198
|
-
remainingArgs.push(userArgs.substring(lastIndex).trim());
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
const positionalArgs = remainingArgs.join(' ').split(/ +/);
|
|
202
|
-
|
|
203
|
-
if (!promptArgs) {
|
|
204
|
-
return promptInputs;
|
|
205
|
-
}
|
|
206
|
-
for (const arg of promptArgs) {
|
|
207
|
-
if (argValues[arg.name]) {
|
|
208
|
-
promptInputs[arg.name] = argValues[arg.name];
|
|
209
|
-
}
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
const unfilledArgs = promptArgs.filter(
|
|
213
|
-
(arg) => arg.required && !promptInputs[arg.name],
|
|
214
|
-
);
|
|
215
|
-
|
|
216
|
-
const missingArgs: string[] = [];
|
|
217
|
-
for (let i = 0; i < unfilledArgs.length; i++) {
|
|
218
|
-
if (positionalArgs.length > i && positionalArgs[i]) {
|
|
219
|
-
promptInputs[unfilledArgs[i].name] = positionalArgs[i];
|
|
220
|
-
} else {
|
|
221
|
-
missingArgs.push(unfilledArgs[i].name);
|
|
222
|
-
}
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
if (missingArgs.length > 0) {
|
|
226
|
-
const missingArgNames = missingArgs.map((name) => `--${name}`).join(', ');
|
|
227
|
-
return new Error(`Missing required argument(s): ${missingArgNames}`);
|
|
228
|
-
}
|
|
229
|
-
return promptInputs;
|
|
230
|
-
}
|
|
231
|
-
}
|
|
@@ -1,100 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { getFSSLinkDatabase } from '../config/database.js';
|
|
8
|
-
|
|
9
|
-
export interface SearchEngineConfig {
|
|
10
|
-
braveApiKey?: string;
|
|
11
|
-
tavilyApiKey?: string;
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
/**
|
|
15
|
-
* Configuration provider for search engine API keys
|
|
16
|
-
* Loads from database and sets environment variables
|
|
17
|
-
*/
|
|
18
|
-
export class SearchEngineConfigProvider {
|
|
19
|
-
private static instance: SearchEngineConfigProvider | null = null;
|
|
20
|
-
|
|
21
|
-
static getInstance(): SearchEngineConfigProvider {
|
|
22
|
-
if (!SearchEngineConfigProvider.instance) {
|
|
23
|
-
SearchEngineConfigProvider.instance = new SearchEngineConfigProvider();
|
|
24
|
-
}
|
|
25
|
-
return SearchEngineConfigProvider.instance;
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
/**
|
|
29
|
-
* Load search engine configuration from database and set environment variables
|
|
30
|
-
*/
|
|
31
|
-
async loadConfiguration(): Promise<SearchEngineConfig> {
|
|
32
|
-
const db = await getFSSLinkDatabase();
|
|
33
|
-
|
|
34
|
-
const braveApiKey = await db.getUserPreference('webscraper.brave_api_key');
|
|
35
|
-
const tavilyApiKey = await db.getUserPreference('webscraper.tavily_api_key');
|
|
36
|
-
|
|
37
|
-
// Set environment variables for immediate use
|
|
38
|
-
if (braveApiKey) {
|
|
39
|
-
process.env['BRAVE_SEARCH_API_KEY'] = braveApiKey;
|
|
40
|
-
}
|
|
41
|
-
if (tavilyApiKey) {
|
|
42
|
-
process.env['TAVILY_API_KEY'] = tavilyApiKey;
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
return {
|
|
46
|
-
braveApiKey: braveApiKey || undefined,
|
|
47
|
-
tavilyApiKey: tavilyApiKey || undefined,
|
|
48
|
-
};
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* Save search engine configuration to database
|
|
53
|
-
*/
|
|
54
|
-
async saveConfiguration(config: SearchEngineConfig): Promise<void> {
|
|
55
|
-
const db = await getFSSLinkDatabase();
|
|
56
|
-
|
|
57
|
-
if (config.braveApiKey) {
|
|
58
|
-
await db.setUserPreference('webscraper.brave_api_key', config.braveApiKey);
|
|
59
|
-
await db.setUserPreference('webscraper.brave_enabled', 'true');
|
|
60
|
-
process.env['BRAVE_SEARCH_API_KEY'] = config.braveApiKey;
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
if (config.tavilyApiKey) {
|
|
64
|
-
await db.setUserPreference('webscraper.tavily_api_key', config.tavilyApiKey);
|
|
65
|
-
await db.setUserPreference('webscraper.tavily_enabled', 'true');
|
|
66
|
-
process.env['TAVILY_API_KEY'] = config.tavilyApiKey;
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
/**
|
|
71
|
-
* Check if search engines are configured
|
|
72
|
-
*/
|
|
73
|
-
async hasConfiguration(): Promise<boolean> {
|
|
74
|
-
const config = await this.loadConfiguration();
|
|
75
|
-
return !!(config.braveApiKey || config.tavilyApiKey);
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
/**
|
|
79
|
-
* Get configuration status for display
|
|
80
|
-
*/
|
|
81
|
-
async getConfigurationStatus(): Promise<string> {
|
|
82
|
-
const config = await this.loadConfiguration();
|
|
83
|
-
|
|
84
|
-
let status = '🔍 Search Engine Configuration:\n';
|
|
85
|
-
|
|
86
|
-
if (config.braveApiKey) {
|
|
87
|
-
status += `✅ Brave Search: Configured\n`;
|
|
88
|
-
} else {
|
|
89
|
-
status += `❌ Brave Search: Not configured\n`;
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
if (config.tavilyApiKey) {
|
|
93
|
-
status += `✅ Tavily Search: Configured\n`;
|
|
94
|
-
} else {
|
|
95
|
-
status += `❌ Tavily Search: Not configured\n`;
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
return status;
|
|
99
|
-
}
|
|
100
|
-
}
|
|
@@ -1,41 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { DefaultArgumentProcessor } from './argumentProcessor.js';
|
|
8
|
-
import { createMockCommandContext } from '../../test-utils/mockCommandContext.js';
|
|
9
|
-
import { describe, it, expect } from 'vitest';
|
|
10
|
-
|
|
11
|
-
describe('Argument Processors', () => {
|
|
12
|
-
describe('DefaultArgumentProcessor', () => {
|
|
13
|
-
const processor = new DefaultArgumentProcessor();
|
|
14
|
-
|
|
15
|
-
it('should append the full command if args are provided', async () => {
|
|
16
|
-
const prompt = 'Parse the command.';
|
|
17
|
-
const context = createMockCommandContext({
|
|
18
|
-
invocation: {
|
|
19
|
-
raw: '/mycommand arg1 "arg two"',
|
|
20
|
-
name: 'mycommand',
|
|
21
|
-
args: 'arg1 "arg two"',
|
|
22
|
-
},
|
|
23
|
-
});
|
|
24
|
-
const result = await processor.process(prompt, context);
|
|
25
|
-
expect(result).toBe('Parse the command.\n\n/mycommand arg1 "arg two"');
|
|
26
|
-
});
|
|
27
|
-
|
|
28
|
-
it('should NOT append the full command if no args are provided', async () => {
|
|
29
|
-
const prompt = 'Parse the command.';
|
|
30
|
-
const context = createMockCommandContext({
|
|
31
|
-
invocation: {
|
|
32
|
-
raw: '/mycommand',
|
|
33
|
-
name: 'mycommand',
|
|
34
|
-
args: '',
|
|
35
|
-
},
|
|
36
|
-
});
|
|
37
|
-
const result = await processor.process(prompt, context);
|
|
38
|
-
expect(result).toBe('Parse the command.');
|
|
39
|
-
});
|
|
40
|
-
});
|
|
41
|
-
});
|
|
@@ -1,23 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { IPromptProcessor } from './types.js';
|
|
8
|
-
import { CommandContext } from '../../ui/commands/types.js';
|
|
9
|
-
|
|
10
|
-
/**
|
|
11
|
-
* Appends the user's full command invocation to the prompt if arguments are
|
|
12
|
-
* provided, allowing the model to perform its own argument parsing.
|
|
13
|
-
*
|
|
14
|
-
* This processor is only used if the prompt does NOT contain {{args}}.
|
|
15
|
-
*/
|
|
16
|
-
export class DefaultArgumentProcessor implements IPromptProcessor {
|
|
17
|
-
async process(prompt: string, context: CommandContext): Promise<string> {
|
|
18
|
-
if (context.invocation!.args) {
|
|
19
|
-
return `${prompt}\n\n${context.invocation!.raw}`;
|
|
20
|
-
}
|
|
21
|
-
return prompt;
|
|
22
|
-
}
|
|
23
|
-
}
|