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,280 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import {
|
|
8
|
-
afterEach,
|
|
9
|
-
beforeEach,
|
|
10
|
-
describe,
|
|
11
|
-
expect,
|
|
12
|
-
it,
|
|
13
|
-
vi,
|
|
14
|
-
MockedFunction,
|
|
15
|
-
} from 'vitest';
|
|
16
|
-
import { act } from 'react';
|
|
17
|
-
import { renderHook } from '@testing-library/react';
|
|
18
|
-
import { useGitBranchName } from './useGitBranchName.js';
|
|
19
|
-
import { fs, vol } from 'memfs'; // For mocking fs
|
|
20
|
-
// NOTE: memfs FSWatcher is incompatible with Node.js FSWatcher (missing ref/unref methods)
|
|
21
|
-
import * as nodeFsPromises from 'node:fs/promises';
|
|
22
|
-
import { EventEmitter } from 'node:events';
|
|
23
|
-
import { exec as mockExec, type ChildProcess } from 'node:child_process';
|
|
24
|
-
// TECHNICAL DEBT: 2 tests skipped due to memfs FSWatcher interface mismatch
|
|
25
|
-
// memfs.FSWatcher lacks ref()/unref() methods and real file event triggering
|
|
26
|
-
// Future fix: Use real temp git directories instead of memfs for file watcher tests
|
|
27
|
-
|
|
28
|
-
// Mock child_process
|
|
29
|
-
vi.mock('child_process');
|
|
30
|
-
|
|
31
|
-
// Mock fs and fs/promises
|
|
32
|
-
vi.mock('node:fs', async () => {
|
|
33
|
-
const memfs = await vi.importActual<typeof import('memfs')>('memfs');
|
|
34
|
-
return memfs.fs;
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
vi.mock('node:fs/promises', async () => {
|
|
38
|
-
const memfs = await vi.importActual<typeof import('memfs')>('memfs');
|
|
39
|
-
return memfs.fs.promises;
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
const CWD = '/test/project';
|
|
43
|
-
const GIT_HEAD_PATH = `${CWD}/.git/HEAD`;
|
|
44
|
-
const GIT_LOGS_HEAD_PATH = `${CWD}/.git/logs/HEAD`;
|
|
45
|
-
|
|
46
|
-
describe('useGitBranchName', () => {
|
|
47
|
-
beforeEach(() => {
|
|
48
|
-
vol.reset(); // Reset in-memory filesystem
|
|
49
|
-
vol.fromJSON({
|
|
50
|
-
[GIT_HEAD_PATH]: 'ref: refs/heads/main',
|
|
51
|
-
[GIT_LOGS_HEAD_PATH]: '0000000000000000000000000000000000000000 1234567890abcdef1234567890abcdef12345678 User <user@example.com> 1609459200 +0000\tcommit (initial): Initial commit\n',
|
|
52
|
-
});
|
|
53
|
-
vi.useFakeTimers(); // Use fake timers for async operations
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
afterEach(() => {
|
|
57
|
-
vi.restoreAllMocks();
|
|
58
|
-
vi.clearAllTimers();
|
|
59
|
-
});
|
|
60
|
-
|
|
61
|
-
it('should return branch name', async () => {
|
|
62
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementation(
|
|
63
|
-
(_command, _options, callback) => {
|
|
64
|
-
callback?.(null, 'main\n', '');
|
|
65
|
-
return new EventEmitter() as ChildProcess;
|
|
66
|
-
},
|
|
67
|
-
);
|
|
68
|
-
|
|
69
|
-
const { result, rerender } = renderHook(() => useGitBranchName(CWD));
|
|
70
|
-
|
|
71
|
-
await act(async () => {
|
|
72
|
-
vi.runAllTimers(); // Advance timers to trigger useEffect and exec callback
|
|
73
|
-
rerender(); // Rerender to get the updated state
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
expect(result.current).toBe('main');
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
it('should return undefined if git command fails', async () => {
|
|
80
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementation(
|
|
81
|
-
(_command, _options, callback) => {
|
|
82
|
-
callback?.(new Error('Git error'), '', 'error output');
|
|
83
|
-
return new EventEmitter() as ChildProcess;
|
|
84
|
-
},
|
|
85
|
-
);
|
|
86
|
-
|
|
87
|
-
const { result, rerender } = renderHook(() => useGitBranchName(CWD));
|
|
88
|
-
expect(result.current).toBeUndefined();
|
|
89
|
-
|
|
90
|
-
await act(async () => {
|
|
91
|
-
vi.runAllTimers();
|
|
92
|
-
rerender();
|
|
93
|
-
});
|
|
94
|
-
expect(result.current).toBeUndefined();
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
it('should return short commit hash if branch is HEAD (detached state)', async () => {
|
|
98
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementation(
|
|
99
|
-
(command, _options, callback) => {
|
|
100
|
-
if (command === 'git rev-parse --abbrev-ref HEAD') {
|
|
101
|
-
callback?.(null, 'HEAD\n', '');
|
|
102
|
-
} else if (command === 'git rev-parse --short HEAD') {
|
|
103
|
-
callback?.(null, 'a1b2c3d\n', '');
|
|
104
|
-
}
|
|
105
|
-
return new EventEmitter() as ChildProcess;
|
|
106
|
-
},
|
|
107
|
-
);
|
|
108
|
-
|
|
109
|
-
const { result, rerender } = renderHook(() => useGitBranchName(CWD));
|
|
110
|
-
await act(async () => {
|
|
111
|
-
vi.runAllTimers();
|
|
112
|
-
rerender();
|
|
113
|
-
});
|
|
114
|
-
expect(result.current).toBe('a1b2c3d');
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
it('should return undefined if branch is HEAD and getting commit hash fails', async () => {
|
|
118
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementation(
|
|
119
|
-
(command, _options, callback) => {
|
|
120
|
-
if (command === 'git rev-parse --abbrev-ref HEAD') {
|
|
121
|
-
callback?.(null, 'HEAD\n', '');
|
|
122
|
-
} else if (command === 'git rev-parse --short HEAD') {
|
|
123
|
-
callback?.(new Error('Git error'), '', 'error output');
|
|
124
|
-
}
|
|
125
|
-
return new EventEmitter() as ChildProcess;
|
|
126
|
-
},
|
|
127
|
-
);
|
|
128
|
-
|
|
129
|
-
const { result, rerender } = renderHook(() => useGitBranchName(CWD));
|
|
130
|
-
await act(async () => {
|
|
131
|
-
vi.runAllTimers();
|
|
132
|
-
rerender();
|
|
133
|
-
});
|
|
134
|
-
expect(result.current).toBeUndefined();
|
|
135
|
-
});
|
|
136
|
-
|
|
137
|
-
it.skip('should update branch name when .git/logs/HEAD changes', async () => {
|
|
138
|
-
// SKIP: memfs FSWatcher incompatible with Node.js FSWatcher (lacks ref/unref methods)
|
|
139
|
-
// Core git functionality verified by passing tests above. Real usage unaffected.
|
|
140
|
-
let capturedWatcher: any;
|
|
141
|
-
|
|
142
|
-
// Critical: Mock fsPromises.access to ensure file exists check passes
|
|
143
|
-
// Since memfs replaces fs/promises, we need to spy on the memfs promises
|
|
144
|
-
vi.spyOn(nodeFsPromises, 'access').mockResolvedValue(undefined);
|
|
145
|
-
|
|
146
|
-
// Mock fs.watch to work with memfs - we need to intercept the memfs fs.watch
|
|
147
|
-
const watchMock = vi.spyOn(fs, 'watch').mockImplementation((path, callback) => {
|
|
148
|
-
// Create a mock watcher that looks like memfs FSWatcher
|
|
149
|
-
const mockWatcher = new EventEmitter();
|
|
150
|
-
(mockWatcher as any).close = vi.fn();
|
|
151
|
-
|
|
152
|
-
capturedWatcher = {
|
|
153
|
-
watcher: mockWatcher,
|
|
154
|
-
path,
|
|
155
|
-
callback,
|
|
156
|
-
};
|
|
157
|
-
|
|
158
|
-
return mockWatcher as any;
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementation(
|
|
162
|
-
(_command, _options, callback) => {
|
|
163
|
-
// First call returns 'main', second call returns 'develop'
|
|
164
|
-
const callCount = (mockExec as any).__call_count__ || 0;
|
|
165
|
-
(mockExec as any).__call_count__ = callCount + 1;
|
|
166
|
-
|
|
167
|
-
if (callCount === 0) {
|
|
168
|
-
callback?.(null, 'main\n', '');
|
|
169
|
-
} else {
|
|
170
|
-
callback?.(null, 'develop\n', '');
|
|
171
|
-
}
|
|
172
|
-
return new EventEmitter() as ChildProcess;
|
|
173
|
-
},
|
|
174
|
-
);
|
|
175
|
-
|
|
176
|
-
const { result, rerender } = renderHook(() => useGitBranchName(CWD));
|
|
177
|
-
|
|
178
|
-
// Wait for initial fetch and watcher setup
|
|
179
|
-
await act(async () => {
|
|
180
|
-
vi.runAllTimers();
|
|
181
|
-
rerender();
|
|
182
|
-
});
|
|
183
|
-
expect(result.current).toBe('main');
|
|
184
|
-
expect(watchMock).toHaveBeenCalledWith(GIT_LOGS_HEAD_PATH, expect.any(Function));
|
|
185
|
-
expect(capturedWatcher).toBeDefined();
|
|
186
|
-
|
|
187
|
-
// Simulate file change event by triggering the watcher
|
|
188
|
-
await act(async () => {
|
|
189
|
-
// Simulate the memfs file watcher detecting a change
|
|
190
|
-
if (capturedWatcher && capturedWatcher.callback) {
|
|
191
|
-
capturedWatcher.callback('change');
|
|
192
|
-
}
|
|
193
|
-
vi.runAllTimers(); // Process timers for exec
|
|
194
|
-
rerender();
|
|
195
|
-
});
|
|
196
|
-
|
|
197
|
-
expect(result.current).toBe('develop');
|
|
198
|
-
});
|
|
199
|
-
|
|
200
|
-
it('should handle watcher setup error silently', async () => {
|
|
201
|
-
// Remove .git/logs/HEAD to cause an error in fs.watch setup
|
|
202
|
-
vol.unlinkSync(GIT_LOGS_HEAD_PATH);
|
|
203
|
-
|
|
204
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementation(
|
|
205
|
-
(_command, _options, callback) => {
|
|
206
|
-
callback?.(null, 'main\n', '');
|
|
207
|
-
return new EventEmitter() as ChildProcess;
|
|
208
|
-
},
|
|
209
|
-
);
|
|
210
|
-
|
|
211
|
-
const { result, rerender } = renderHook(() => useGitBranchName(CWD));
|
|
212
|
-
|
|
213
|
-
await act(async () => {
|
|
214
|
-
vi.runAllTimers();
|
|
215
|
-
rerender();
|
|
216
|
-
});
|
|
217
|
-
|
|
218
|
-
expect(result.current).toBe('main'); // Branch name should still be fetched initially
|
|
219
|
-
|
|
220
|
-
// Try to trigger a change that would normally be caught by the watcher
|
|
221
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementationOnce(
|
|
222
|
-
(_command, _options, callback) => {
|
|
223
|
-
callback?.(null, 'develop\n', '');
|
|
224
|
-
return new EventEmitter() as ChildProcess;
|
|
225
|
-
},
|
|
226
|
-
);
|
|
227
|
-
|
|
228
|
-
// This write would trigger the watcher if it was set up
|
|
229
|
-
// but since it failed, the branch name should not update
|
|
230
|
-
// We need to create the file again for writeFileSync to not throw
|
|
231
|
-
vol.fromJSON({
|
|
232
|
-
[GIT_LOGS_HEAD_PATH]: '1234567890abcdef1234567890abcdef12345678 abcdef1234567890abcdef1234567890abcdef12 User <user@example.com> 1609462800 +0000\tcheckout: moving from main to develop\n',
|
|
233
|
-
});
|
|
234
|
-
|
|
235
|
-
await act(async () => {
|
|
236
|
-
fs.writeFileSync(GIT_LOGS_HEAD_PATH, '1234567890abcdef1234567890abcdef12345678 abcdef1234567890abcdef1234567890abcdef12 User <user@example.com> 1609462800 +0000\tcheckout: moving from main to develop\n');
|
|
237
|
-
vi.runAllTimers();
|
|
238
|
-
rerender();
|
|
239
|
-
});
|
|
240
|
-
|
|
241
|
-
// Branch name should not change because watcher setup failed
|
|
242
|
-
expect(result.current).toBe('main');
|
|
243
|
-
});
|
|
244
|
-
|
|
245
|
-
it.skip('should cleanup watcher on unmount', async () => {
|
|
246
|
-
// SKIP: memfs FSWatcher incompatible with Node.js FSWatcher interface
|
|
247
|
-
// Cleanup logic verified by manual testing and real-world usage.
|
|
248
|
-
const closeMock = vi.fn();
|
|
249
|
-
|
|
250
|
-
// Critical: Mock fsPromises.access to ensure file exists check passes
|
|
251
|
-
// Since memfs replaces fs/promises, we need to spy on the memfs promises
|
|
252
|
-
vi.spyOn(nodeFsPromises, 'access').mockResolvedValue(undefined);
|
|
253
|
-
|
|
254
|
-
// Mock fs.watch to work with memfs
|
|
255
|
-
const watchMock = vi.spyOn(fs, 'watch').mockImplementation((_path, _callback) => {
|
|
256
|
-
const mockWatcher = new EventEmitter();
|
|
257
|
-
(mockWatcher as any).close = closeMock;
|
|
258
|
-
return mockWatcher as any;
|
|
259
|
-
});
|
|
260
|
-
|
|
261
|
-
(mockExec as MockedFunction<typeof mockExec>).mockImplementation(
|
|
262
|
-
(_command, _options, callback) => {
|
|
263
|
-
callback?.(null, 'main\n', '');
|
|
264
|
-
return new EventEmitter() as ChildProcess;
|
|
265
|
-
},
|
|
266
|
-
);
|
|
267
|
-
|
|
268
|
-
const { unmount, rerender } = renderHook(() => useGitBranchName(CWD));
|
|
269
|
-
|
|
270
|
-
// Wait for watcher setup
|
|
271
|
-
await act(async () => {
|
|
272
|
-
vi.runAllTimers();
|
|
273
|
-
rerender();
|
|
274
|
-
});
|
|
275
|
-
|
|
276
|
-
unmount();
|
|
277
|
-
expect(watchMock).toHaveBeenCalledWith(GIT_LOGS_HEAD_PATH, expect.any(Function));
|
|
278
|
-
expect(closeMock).toHaveBeenCalled();
|
|
279
|
-
});
|
|
280
|
-
});
|
|
@@ -1,79 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { useState, useEffect, useCallback } from 'react';
|
|
8
|
-
import { exec } from 'node:child_process';
|
|
9
|
-
import fs from 'node:fs';
|
|
10
|
-
import fsPromises from 'node:fs/promises';
|
|
11
|
-
import path from 'path';
|
|
12
|
-
|
|
13
|
-
export function useGitBranchName(cwd: string): string | undefined {
|
|
14
|
-
const [branchName, setBranchName] = useState<string | undefined>(undefined);
|
|
15
|
-
|
|
16
|
-
const fetchBranchName = useCallback(
|
|
17
|
-
() =>
|
|
18
|
-
exec(
|
|
19
|
-
'git rev-parse --abbrev-ref HEAD',
|
|
20
|
-
{ cwd },
|
|
21
|
-
(error, stdout, _stderr) => {
|
|
22
|
-
if (error) {
|
|
23
|
-
setBranchName(undefined);
|
|
24
|
-
return;
|
|
25
|
-
}
|
|
26
|
-
const branch = stdout.toString().trim();
|
|
27
|
-
if (branch && branch !== 'HEAD') {
|
|
28
|
-
setBranchName(branch);
|
|
29
|
-
} else {
|
|
30
|
-
exec(
|
|
31
|
-
'git rev-parse --short HEAD',
|
|
32
|
-
{ cwd },
|
|
33
|
-
(error, stdout, _stderr) => {
|
|
34
|
-
if (error) {
|
|
35
|
-
setBranchName(undefined);
|
|
36
|
-
return;
|
|
37
|
-
}
|
|
38
|
-
setBranchName(stdout.toString().trim());
|
|
39
|
-
},
|
|
40
|
-
);
|
|
41
|
-
}
|
|
42
|
-
},
|
|
43
|
-
),
|
|
44
|
-
[cwd, setBranchName],
|
|
45
|
-
);
|
|
46
|
-
|
|
47
|
-
useEffect(() => {
|
|
48
|
-
fetchBranchName(); // Initial fetch
|
|
49
|
-
|
|
50
|
-
const gitLogsHeadPath = path.join(cwd, '.git', 'logs', 'HEAD');
|
|
51
|
-
let watcher: fs.FSWatcher | undefined;
|
|
52
|
-
|
|
53
|
-
const setupWatcher = async () => {
|
|
54
|
-
try {
|
|
55
|
-
// Check if .git/logs/HEAD exists, as it might not in a new repo or orphaned head
|
|
56
|
-
await fsPromises.access(gitLogsHeadPath, fs.constants.F_OK);
|
|
57
|
-
watcher = fs.watch(gitLogsHeadPath, (eventType: string) => {
|
|
58
|
-
// Changes to .git/logs/HEAD (appends) indicate HEAD has likely changed
|
|
59
|
-
if (eventType === 'change' || eventType === 'rename') {
|
|
60
|
-
// Handle rename just in case
|
|
61
|
-
fetchBranchName();
|
|
62
|
-
}
|
|
63
|
-
});
|
|
64
|
-
} catch (_watchError) {
|
|
65
|
-
// Silently ignore watcher errors (e.g. permissions or file not existing),
|
|
66
|
-
// similar to how exec errors are handled.
|
|
67
|
-
// The branch name will simply not update automatically.
|
|
68
|
-
}
|
|
69
|
-
};
|
|
70
|
-
|
|
71
|
-
setupWatcher();
|
|
72
|
-
|
|
73
|
-
return () => {
|
|
74
|
-
watcher?.close();
|
|
75
|
-
};
|
|
76
|
-
}, [cwd, fetchBranchName]);
|
|
77
|
-
|
|
78
|
-
return branchName;
|
|
79
|
-
}
|
|
@@ -1,202 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { describe, it, expect } from 'vitest';
|
|
8
|
-
import { renderHook, act } from '@testing-library/react';
|
|
9
|
-
import { useHistory } from './useHistoryManager.js';
|
|
10
|
-
import { HistoryItem } from '../types.js';
|
|
11
|
-
|
|
12
|
-
describe('useHistoryManager', () => {
|
|
13
|
-
it('should initialize with an empty history', () => {
|
|
14
|
-
const { result } = renderHook(() => useHistory());
|
|
15
|
-
expect(result.current.history).toEqual([]);
|
|
16
|
-
});
|
|
17
|
-
|
|
18
|
-
it('should add an item to history with a unique ID', () => {
|
|
19
|
-
const { result } = renderHook(() => useHistory());
|
|
20
|
-
const timestamp = Date.now();
|
|
21
|
-
const itemData: Omit<HistoryItem, 'id'> = {
|
|
22
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
23
|
-
text: 'Hello',
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
act(() => {
|
|
27
|
-
result.current.addItem(itemData, timestamp);
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
expect(result.current.history).toHaveLength(1);
|
|
31
|
-
expect(result.current.history[0]).toEqual(
|
|
32
|
-
expect.objectContaining({
|
|
33
|
-
...itemData,
|
|
34
|
-
id: expect.any(Number),
|
|
35
|
-
}),
|
|
36
|
-
);
|
|
37
|
-
// Basic check that ID incorporates timestamp
|
|
38
|
-
expect(result.current.history[0].id).toBeGreaterThanOrEqual(timestamp);
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
it('should generate unique IDs for items added with the same base timestamp', () => {
|
|
42
|
-
const { result } = renderHook(() => useHistory());
|
|
43
|
-
const timestamp = Date.now();
|
|
44
|
-
const itemData1: Omit<HistoryItem, 'id'> = {
|
|
45
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
46
|
-
text: 'First',
|
|
47
|
-
};
|
|
48
|
-
const itemData2: Omit<HistoryItem, 'id'> = {
|
|
49
|
-
type: 'gemini', // Replaced HistoryItemType.Gemini
|
|
50
|
-
text: 'Second',
|
|
51
|
-
};
|
|
52
|
-
|
|
53
|
-
let id1!: number;
|
|
54
|
-
let id2!: number;
|
|
55
|
-
|
|
56
|
-
act(() => {
|
|
57
|
-
id1 = result.current.addItem(itemData1, timestamp);
|
|
58
|
-
id2 = result.current.addItem(itemData2, timestamp);
|
|
59
|
-
});
|
|
60
|
-
|
|
61
|
-
expect(result.current.history).toHaveLength(2);
|
|
62
|
-
expect(id1).not.toEqual(id2);
|
|
63
|
-
expect(result.current.history[0].id).toEqual(id1);
|
|
64
|
-
expect(result.current.history[1].id).toEqual(id2);
|
|
65
|
-
// IDs should be sequential based on the counter
|
|
66
|
-
expect(id2).toBeGreaterThan(id1);
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
it('should update an existing history item', () => {
|
|
70
|
-
const { result } = renderHook(() => useHistory());
|
|
71
|
-
const timestamp = Date.now();
|
|
72
|
-
const initialItem: Omit<HistoryItem, 'id'> = {
|
|
73
|
-
type: 'gemini', // Replaced HistoryItemType.Gemini
|
|
74
|
-
text: 'Initial content',
|
|
75
|
-
};
|
|
76
|
-
let itemId!: number;
|
|
77
|
-
|
|
78
|
-
act(() => {
|
|
79
|
-
itemId = result.current.addItem(initialItem, timestamp);
|
|
80
|
-
});
|
|
81
|
-
|
|
82
|
-
const updatedText = 'Updated content';
|
|
83
|
-
act(() => {
|
|
84
|
-
result.current.updateItem(itemId, { text: updatedText });
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
expect(result.current.history).toHaveLength(1);
|
|
88
|
-
expect(result.current.history[0]).toEqual({
|
|
89
|
-
...initialItem,
|
|
90
|
-
id: itemId,
|
|
91
|
-
text: updatedText,
|
|
92
|
-
});
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
it('should not change history if updateHistoryItem is called with a nonexistent ID', () => {
|
|
96
|
-
const { result } = renderHook(() => useHistory());
|
|
97
|
-
const timestamp = Date.now();
|
|
98
|
-
const itemData: Omit<HistoryItem, 'id'> = {
|
|
99
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
100
|
-
text: 'Hello',
|
|
101
|
-
};
|
|
102
|
-
|
|
103
|
-
act(() => {
|
|
104
|
-
result.current.addItem(itemData, timestamp);
|
|
105
|
-
});
|
|
106
|
-
|
|
107
|
-
const originalHistory = [...result.current.history]; // Clone before update attempt
|
|
108
|
-
|
|
109
|
-
act(() => {
|
|
110
|
-
result.current.updateItem(99999, { text: 'Should not apply' }); // Nonexistent ID
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
expect(result.current.history).toEqual(originalHistory);
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
it('should clear the history', () => {
|
|
117
|
-
const { result } = renderHook(() => useHistory());
|
|
118
|
-
const timestamp = Date.now();
|
|
119
|
-
const itemData1: Omit<HistoryItem, 'id'> = {
|
|
120
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
121
|
-
text: 'First',
|
|
122
|
-
};
|
|
123
|
-
const itemData2: Omit<HistoryItem, 'id'> = {
|
|
124
|
-
type: 'gemini', // Replaced HistoryItemType.Gemini
|
|
125
|
-
text: 'Second',
|
|
126
|
-
};
|
|
127
|
-
|
|
128
|
-
act(() => {
|
|
129
|
-
result.current.addItem(itemData1, timestamp);
|
|
130
|
-
result.current.addItem(itemData2, timestamp);
|
|
131
|
-
});
|
|
132
|
-
|
|
133
|
-
expect(result.current.history).toHaveLength(2);
|
|
134
|
-
|
|
135
|
-
act(() => {
|
|
136
|
-
result.current.clearItems();
|
|
137
|
-
});
|
|
138
|
-
|
|
139
|
-
expect(result.current.history).toEqual([]);
|
|
140
|
-
});
|
|
141
|
-
|
|
142
|
-
it('should not add consecutive duplicate user messages', () => {
|
|
143
|
-
const { result } = renderHook(() => useHistory());
|
|
144
|
-
const timestamp = Date.now();
|
|
145
|
-
const itemData1: Omit<HistoryItem, 'id'> = {
|
|
146
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
147
|
-
text: 'Duplicate message',
|
|
148
|
-
};
|
|
149
|
-
const itemData2: Omit<HistoryItem, 'id'> = {
|
|
150
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
151
|
-
text: 'Duplicate message',
|
|
152
|
-
};
|
|
153
|
-
const itemData3: Omit<HistoryItem, 'id'> = {
|
|
154
|
-
type: 'gemini', // Replaced HistoryItemType.Gemini
|
|
155
|
-
text: 'Gemini response',
|
|
156
|
-
};
|
|
157
|
-
const itemData4: Omit<HistoryItem, 'id'> = {
|
|
158
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
159
|
-
text: 'Another user message',
|
|
160
|
-
};
|
|
161
|
-
|
|
162
|
-
act(() => {
|
|
163
|
-
result.current.addItem(itemData1, timestamp);
|
|
164
|
-
result.current.addItem(itemData2, timestamp + 1); // Same text, different timestamp
|
|
165
|
-
result.current.addItem(itemData3, timestamp + 2);
|
|
166
|
-
result.current.addItem(itemData4, timestamp + 3);
|
|
167
|
-
});
|
|
168
|
-
|
|
169
|
-
expect(result.current.history).toHaveLength(3);
|
|
170
|
-
expect(result.current.history[0].text).toBe('Duplicate message');
|
|
171
|
-
expect(result.current.history[1].text).toBe('Gemini response');
|
|
172
|
-
expect(result.current.history[2].text).toBe('Another user message');
|
|
173
|
-
});
|
|
174
|
-
|
|
175
|
-
it('should add duplicate user messages if they are not consecutive', () => {
|
|
176
|
-
const { result } = renderHook(() => useHistory());
|
|
177
|
-
const timestamp = Date.now();
|
|
178
|
-
const itemData1: Omit<HistoryItem, 'id'> = {
|
|
179
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
180
|
-
text: 'Message 1',
|
|
181
|
-
};
|
|
182
|
-
const itemData2: Omit<HistoryItem, 'id'> = {
|
|
183
|
-
type: 'gemini', // Replaced HistoryItemType.Gemini
|
|
184
|
-
text: 'Gemini response',
|
|
185
|
-
};
|
|
186
|
-
const itemData3: Omit<HistoryItem, 'id'> = {
|
|
187
|
-
type: 'user', // Replaced HistoryItemType.User
|
|
188
|
-
text: 'Message 1', // Duplicate text, but not consecutive
|
|
189
|
-
};
|
|
190
|
-
|
|
191
|
-
act(() => {
|
|
192
|
-
result.current.addItem(itemData1, timestamp);
|
|
193
|
-
result.current.addItem(itemData2, timestamp + 1);
|
|
194
|
-
result.current.addItem(itemData3, timestamp + 2);
|
|
195
|
-
});
|
|
196
|
-
|
|
197
|
-
expect(result.current.history).toHaveLength(3);
|
|
198
|
-
expect(result.current.history[0].text).toBe('Message 1');
|
|
199
|
-
expect(result.current.history[1].text).toBe('Gemini response');
|
|
200
|
-
expect(result.current.history[2].text).toBe('Message 1');
|
|
201
|
-
});
|
|
202
|
-
});
|
|
@@ -1,111 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { useState, useRef, useCallback } from 'react';
|
|
8
|
-
import { HistoryItem } from '../types.js';
|
|
9
|
-
|
|
10
|
-
// Type for the updater function passed to updateHistoryItem
|
|
11
|
-
type HistoryItemUpdater = (
|
|
12
|
-
prevItem: HistoryItem,
|
|
13
|
-
) => Partial<Omit<HistoryItem, 'id'>>;
|
|
14
|
-
|
|
15
|
-
export interface UseHistoryManagerReturn {
|
|
16
|
-
history: HistoryItem[];
|
|
17
|
-
addItem: (itemData: Omit<HistoryItem, 'id'>, baseTimestamp: number) => number; // Returns the generated ID
|
|
18
|
-
updateItem: (
|
|
19
|
-
id: number,
|
|
20
|
-
updates: Partial<Omit<HistoryItem, 'id'>> | HistoryItemUpdater,
|
|
21
|
-
) => void;
|
|
22
|
-
clearItems: () => void;
|
|
23
|
-
loadHistory: (newHistory: HistoryItem[]) => void;
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
/**
|
|
27
|
-
* Custom hook to manage the chat history state.
|
|
28
|
-
*
|
|
29
|
-
* Encapsulates the history array, message ID generation, adding items,
|
|
30
|
-
* updating items, and clearing the history.
|
|
31
|
-
*/
|
|
32
|
-
export function useHistory(): UseHistoryManagerReturn {
|
|
33
|
-
const [history, setHistory] = useState<HistoryItem[]>([]);
|
|
34
|
-
const messageIdCounterRef = useRef(0);
|
|
35
|
-
|
|
36
|
-
// Generates a unique message ID based on a timestamp and a counter.
|
|
37
|
-
const getNextMessageId = useCallback((baseTimestamp: number): number => {
|
|
38
|
-
messageIdCounterRef.current += 1;
|
|
39
|
-
return baseTimestamp + messageIdCounterRef.current;
|
|
40
|
-
}, []);
|
|
41
|
-
|
|
42
|
-
const loadHistory = useCallback((newHistory: HistoryItem[]) => {
|
|
43
|
-
setHistory(newHistory);
|
|
44
|
-
}, []);
|
|
45
|
-
|
|
46
|
-
// Adds a new item to the history state with a unique ID.
|
|
47
|
-
const addItem = useCallback(
|
|
48
|
-
(itemData: Omit<HistoryItem, 'id'>, baseTimestamp: number): number => {
|
|
49
|
-
const id = getNextMessageId(baseTimestamp);
|
|
50
|
-
const newItem: HistoryItem = { ...itemData, id } as HistoryItem;
|
|
51
|
-
|
|
52
|
-
setHistory((prevHistory) => {
|
|
53
|
-
if (prevHistory.length > 0) {
|
|
54
|
-
const lastItem = prevHistory[prevHistory.length - 1];
|
|
55
|
-
// Prevent adding duplicate consecutive user messages
|
|
56
|
-
if (
|
|
57
|
-
lastItem.type === 'user' &&
|
|
58
|
-
newItem.type === 'user' &&
|
|
59
|
-
lastItem.text === newItem.text
|
|
60
|
-
) {
|
|
61
|
-
return prevHistory; // Don't add the duplicate
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
return [...prevHistory, newItem];
|
|
65
|
-
});
|
|
66
|
-
return id; // Return the generated ID (even if not added, to keep signature)
|
|
67
|
-
},
|
|
68
|
-
[getNextMessageId],
|
|
69
|
-
);
|
|
70
|
-
|
|
71
|
-
/**
|
|
72
|
-
* Updates an existing history item identified by its ID.
|
|
73
|
-
* @deprecated Prefer not to update history item directly as we are currently
|
|
74
|
-
* rendering all history items in <Static /> for performance reasons. Only use
|
|
75
|
-
* if ABSOLUTELY NECESSARY
|
|
76
|
-
*/
|
|
77
|
-
//
|
|
78
|
-
const updateItem = useCallback(
|
|
79
|
-
(
|
|
80
|
-
id: number,
|
|
81
|
-
updates: Partial<Omit<HistoryItem, 'id'>> | HistoryItemUpdater,
|
|
82
|
-
) => {
|
|
83
|
-
setHistory((prevHistory) =>
|
|
84
|
-
prevHistory.map((item) => {
|
|
85
|
-
if (item.id === id) {
|
|
86
|
-
// Apply updates based on whether it's an object or a function
|
|
87
|
-
const newUpdates =
|
|
88
|
-
typeof updates === 'function' ? updates(item) : updates;
|
|
89
|
-
return { ...item, ...newUpdates } as HistoryItem;
|
|
90
|
-
}
|
|
91
|
-
return item;
|
|
92
|
-
}),
|
|
93
|
-
);
|
|
94
|
-
},
|
|
95
|
-
[],
|
|
96
|
-
);
|
|
97
|
-
|
|
98
|
-
// Clears the entire history state and resets the ID counter.
|
|
99
|
-
const clearItems = useCallback(() => {
|
|
100
|
-
setHistory([]);
|
|
101
|
-
messageIdCounterRef.current = 0;
|
|
102
|
-
}, []);
|
|
103
|
-
|
|
104
|
-
return {
|
|
105
|
-
history,
|
|
106
|
-
addItem,
|
|
107
|
-
updateItem,
|
|
108
|
-
clearItems,
|
|
109
|
-
loadHistory,
|
|
110
|
-
};
|
|
111
|
-
}
|