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,31 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { useState } from 'react';
|
|
8
|
-
import {
|
|
9
|
-
isKittyProtocolEnabled,
|
|
10
|
-
isKittyProtocolSupported,
|
|
11
|
-
} from '../utils/kittyProtocolDetector.js';
|
|
12
|
-
|
|
13
|
-
export interface KittyProtocolStatus {
|
|
14
|
-
supported: boolean;
|
|
15
|
-
enabled: boolean;
|
|
16
|
-
checking: boolean;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
/**
|
|
20
|
-
* Hook that returns the cached Kitty keyboard protocol status.
|
|
21
|
-
* Detection is done once at app startup to avoid repeated queries.
|
|
22
|
-
*/
|
|
23
|
-
export function useKittyKeyboardProtocol(): KittyProtocolStatus {
|
|
24
|
-
const [status] = useState<KittyProtocolStatus>({
|
|
25
|
-
supported: isKittyProtocolSupported(),
|
|
26
|
-
enabled: isKittyProtocolEnabled(),
|
|
27
|
-
checking: false,
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
return status;
|
|
31
|
-
}
|
|
@@ -1,139 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
8
|
-
import { renderHook, act } from '@testing-library/react';
|
|
9
|
-
import { useLoadingIndicator } from './useLoadingIndicator.js';
|
|
10
|
-
import { StreamingState } from '../types.js';
|
|
11
|
-
import {
|
|
12
|
-
WITTY_LOADING_PHRASES,
|
|
13
|
-
PHRASE_CHANGE_INTERVAL_MS,
|
|
14
|
-
} from './usePhraseCycler.js';
|
|
15
|
-
|
|
16
|
-
describe('useLoadingIndicator', () => {
|
|
17
|
-
beforeEach(() => {
|
|
18
|
-
vi.useFakeTimers();
|
|
19
|
-
});
|
|
20
|
-
|
|
21
|
-
afterEach(() => {
|
|
22
|
-
vi.useRealTimers(); // Restore real timers after each test
|
|
23
|
-
act(() => vi.runOnlyPendingTimers);
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
it('should initialize with default values when Idle', () => {
|
|
27
|
-
const { result } = renderHook(() =>
|
|
28
|
-
useLoadingIndicator(StreamingState.Idle),
|
|
29
|
-
);
|
|
30
|
-
expect(result.current.elapsedTime).toBe(0);
|
|
31
|
-
expect(result.current.currentLoadingPhrase).toBe(WITTY_LOADING_PHRASES[0]);
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
it('should reflect values when Responding', async () => {
|
|
35
|
-
const { result } = renderHook(() =>
|
|
36
|
-
useLoadingIndicator(StreamingState.Responding),
|
|
37
|
-
);
|
|
38
|
-
|
|
39
|
-
// Initial state before timers advance
|
|
40
|
-
expect(result.current.elapsedTime).toBe(0);
|
|
41
|
-
expect(WITTY_LOADING_PHRASES).toContain(
|
|
42
|
-
result.current.currentLoadingPhrase,
|
|
43
|
-
);
|
|
44
|
-
|
|
45
|
-
await act(async () => {
|
|
46
|
-
await vi.advanceTimersByTimeAsync(PHRASE_CHANGE_INTERVAL_MS + 1);
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
// Phrase should cycle if PHRASE_CHANGE_INTERVAL_MS has passed
|
|
50
|
-
expect(WITTY_LOADING_PHRASES).toContain(
|
|
51
|
-
result.current.currentLoadingPhrase,
|
|
52
|
-
);
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
it('should show waiting phrase and retain elapsedTime when WaitingForConfirmation', async () => {
|
|
56
|
-
const { result, rerender } = renderHook(
|
|
57
|
-
({ streamingState }) => useLoadingIndicator(streamingState),
|
|
58
|
-
{ initialProps: { streamingState: StreamingState.Responding } },
|
|
59
|
-
);
|
|
60
|
-
|
|
61
|
-
await act(async () => {
|
|
62
|
-
await vi.advanceTimersByTimeAsync(60000);
|
|
63
|
-
});
|
|
64
|
-
expect(result.current.elapsedTime).toBe(60);
|
|
65
|
-
|
|
66
|
-
act(() => {
|
|
67
|
-
rerender({ streamingState: StreamingState.WaitingForConfirmation });
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
expect(result.current.currentLoadingPhrase).toBe(
|
|
71
|
-
'Waiting for user confirmation...',
|
|
72
|
-
);
|
|
73
|
-
expect(result.current.elapsedTime).toBe(60); // Elapsed time should be retained
|
|
74
|
-
|
|
75
|
-
// Timer should not advance further
|
|
76
|
-
await act(async () => {
|
|
77
|
-
await vi.advanceTimersByTimeAsync(2000);
|
|
78
|
-
});
|
|
79
|
-
expect(result.current.elapsedTime).toBe(60);
|
|
80
|
-
});
|
|
81
|
-
|
|
82
|
-
it('should reset elapsedTime and use a witty phrase when transitioning from WaitingForConfirmation to Responding', async () => {
|
|
83
|
-
const { result, rerender } = renderHook(
|
|
84
|
-
({ streamingState }) => useLoadingIndicator(streamingState),
|
|
85
|
-
{ initialProps: { streamingState: StreamingState.Responding } },
|
|
86
|
-
);
|
|
87
|
-
|
|
88
|
-
await act(async () => {
|
|
89
|
-
await vi.advanceTimersByTimeAsync(5000); // 5s
|
|
90
|
-
});
|
|
91
|
-
expect(result.current.elapsedTime).toBe(5);
|
|
92
|
-
|
|
93
|
-
act(() => {
|
|
94
|
-
rerender({ streamingState: StreamingState.WaitingForConfirmation });
|
|
95
|
-
});
|
|
96
|
-
expect(result.current.elapsedTime).toBe(5);
|
|
97
|
-
expect(result.current.currentLoadingPhrase).toBe(
|
|
98
|
-
'Waiting for user confirmation...',
|
|
99
|
-
);
|
|
100
|
-
|
|
101
|
-
act(() => {
|
|
102
|
-
rerender({ streamingState: StreamingState.Responding });
|
|
103
|
-
});
|
|
104
|
-
expect(result.current.elapsedTime).toBe(0); // Should reset
|
|
105
|
-
expect(WITTY_LOADING_PHRASES).toContain(
|
|
106
|
-
result.current.currentLoadingPhrase,
|
|
107
|
-
);
|
|
108
|
-
|
|
109
|
-
await act(async () => {
|
|
110
|
-
await vi.advanceTimersByTimeAsync(1000);
|
|
111
|
-
});
|
|
112
|
-
expect(result.current.elapsedTime).toBe(1);
|
|
113
|
-
});
|
|
114
|
-
|
|
115
|
-
it('should reset timer and phrase when streamingState changes from Responding to Idle', async () => {
|
|
116
|
-
const { result, rerender } = renderHook(
|
|
117
|
-
({ streamingState }) => useLoadingIndicator(streamingState),
|
|
118
|
-
{ initialProps: { streamingState: StreamingState.Responding } },
|
|
119
|
-
);
|
|
120
|
-
|
|
121
|
-
await act(async () => {
|
|
122
|
-
await vi.advanceTimersByTimeAsync(10000); // 10s
|
|
123
|
-
});
|
|
124
|
-
expect(result.current.elapsedTime).toBe(10);
|
|
125
|
-
|
|
126
|
-
act(() => {
|
|
127
|
-
rerender({ streamingState: StreamingState.Idle });
|
|
128
|
-
});
|
|
129
|
-
|
|
130
|
-
expect(result.current.elapsedTime).toBe(0);
|
|
131
|
-
expect(result.current.currentLoadingPhrase).toBe(WITTY_LOADING_PHRASES[0]);
|
|
132
|
-
|
|
133
|
-
// Timer should not advance
|
|
134
|
-
await act(async () => {
|
|
135
|
-
await vi.advanceTimersByTimeAsync(2000);
|
|
136
|
-
});
|
|
137
|
-
expect(result.current.elapsedTime).toBe(0);
|
|
138
|
-
});
|
|
139
|
-
});
|
|
@@ -1,57 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { StreamingState } from '../types.js';
|
|
8
|
-
import { useTimer } from './useTimer.js';
|
|
9
|
-
import { usePhraseCycler } from './usePhraseCycler.js';
|
|
10
|
-
import { useState, useEffect, useRef } from 'react'; // Added useRef
|
|
11
|
-
|
|
12
|
-
export const useLoadingIndicator = (streamingState: StreamingState) => {
|
|
13
|
-
const [timerResetKey, setTimerResetKey] = useState(0);
|
|
14
|
-
const isTimerActive = streamingState === StreamingState.Responding;
|
|
15
|
-
|
|
16
|
-
const elapsedTimeFromTimer = useTimer(isTimerActive, timerResetKey);
|
|
17
|
-
|
|
18
|
-
const isPhraseCyclingActive = streamingState === StreamingState.Responding;
|
|
19
|
-
const isWaiting = streamingState === StreamingState.WaitingForConfirmation;
|
|
20
|
-
const currentLoadingPhrase = usePhraseCycler(
|
|
21
|
-
isPhraseCyclingActive,
|
|
22
|
-
isWaiting,
|
|
23
|
-
);
|
|
24
|
-
|
|
25
|
-
const [retainedElapsedTime, setRetainedElapsedTime] = useState(0);
|
|
26
|
-
const prevStreamingStateRef = useRef<StreamingState | null>(null);
|
|
27
|
-
|
|
28
|
-
useEffect(() => {
|
|
29
|
-
if (
|
|
30
|
-
prevStreamingStateRef.current === StreamingState.WaitingForConfirmation &&
|
|
31
|
-
streamingState === StreamingState.Responding
|
|
32
|
-
) {
|
|
33
|
-
setTimerResetKey((prevKey) => prevKey + 1);
|
|
34
|
-
setRetainedElapsedTime(0); // Clear retained time when going back to responding
|
|
35
|
-
} else if (
|
|
36
|
-
streamingState === StreamingState.Idle &&
|
|
37
|
-
prevStreamingStateRef.current === StreamingState.Responding
|
|
38
|
-
) {
|
|
39
|
-
setTimerResetKey((prevKey) => prevKey + 1); // Reset timer when becoming idle from responding
|
|
40
|
-
setRetainedElapsedTime(0);
|
|
41
|
-
} else if (streamingState === StreamingState.WaitingForConfirmation) {
|
|
42
|
-
// Capture the time when entering WaitingForConfirmation
|
|
43
|
-
// elapsedTimeFromTimer will hold the last value from when isTimerActive was true.
|
|
44
|
-
setRetainedElapsedTime(elapsedTimeFromTimer);
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
prevStreamingStateRef.current = streamingState;
|
|
48
|
-
}, [streamingState, elapsedTimeFromTimer]);
|
|
49
|
-
|
|
50
|
-
return {
|
|
51
|
-
elapsedTime:
|
|
52
|
-
streamingState === StreamingState.WaitingForConfirmation
|
|
53
|
-
? retainedElapsedTime
|
|
54
|
-
: elapsedTimeFromTimer,
|
|
55
|
-
currentLoadingPhrase,
|
|
56
|
-
};
|
|
57
|
-
};
|
|
@@ -1,32 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { useState, useEffect } from 'react';
|
|
8
|
-
import { sessionId, Logger } from 'fss-link-core';
|
|
9
|
-
|
|
10
|
-
/**
|
|
11
|
-
* Hook to manage the logger instance.
|
|
12
|
-
*/
|
|
13
|
-
export const useLogger = () => {
|
|
14
|
-
const [logger, setLogger] = useState<Logger | null>(null);
|
|
15
|
-
|
|
16
|
-
useEffect(() => {
|
|
17
|
-
const newLogger = new Logger(sessionId);
|
|
18
|
-
/**
|
|
19
|
-
* Start async initialization, no need to await. Using await slows down the
|
|
20
|
-
* time from launch to see the gemini-cli prompt and it's better to not save
|
|
21
|
-
* messages than for the cli to hanging waiting for the logger to loading.
|
|
22
|
-
*/
|
|
23
|
-
newLogger
|
|
24
|
-
.initialize()
|
|
25
|
-
.then(() => {
|
|
26
|
-
setLogger(newLogger);
|
|
27
|
-
})
|
|
28
|
-
.catch(() => {});
|
|
29
|
-
}, []);
|
|
30
|
-
|
|
31
|
-
return logger;
|
|
32
|
-
};
|
|
@@ -1,226 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
8
|
-
import { renderHook, act } from '@testing-library/react';
|
|
9
|
-
import { useMessageQueue } from './useMessageQueue.js';
|
|
10
|
-
import { StreamingState } from '../types.js';
|
|
11
|
-
|
|
12
|
-
describe('useMessageQueue', () => {
|
|
13
|
-
let mockSubmitQuery: ReturnType<typeof vi.fn>;
|
|
14
|
-
|
|
15
|
-
beforeEach(() => {
|
|
16
|
-
mockSubmitQuery = vi.fn();
|
|
17
|
-
vi.useFakeTimers();
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
afterEach(() => {
|
|
21
|
-
vi.useRealTimers();
|
|
22
|
-
vi.clearAllMocks();
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
it('should initialize with empty queue', () => {
|
|
26
|
-
const { result } = renderHook(() =>
|
|
27
|
-
useMessageQueue({
|
|
28
|
-
streamingState: StreamingState.Idle,
|
|
29
|
-
submitQuery: mockSubmitQuery,
|
|
30
|
-
}),
|
|
31
|
-
);
|
|
32
|
-
|
|
33
|
-
expect(result.current.messageQueue).toEqual([]);
|
|
34
|
-
expect(result.current.getQueuedMessagesText()).toBe('');
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
it('should add messages to queue', () => {
|
|
38
|
-
const { result } = renderHook(() =>
|
|
39
|
-
useMessageQueue({
|
|
40
|
-
streamingState: StreamingState.Responding,
|
|
41
|
-
submitQuery: mockSubmitQuery,
|
|
42
|
-
}),
|
|
43
|
-
);
|
|
44
|
-
|
|
45
|
-
act(() => {
|
|
46
|
-
result.current.addMessage('Test message 1');
|
|
47
|
-
result.current.addMessage('Test message 2');
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
expect(result.current.messageQueue).toEqual([
|
|
51
|
-
'Test message 1',
|
|
52
|
-
'Test message 2',
|
|
53
|
-
]);
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
it('should filter out empty messages', () => {
|
|
57
|
-
const { result } = renderHook(() =>
|
|
58
|
-
useMessageQueue({
|
|
59
|
-
streamingState: StreamingState.Responding,
|
|
60
|
-
submitQuery: mockSubmitQuery,
|
|
61
|
-
}),
|
|
62
|
-
);
|
|
63
|
-
|
|
64
|
-
act(() => {
|
|
65
|
-
result.current.addMessage('Valid message');
|
|
66
|
-
result.current.addMessage(' '); // Only whitespace
|
|
67
|
-
result.current.addMessage(''); // Empty
|
|
68
|
-
result.current.addMessage('Another valid message');
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
expect(result.current.messageQueue).toEqual([
|
|
72
|
-
'Valid message',
|
|
73
|
-
'Another valid message',
|
|
74
|
-
]);
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
it('should clear queue', () => {
|
|
78
|
-
const { result } = renderHook(() =>
|
|
79
|
-
useMessageQueue({
|
|
80
|
-
streamingState: StreamingState.Responding,
|
|
81
|
-
submitQuery: mockSubmitQuery,
|
|
82
|
-
}),
|
|
83
|
-
);
|
|
84
|
-
|
|
85
|
-
act(() => {
|
|
86
|
-
result.current.addMessage('Test message');
|
|
87
|
-
});
|
|
88
|
-
|
|
89
|
-
expect(result.current.messageQueue).toEqual(['Test message']);
|
|
90
|
-
|
|
91
|
-
act(() => {
|
|
92
|
-
result.current.clearQueue();
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
expect(result.current.messageQueue).toEqual([]);
|
|
96
|
-
});
|
|
97
|
-
|
|
98
|
-
it('should return queued messages as text with double newlines', () => {
|
|
99
|
-
const { result } = renderHook(() =>
|
|
100
|
-
useMessageQueue({
|
|
101
|
-
streamingState: StreamingState.Responding,
|
|
102
|
-
submitQuery: mockSubmitQuery,
|
|
103
|
-
}),
|
|
104
|
-
);
|
|
105
|
-
|
|
106
|
-
act(() => {
|
|
107
|
-
result.current.addMessage('Message 1');
|
|
108
|
-
result.current.addMessage('Message 2');
|
|
109
|
-
result.current.addMessage('Message 3');
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
expect(result.current.getQueuedMessagesText()).toBe(
|
|
113
|
-
'Message 1\n\nMessage 2\n\nMessage 3',
|
|
114
|
-
);
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
it('should auto-submit queued messages when transitioning to Idle', () => {
|
|
118
|
-
const { result, rerender } = renderHook(
|
|
119
|
-
({ streamingState }) =>
|
|
120
|
-
useMessageQueue({
|
|
121
|
-
streamingState,
|
|
122
|
-
submitQuery: mockSubmitQuery,
|
|
123
|
-
}),
|
|
124
|
-
{
|
|
125
|
-
initialProps: { streamingState: StreamingState.Responding },
|
|
126
|
-
},
|
|
127
|
-
);
|
|
128
|
-
|
|
129
|
-
// Add some messages
|
|
130
|
-
act(() => {
|
|
131
|
-
result.current.addMessage('Message 1');
|
|
132
|
-
result.current.addMessage('Message 2');
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
expect(result.current.messageQueue).toEqual(['Message 1', 'Message 2']);
|
|
136
|
-
|
|
137
|
-
// Transition to Idle
|
|
138
|
-
rerender({ streamingState: StreamingState.Idle });
|
|
139
|
-
|
|
140
|
-
expect(mockSubmitQuery).toHaveBeenCalledWith('Message 1\n\nMessage 2');
|
|
141
|
-
expect(result.current.messageQueue).toEqual([]);
|
|
142
|
-
});
|
|
143
|
-
|
|
144
|
-
it('should not auto-submit when queue is empty', () => {
|
|
145
|
-
const { rerender } = renderHook(
|
|
146
|
-
({ streamingState }) =>
|
|
147
|
-
useMessageQueue({
|
|
148
|
-
streamingState,
|
|
149
|
-
submitQuery: mockSubmitQuery,
|
|
150
|
-
}),
|
|
151
|
-
{
|
|
152
|
-
initialProps: { streamingState: StreamingState.Responding },
|
|
153
|
-
},
|
|
154
|
-
);
|
|
155
|
-
|
|
156
|
-
// Transition to Idle with empty queue
|
|
157
|
-
rerender({ streamingState: StreamingState.Idle });
|
|
158
|
-
|
|
159
|
-
expect(mockSubmitQuery).not.toHaveBeenCalled();
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
it('should not auto-submit when not transitioning to Idle', () => {
|
|
163
|
-
const { result, rerender } = renderHook(
|
|
164
|
-
({ streamingState }) =>
|
|
165
|
-
useMessageQueue({
|
|
166
|
-
streamingState,
|
|
167
|
-
submitQuery: mockSubmitQuery,
|
|
168
|
-
}),
|
|
169
|
-
{
|
|
170
|
-
initialProps: { streamingState: StreamingState.Responding },
|
|
171
|
-
},
|
|
172
|
-
);
|
|
173
|
-
|
|
174
|
-
// Add messages
|
|
175
|
-
act(() => {
|
|
176
|
-
result.current.addMessage('Message 1');
|
|
177
|
-
});
|
|
178
|
-
|
|
179
|
-
// Transition to WaitingForConfirmation (not Idle)
|
|
180
|
-
rerender({ streamingState: StreamingState.WaitingForConfirmation });
|
|
181
|
-
|
|
182
|
-
expect(mockSubmitQuery).not.toHaveBeenCalled();
|
|
183
|
-
expect(result.current.messageQueue).toEqual(['Message 1']);
|
|
184
|
-
});
|
|
185
|
-
|
|
186
|
-
it('should handle multiple state transitions correctly', () => {
|
|
187
|
-
const { result, rerender } = renderHook(
|
|
188
|
-
({ streamingState }) =>
|
|
189
|
-
useMessageQueue({
|
|
190
|
-
streamingState,
|
|
191
|
-
submitQuery: mockSubmitQuery,
|
|
192
|
-
}),
|
|
193
|
-
{
|
|
194
|
-
initialProps: { streamingState: StreamingState.Idle },
|
|
195
|
-
},
|
|
196
|
-
);
|
|
197
|
-
|
|
198
|
-
// Start responding
|
|
199
|
-
rerender({ streamingState: StreamingState.Responding });
|
|
200
|
-
|
|
201
|
-
// Add messages while responding
|
|
202
|
-
act(() => {
|
|
203
|
-
result.current.addMessage('First batch');
|
|
204
|
-
});
|
|
205
|
-
|
|
206
|
-
// Go back to idle - should submit
|
|
207
|
-
rerender({ streamingState: StreamingState.Idle });
|
|
208
|
-
|
|
209
|
-
expect(mockSubmitQuery).toHaveBeenCalledWith('First batch');
|
|
210
|
-
expect(result.current.messageQueue).toEqual([]);
|
|
211
|
-
|
|
212
|
-
// Start responding again
|
|
213
|
-
rerender({ streamingState: StreamingState.Responding });
|
|
214
|
-
|
|
215
|
-
// Add more messages
|
|
216
|
-
act(() => {
|
|
217
|
-
result.current.addMessage('Second batch');
|
|
218
|
-
});
|
|
219
|
-
|
|
220
|
-
// Go back to idle - should submit again
|
|
221
|
-
rerender({ streamingState: StreamingState.Idle });
|
|
222
|
-
|
|
223
|
-
expect(mockSubmitQuery).toHaveBeenCalledWith('Second batch');
|
|
224
|
-
expect(mockSubmitQuery).toHaveBeenCalledTimes(2);
|
|
225
|
-
});
|
|
226
|
-
});
|
|
@@ -1,69 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { useCallback, useEffect, useState } from 'react';
|
|
8
|
-
import { StreamingState } from '../types.js';
|
|
9
|
-
|
|
10
|
-
export interface UseMessageQueueOptions {
|
|
11
|
-
streamingState: StreamingState;
|
|
12
|
-
submitQuery: (query: string) => void;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export interface UseMessageQueueReturn {
|
|
16
|
-
messageQueue: string[];
|
|
17
|
-
addMessage: (message: string) => void;
|
|
18
|
-
clearQueue: () => void;
|
|
19
|
-
getQueuedMessagesText: () => string;
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
/**
|
|
23
|
-
* Hook for managing message queuing during streaming responses.
|
|
24
|
-
* Allows users to queue messages while the AI is responding and automatically
|
|
25
|
-
* sends them when streaming completes.
|
|
26
|
-
*/
|
|
27
|
-
export function useMessageQueue({
|
|
28
|
-
streamingState,
|
|
29
|
-
submitQuery,
|
|
30
|
-
}: UseMessageQueueOptions): UseMessageQueueReturn {
|
|
31
|
-
const [messageQueue, setMessageQueue] = useState<string[]>([]);
|
|
32
|
-
|
|
33
|
-
// Add a message to the queue
|
|
34
|
-
const addMessage = useCallback((message: string) => {
|
|
35
|
-
const trimmedMessage = message.trim();
|
|
36
|
-
if (trimmedMessage.length > 0) {
|
|
37
|
-
setMessageQueue((prev) => [...prev, trimmedMessage]);
|
|
38
|
-
}
|
|
39
|
-
}, []);
|
|
40
|
-
|
|
41
|
-
// Clear the entire queue
|
|
42
|
-
const clearQueue = useCallback(() => {
|
|
43
|
-
setMessageQueue([]);
|
|
44
|
-
}, []);
|
|
45
|
-
|
|
46
|
-
// Get all queued messages as a single text string
|
|
47
|
-
const getQueuedMessagesText = useCallback(() => {
|
|
48
|
-
if (messageQueue.length === 0) return '';
|
|
49
|
-
return messageQueue.join('\n\n');
|
|
50
|
-
}, [messageQueue]);
|
|
51
|
-
|
|
52
|
-
// Process queued messages when streaming becomes idle
|
|
53
|
-
useEffect(() => {
|
|
54
|
-
if (streamingState === StreamingState.Idle && messageQueue.length > 0) {
|
|
55
|
-
// Combine all messages with double newlines for clarity
|
|
56
|
-
const combinedMessage = messageQueue.join('\n\n');
|
|
57
|
-
// Clear the queue and submit
|
|
58
|
-
setMessageQueue([]);
|
|
59
|
-
submitQuery(combinedMessage);
|
|
60
|
-
}
|
|
61
|
-
}, [streamingState, messageQueue, submitQuery]);
|
|
62
|
-
|
|
63
|
-
return {
|
|
64
|
-
messageQueue,
|
|
65
|
-
addMessage,
|
|
66
|
-
clearQueue,
|
|
67
|
-
getQueuedMessagesText,
|
|
68
|
-
};
|
|
69
|
-
}
|
|
@@ -1,145 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
8
|
-
import { renderHook, act } from '@testing-library/react';
|
|
9
|
-
import {
|
|
10
|
-
usePhraseCycler,
|
|
11
|
-
WITTY_LOADING_PHRASES,
|
|
12
|
-
PHRASE_CHANGE_INTERVAL_MS,
|
|
13
|
-
} from './usePhraseCycler.js';
|
|
14
|
-
|
|
15
|
-
describe('usePhraseCycler', () => {
|
|
16
|
-
beforeEach(() => {
|
|
17
|
-
vi.useFakeTimers();
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
afterEach(() => {
|
|
21
|
-
vi.restoreAllMocks();
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
it('should initialize with the first witty phrase when not active and not waiting', () => {
|
|
25
|
-
const { result } = renderHook(() => usePhraseCycler(false, false));
|
|
26
|
-
expect(result.current).toBe(WITTY_LOADING_PHRASES[0]);
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
it('should show "Waiting for user confirmation..." when isWaiting is true', () => {
|
|
30
|
-
const { result, rerender } = renderHook(
|
|
31
|
-
({ isActive, isWaiting }) => usePhraseCycler(isActive, isWaiting),
|
|
32
|
-
{ initialProps: { isActive: true, isWaiting: false } },
|
|
33
|
-
);
|
|
34
|
-
rerender({ isActive: true, isWaiting: true });
|
|
35
|
-
expect(result.current).toBe('Waiting for user confirmation...');
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
it('should not cycle phrases if isActive is false and not waiting', () => {
|
|
39
|
-
const { result } = renderHook(() => usePhraseCycler(false, false));
|
|
40
|
-
act(() => {
|
|
41
|
-
vi.advanceTimersByTime(PHRASE_CHANGE_INTERVAL_MS * 2);
|
|
42
|
-
});
|
|
43
|
-
expect(result.current).toBe(WITTY_LOADING_PHRASES[0]);
|
|
44
|
-
});
|
|
45
|
-
|
|
46
|
-
it('should cycle through witty phrases when isActive is true and not waiting', () => {
|
|
47
|
-
const { result } = renderHook(() => usePhraseCycler(true, false));
|
|
48
|
-
// Initial phrase should be one of the witty phrases
|
|
49
|
-
expect(WITTY_LOADING_PHRASES).toContain(result.current);
|
|
50
|
-
const _initialPhrase = result.current;
|
|
51
|
-
|
|
52
|
-
act(() => {
|
|
53
|
-
vi.advanceTimersByTime(PHRASE_CHANGE_INTERVAL_MS);
|
|
54
|
-
});
|
|
55
|
-
// Phrase should change and be one of the witty phrases
|
|
56
|
-
expect(WITTY_LOADING_PHRASES).toContain(result.current);
|
|
57
|
-
|
|
58
|
-
const _secondPhrase = result.current;
|
|
59
|
-
act(() => {
|
|
60
|
-
vi.advanceTimersByTime(PHRASE_CHANGE_INTERVAL_MS);
|
|
61
|
-
});
|
|
62
|
-
expect(WITTY_LOADING_PHRASES).toContain(result.current);
|
|
63
|
-
});
|
|
64
|
-
|
|
65
|
-
it('should reset to a witty phrase when isActive becomes true after being false (and not waiting)', () => {
|
|
66
|
-
// Ensure there are at least two phrases for this test to be meaningful.
|
|
67
|
-
if (WITTY_LOADING_PHRASES.length < 2) {
|
|
68
|
-
return;
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
// Mock Math.random to make the test deterministic.
|
|
72
|
-
let callCount = 0;
|
|
73
|
-
vi.spyOn(Math, 'random').mockImplementation(() => {
|
|
74
|
-
// Cycle through 0, 1, 0, 1, ...
|
|
75
|
-
const val = callCount % 2;
|
|
76
|
-
callCount++;
|
|
77
|
-
return val / WITTY_LOADING_PHRASES.length;
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
const { result, rerender } = renderHook(
|
|
81
|
-
({ isActive, isWaiting }) => usePhraseCycler(isActive, isWaiting),
|
|
82
|
-
{ initialProps: { isActive: false, isWaiting: false } },
|
|
83
|
-
);
|
|
84
|
-
|
|
85
|
-
// Activate
|
|
86
|
-
rerender({ isActive: true, isWaiting: false });
|
|
87
|
-
const firstActivePhrase = result.current;
|
|
88
|
-
expect(WITTY_LOADING_PHRASES).toContain(firstActivePhrase);
|
|
89
|
-
// With our mock, this should be the first phrase.
|
|
90
|
-
expect(firstActivePhrase).toBe(WITTY_LOADING_PHRASES[0]);
|
|
91
|
-
|
|
92
|
-
act(() => {
|
|
93
|
-
vi.advanceTimersByTime(PHRASE_CHANGE_INTERVAL_MS);
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
// Phrase should change to the second phrase.
|
|
97
|
-
expect(result.current).not.toBe(firstActivePhrase);
|
|
98
|
-
expect(result.current).toBe(WITTY_LOADING_PHRASES[1]);
|
|
99
|
-
|
|
100
|
-
// Set to inactive - should reset to the default initial phrase
|
|
101
|
-
rerender({ isActive: false, isWaiting: false });
|
|
102
|
-
expect(result.current).toBe(WITTY_LOADING_PHRASES[0]);
|
|
103
|
-
|
|
104
|
-
// Set back to active - should pick a random witty phrase (which our mock controls)
|
|
105
|
-
act(() => {
|
|
106
|
-
rerender({ isActive: true, isWaiting: false });
|
|
107
|
-
});
|
|
108
|
-
// The random mock will now return 0, so it should be the first phrase again.
|
|
109
|
-
expect(result.current).toBe(WITTY_LOADING_PHRASES[0]);
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
it('should clear phrase interval on unmount when active', () => {
|
|
113
|
-
const { unmount } = renderHook(() => usePhraseCycler(true, false));
|
|
114
|
-
const clearIntervalSpy = vi.spyOn(global, 'clearInterval');
|
|
115
|
-
unmount();
|
|
116
|
-
expect(clearIntervalSpy).toHaveBeenCalledOnce();
|
|
117
|
-
});
|
|
118
|
-
|
|
119
|
-
it('should reset to a witty phrase when transitioning from waiting to active', () => {
|
|
120
|
-
const { result, rerender } = renderHook(
|
|
121
|
-
({ isActive, isWaiting }) => usePhraseCycler(isActive, isWaiting),
|
|
122
|
-
{ initialProps: { isActive: true, isWaiting: false } },
|
|
123
|
-
);
|
|
124
|
-
|
|
125
|
-
const _initialPhrase = result.current;
|
|
126
|
-
expect(WITTY_LOADING_PHRASES).toContain(_initialPhrase);
|
|
127
|
-
|
|
128
|
-
// Cycle to a different phrase (potentially)
|
|
129
|
-
act(() => {
|
|
130
|
-
vi.advanceTimersByTime(PHRASE_CHANGE_INTERVAL_MS);
|
|
131
|
-
});
|
|
132
|
-
if (WITTY_LOADING_PHRASES.length > 1) {
|
|
133
|
-
// This check is probabilistic with random selection
|
|
134
|
-
}
|
|
135
|
-
expect(WITTY_LOADING_PHRASES).toContain(result.current);
|
|
136
|
-
|
|
137
|
-
// Go to waiting state
|
|
138
|
-
rerender({ isActive: false, isWaiting: true });
|
|
139
|
-
expect(result.current).toBe('Waiting for user confirmation...');
|
|
140
|
-
|
|
141
|
-
// Go back to active cycling - should pick a random witty phrase
|
|
142
|
-
rerender({ isActive: true, isWaiting: false });
|
|
143
|
-
expect(WITTY_LOADING_PHRASES).toContain(result.current);
|
|
144
|
-
});
|
|
145
|
-
});
|