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,238 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import os from 'node:os';
|
|
8
|
-
import path from 'node:path';
|
|
9
|
-
import fs from 'node:fs/promises';
|
|
10
|
-
|
|
11
|
-
import { vi, describe, expect, it, afterEach, beforeEach } from 'vitest';
|
|
12
|
-
import * as gitUtils from '../../utils/gitUtils.js';
|
|
13
|
-
import {
|
|
14
|
-
setupGithubCommand,
|
|
15
|
-
updateGitignore,
|
|
16
|
-
GITHUB_WORKFLOW_PATHS,
|
|
17
|
-
} from './setupGithubCommand.js';
|
|
18
|
-
import { CommandContext, ToolActionReturn } from './types.js';
|
|
19
|
-
import * as commandUtils from '../utils/commandUtils.js';
|
|
20
|
-
|
|
21
|
-
vi.mock('child_process');
|
|
22
|
-
|
|
23
|
-
// Mock fetch globally
|
|
24
|
-
global.fetch = vi.fn();
|
|
25
|
-
|
|
26
|
-
vi.mock('../../utils/gitUtils.js', () => ({
|
|
27
|
-
isGitHubRepository: vi.fn(),
|
|
28
|
-
getGitRepoRoot: vi.fn(),
|
|
29
|
-
getLatestGitHubRelease: vi.fn(),
|
|
30
|
-
getGitHubRepoInfo: vi.fn(),
|
|
31
|
-
}));
|
|
32
|
-
|
|
33
|
-
vi.mock('../utils/commandUtils.js', () => ({
|
|
34
|
-
getUrlOpenCommand: vi.fn(),
|
|
35
|
-
}));
|
|
36
|
-
|
|
37
|
-
describe('setupGithubCommand', async () => {
|
|
38
|
-
let scratchDir = '';
|
|
39
|
-
|
|
40
|
-
beforeEach(async () => {
|
|
41
|
-
vi.resetAllMocks();
|
|
42
|
-
scratchDir = await fs.mkdtemp(
|
|
43
|
-
path.join(os.tmpdir(), 'setup-github-command-'),
|
|
44
|
-
);
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
afterEach(async () => {
|
|
48
|
-
vi.restoreAllMocks();
|
|
49
|
-
if (scratchDir) await fs.rm(scratchDir, { recursive: true });
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
it('returns a tool action to download github workflows and handles paths', async () => {
|
|
53
|
-
const fakeRepoOwner = 'fake';
|
|
54
|
-
const fakeRepoName = 'repo';
|
|
55
|
-
const fakeRepoRoot = scratchDir;
|
|
56
|
-
const fakeReleaseVersion = 'v1.2.3';
|
|
57
|
-
|
|
58
|
-
const workflows = GITHUB_WORKFLOW_PATHS.map((p) => path.basename(p));
|
|
59
|
-
for (const workflow of workflows) {
|
|
60
|
-
vi.mocked(global.fetch).mockReturnValueOnce(
|
|
61
|
-
Promise.resolve(new Response(workflow)),
|
|
62
|
-
);
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
vi.mocked(gitUtils.isGitHubRepository).mockReturnValueOnce(true);
|
|
66
|
-
vi.mocked(gitUtils.getGitRepoRoot).mockReturnValueOnce(fakeRepoRoot);
|
|
67
|
-
vi.mocked(gitUtils.getLatestGitHubRelease).mockResolvedValueOnce(
|
|
68
|
-
fakeReleaseVersion,
|
|
69
|
-
);
|
|
70
|
-
vi.mocked(gitUtils.getGitHubRepoInfo).mockReturnValue({
|
|
71
|
-
owner: fakeRepoOwner,
|
|
72
|
-
repo: fakeRepoName,
|
|
73
|
-
});
|
|
74
|
-
vi.mocked(commandUtils.getUrlOpenCommand).mockReturnValueOnce(
|
|
75
|
-
'fakeOpenCommand',
|
|
76
|
-
);
|
|
77
|
-
|
|
78
|
-
const result = (await setupGithubCommand.action?.(
|
|
79
|
-
{} as CommandContext,
|
|
80
|
-
'',
|
|
81
|
-
)) as ToolActionReturn;
|
|
82
|
-
|
|
83
|
-
const { command } = result.toolArgs;
|
|
84
|
-
|
|
85
|
-
const expectedSubstrings = [
|
|
86
|
-
`set -eEuo pipefail`,
|
|
87
|
-
`fakeOpenCommand "https://github.com/google-github-actions/run-gemini-cli`,
|
|
88
|
-
];
|
|
89
|
-
|
|
90
|
-
for (const substring of expectedSubstrings) {
|
|
91
|
-
expect(command).toContain(substring);
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
for (const workflow of workflows) {
|
|
95
|
-
const workflowFile = path.join(
|
|
96
|
-
scratchDir,
|
|
97
|
-
'.github',
|
|
98
|
-
'workflows',
|
|
99
|
-
workflow,
|
|
100
|
-
);
|
|
101
|
-
const contents = await fs.readFile(workflowFile, 'utf8');
|
|
102
|
-
expect(contents).toContain(workflow);
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
// Verify that .gitignore was created with the expected entries
|
|
106
|
-
const gitignorePath = path.join(scratchDir, '.gitignore');
|
|
107
|
-
const gitignoreExists = await fs
|
|
108
|
-
.access(gitignorePath)
|
|
109
|
-
.then(() => true)
|
|
110
|
-
.catch(() => false);
|
|
111
|
-
expect(gitignoreExists).toBe(true);
|
|
112
|
-
|
|
113
|
-
if (gitignoreExists) {
|
|
114
|
-
const gitignoreContent = await fs.readFile(gitignorePath, 'utf8');
|
|
115
|
-
expect(gitignoreContent).toContain('.gemini/');
|
|
116
|
-
expect(gitignoreContent).toContain('gha-creds-*.json');
|
|
117
|
-
}
|
|
118
|
-
});
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
describe('updateGitignore', () => {
|
|
122
|
-
let scratchDir = '';
|
|
123
|
-
|
|
124
|
-
beforeEach(async () => {
|
|
125
|
-
scratchDir = await fs.mkdtemp(path.join(os.tmpdir(), 'update-gitignore-'));
|
|
126
|
-
});
|
|
127
|
-
|
|
128
|
-
afterEach(async () => {
|
|
129
|
-
if (scratchDir) await fs.rm(scratchDir, { recursive: true });
|
|
130
|
-
});
|
|
131
|
-
|
|
132
|
-
it('creates a new .gitignore file when none exists', async () => {
|
|
133
|
-
await updateGitignore(scratchDir);
|
|
134
|
-
|
|
135
|
-
const gitignorePath = path.join(scratchDir, '.gitignore');
|
|
136
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
137
|
-
|
|
138
|
-
expect(content).toBe('.gemini/\ngha-creds-*.json\n');
|
|
139
|
-
});
|
|
140
|
-
|
|
141
|
-
it('appends entries to existing .gitignore file', async () => {
|
|
142
|
-
const gitignorePath = path.join(scratchDir, '.gitignore');
|
|
143
|
-
const existingContent = '# Existing content\nnode_modules/\n';
|
|
144
|
-
await fs.writeFile(gitignorePath, existingContent);
|
|
145
|
-
|
|
146
|
-
await updateGitignore(scratchDir);
|
|
147
|
-
|
|
148
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
149
|
-
|
|
150
|
-
expect(content).toBe(
|
|
151
|
-
'# Existing content\nnode_modules/\n\n.gemini/\ngha-creds-*.json\n',
|
|
152
|
-
);
|
|
153
|
-
});
|
|
154
|
-
|
|
155
|
-
it('does not add duplicate entries', async () => {
|
|
156
|
-
const gitignorePath = path.join(scratchDir, '.gitignore');
|
|
157
|
-
const existingContent = '.gemini/\nsome-other-file\ngha-creds-*.json\n';
|
|
158
|
-
await fs.writeFile(gitignorePath, existingContent);
|
|
159
|
-
|
|
160
|
-
await updateGitignore(scratchDir);
|
|
161
|
-
|
|
162
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
163
|
-
|
|
164
|
-
expect(content).toBe(existingContent);
|
|
165
|
-
});
|
|
166
|
-
|
|
167
|
-
it('adds only missing entries when some already exist', async () => {
|
|
168
|
-
const gitignorePath = path.join(scratchDir, '.gitignore');
|
|
169
|
-
const existingContent = '.gemini/\nsome-other-file\n';
|
|
170
|
-
await fs.writeFile(gitignorePath, existingContent);
|
|
171
|
-
|
|
172
|
-
await updateGitignore(scratchDir);
|
|
173
|
-
|
|
174
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
175
|
-
|
|
176
|
-
// Should add only the missing gha-creds-*.json entry
|
|
177
|
-
expect(content).toBe('.gemini/\nsome-other-file\n\ngha-creds-*.json\n');
|
|
178
|
-
expect(content).toContain('gha-creds-*.json');
|
|
179
|
-
// Should not duplicate .gemini/ entry
|
|
180
|
-
expect((content.match(/\.gemini\//g) || []).length).toBe(1);
|
|
181
|
-
});
|
|
182
|
-
|
|
183
|
-
it('does not get confused by entries in comments or as substrings', async () => {
|
|
184
|
-
const gitignorePath = path.join(scratchDir, '.gitignore');
|
|
185
|
-
const existingContent = [
|
|
186
|
-
'# This is a comment mentioning .gemini/ folder',
|
|
187
|
-
'my-app.gemini/config',
|
|
188
|
-
'# Another comment with gha-creds-*.json pattern',
|
|
189
|
-
'some-other-gha-creds-file.json',
|
|
190
|
-
'',
|
|
191
|
-
].join('\n');
|
|
192
|
-
await fs.writeFile(gitignorePath, existingContent);
|
|
193
|
-
|
|
194
|
-
await updateGitignore(scratchDir);
|
|
195
|
-
|
|
196
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
197
|
-
|
|
198
|
-
// Should add both entries since they don't actually exist as gitignore rules
|
|
199
|
-
expect(content).toContain('.gemini/');
|
|
200
|
-
expect(content).toContain('gha-creds-*.json');
|
|
201
|
-
|
|
202
|
-
// Verify the entries were added (not just mentioned in comments)
|
|
203
|
-
const lines = content
|
|
204
|
-
.split('\n')
|
|
205
|
-
.map((line) => line.split('#')[0].trim())
|
|
206
|
-
.filter((line) => line);
|
|
207
|
-
expect(lines).toContain('.gemini/');
|
|
208
|
-
expect(lines).toContain('gha-creds-*.json');
|
|
209
|
-
expect(lines).toContain('my-app.gemini/config');
|
|
210
|
-
expect(lines).toContain('some-other-gha-creds-file.json');
|
|
211
|
-
});
|
|
212
|
-
|
|
213
|
-
it('handles file system errors gracefully', async () => {
|
|
214
|
-
// Try to update gitignore in a non-existent directory
|
|
215
|
-
const nonExistentDir = path.join(scratchDir, 'non-existent');
|
|
216
|
-
|
|
217
|
-
// This should not throw an error
|
|
218
|
-
await expect(updateGitignore(nonExistentDir)).resolves.toBeUndefined();
|
|
219
|
-
});
|
|
220
|
-
|
|
221
|
-
it('handles permission errors gracefully', async () => {
|
|
222
|
-
const consoleSpy = vi.spyOn(console, 'debug').mockImplementation(() => {});
|
|
223
|
-
|
|
224
|
-
const fsModule = await import('node:fs');
|
|
225
|
-
const writeFileSpy = vi
|
|
226
|
-
.spyOn(fsModule.promises, 'writeFile')
|
|
227
|
-
.mockRejectedValue(new Error('Permission denied'));
|
|
228
|
-
|
|
229
|
-
await expect(updateGitignore(scratchDir)).resolves.toBeUndefined();
|
|
230
|
-
expect(consoleSpy).toHaveBeenCalledWith(
|
|
231
|
-
'Failed to update .gitignore:',
|
|
232
|
-
expect.any(Error),
|
|
233
|
-
);
|
|
234
|
-
|
|
235
|
-
writeFileSpy.mockRestore();
|
|
236
|
-
consoleSpy.mockRestore();
|
|
237
|
-
});
|
|
238
|
-
});
|
|
@@ -1,212 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import path from 'node:path';
|
|
8
|
-
import * as fs from 'node:fs';
|
|
9
|
-
import { Writable } from 'node:stream';
|
|
10
|
-
import { ProxyAgent } from 'undici';
|
|
11
|
-
|
|
12
|
-
import { CommandContext } from '../../ui/commands/types.js';
|
|
13
|
-
import {
|
|
14
|
-
getGitRepoRoot,
|
|
15
|
-
getLatestGitHubRelease,
|
|
16
|
-
isGitHubRepository,
|
|
17
|
-
getGitHubRepoInfo,
|
|
18
|
-
} from '../../utils/gitUtils.js';
|
|
19
|
-
|
|
20
|
-
import {
|
|
21
|
-
CommandKind,
|
|
22
|
-
SlashCommand,
|
|
23
|
-
SlashCommandActionReturn,
|
|
24
|
-
} from './types.js';
|
|
25
|
-
import { getUrlOpenCommand } from '../../ui/utils/commandUtils.js';
|
|
26
|
-
|
|
27
|
-
export const GITHUB_WORKFLOW_PATHS = [
|
|
28
|
-
'gemini-dispatch/gemini-dispatch.yml',
|
|
29
|
-
'gemini-assistant/gemini-invoke.yml',
|
|
30
|
-
'issue-triage/gemini-triage.yml',
|
|
31
|
-
'issue-triage/gemini-scheduled-triage.yml',
|
|
32
|
-
'pr-review/gemini-review.yml',
|
|
33
|
-
];
|
|
34
|
-
|
|
35
|
-
// Generate OS-specific commands to open the GitHub pages needed for setup.
|
|
36
|
-
function getOpenUrlsCommands(readmeUrl: string): string[] {
|
|
37
|
-
// Determine the OS-specific command to open URLs, ex: 'open', 'xdg-open', etc
|
|
38
|
-
const openCmd = getUrlOpenCommand();
|
|
39
|
-
|
|
40
|
-
// Build a list of URLs to open
|
|
41
|
-
const urlsToOpen = [readmeUrl];
|
|
42
|
-
|
|
43
|
-
const repoInfo = getGitHubRepoInfo();
|
|
44
|
-
if (repoInfo) {
|
|
45
|
-
urlsToOpen.push(
|
|
46
|
-
`https://github.com/${repoInfo.owner}/${repoInfo.repo}/settings/secrets/actions`,
|
|
47
|
-
);
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
// Create and join the individual commands
|
|
51
|
-
const commands = urlsToOpen.map((url) => `${openCmd} "${url}"`);
|
|
52
|
-
return commands;
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
// Add Gemini CLI specific entries to .gitignore file
|
|
56
|
-
export async function updateGitignore(gitRepoRoot: string): Promise<void> {
|
|
57
|
-
const gitignoreEntries = ['.gemini/', 'gha-creds-*.json'];
|
|
58
|
-
|
|
59
|
-
const gitignorePath = path.join(gitRepoRoot, '.gitignore');
|
|
60
|
-
try {
|
|
61
|
-
// Check if .gitignore exists and read its content
|
|
62
|
-
let existingContent = '';
|
|
63
|
-
let fileExists = true;
|
|
64
|
-
try {
|
|
65
|
-
existingContent = await fs.promises.readFile(gitignorePath, 'utf8');
|
|
66
|
-
} catch (_error) {
|
|
67
|
-
// File doesn't exist
|
|
68
|
-
fileExists = false;
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
if (!fileExists) {
|
|
72
|
-
// Create new .gitignore file with the entries
|
|
73
|
-
const contentToWrite = gitignoreEntries.join('\n') + '\n';
|
|
74
|
-
await fs.promises.writeFile(gitignorePath, contentToWrite);
|
|
75
|
-
} else {
|
|
76
|
-
// Check which entries are missing
|
|
77
|
-
const missingEntries = gitignoreEntries.filter(
|
|
78
|
-
(entry) =>
|
|
79
|
-
!existingContent
|
|
80
|
-
.split(/\r?\n/)
|
|
81
|
-
.some((line) => line.split('#')[0].trim() === entry),
|
|
82
|
-
);
|
|
83
|
-
|
|
84
|
-
if (missingEntries.length > 0) {
|
|
85
|
-
const contentToAdd = '\n' + missingEntries.join('\n') + '\n';
|
|
86
|
-
await fs.promises.appendFile(gitignorePath, contentToAdd);
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
} catch (error) {
|
|
90
|
-
console.debug('Failed to update .gitignore:', error);
|
|
91
|
-
// Continue without failing the whole command
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
export const setupGithubCommand: SlashCommand = {
|
|
96
|
-
name: 'setup-github',
|
|
97
|
-
description: 'Set up GitHub Actions',
|
|
98
|
-
kind: CommandKind.BUILT_IN,
|
|
99
|
-
action: async (
|
|
100
|
-
context: CommandContext,
|
|
101
|
-
): Promise<SlashCommandActionReturn> => {
|
|
102
|
-
const abortController = new AbortController();
|
|
103
|
-
|
|
104
|
-
if (!isGitHubRepository()) {
|
|
105
|
-
throw new Error(
|
|
106
|
-
'Unable to determine the GitHub repository. /setup-github must be run from a git repository.',
|
|
107
|
-
);
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
// Find the root directory of the repo
|
|
111
|
-
let gitRepoRoot: string;
|
|
112
|
-
try {
|
|
113
|
-
gitRepoRoot = getGitRepoRoot();
|
|
114
|
-
} catch (_error) {
|
|
115
|
-
console.debug(`Failed to get git repo root:`, _error);
|
|
116
|
-
throw new Error(
|
|
117
|
-
'Unable to determine the GitHub repository. /setup-github must be run from a git repository.',
|
|
118
|
-
);
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
// Get the latest release tag from GitHub
|
|
122
|
-
const proxy = context?.services?.config?.getProxy();
|
|
123
|
-
const releaseTag = await getLatestGitHubRelease(proxy);
|
|
124
|
-
const readmeUrl = `https://github.com/google-github-actions/run-gemini-cli/blob/${releaseTag}/README.md#quick-start`;
|
|
125
|
-
|
|
126
|
-
// Create the .github/workflows directory to download the files into
|
|
127
|
-
const githubWorkflowsDir = path.join(gitRepoRoot, '.github', 'workflows');
|
|
128
|
-
try {
|
|
129
|
-
await fs.promises.mkdir(githubWorkflowsDir, { recursive: true });
|
|
130
|
-
} catch (_error) {
|
|
131
|
-
console.debug(
|
|
132
|
-
`Failed to create ${githubWorkflowsDir} directory:`,
|
|
133
|
-
_error,
|
|
134
|
-
);
|
|
135
|
-
throw new Error(
|
|
136
|
-
`Unable to create ${githubWorkflowsDir} directory. Do you have file permissions in the current directory?`,
|
|
137
|
-
);
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
// Download each workflow in parallel - there aren't enough files to warrant
|
|
141
|
-
// a full workerpool model here.
|
|
142
|
-
const downloads = [];
|
|
143
|
-
for (const workflow of GITHUB_WORKFLOW_PATHS) {
|
|
144
|
-
downloads.push(
|
|
145
|
-
(async () => {
|
|
146
|
-
const endpoint = `https://raw.githubusercontent.com/google-github-actions/run-gemini-cli/refs/tags/${releaseTag}/examples/workflows/${workflow}`;
|
|
147
|
-
const response = await fetch(endpoint, {
|
|
148
|
-
method: 'GET',
|
|
149
|
-
dispatcher: proxy ? new ProxyAgent(proxy) : undefined,
|
|
150
|
-
signal: AbortSignal.any([
|
|
151
|
-
AbortSignal.timeout(30_000),
|
|
152
|
-
abortController.signal,
|
|
153
|
-
]),
|
|
154
|
-
} as RequestInit);
|
|
155
|
-
|
|
156
|
-
if (!response.ok) {
|
|
157
|
-
throw new Error(
|
|
158
|
-
`Invalid response code downloading ${endpoint}: ${response.status} - ${response.statusText}`,
|
|
159
|
-
);
|
|
160
|
-
}
|
|
161
|
-
const body = response.body;
|
|
162
|
-
if (!body) {
|
|
163
|
-
throw new Error(
|
|
164
|
-
`Empty body while downloading ${endpoint}: ${response.status} - ${response.statusText}`,
|
|
165
|
-
);
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
const destination = path.resolve(
|
|
169
|
-
githubWorkflowsDir,
|
|
170
|
-
path.basename(workflow),
|
|
171
|
-
);
|
|
172
|
-
|
|
173
|
-
const fileStream = fs.createWriteStream(destination, {
|
|
174
|
-
mode: 0o644, // -rw-r--r--, user(rw), group(r), other(r)
|
|
175
|
-
flags: 'w', // write and overwrite
|
|
176
|
-
flush: true,
|
|
177
|
-
});
|
|
178
|
-
|
|
179
|
-
await body.pipeTo(Writable.toWeb(fileStream));
|
|
180
|
-
})(),
|
|
181
|
-
);
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
// Wait for all downloads to complete
|
|
185
|
-
await Promise.all(downloads).finally(() => {
|
|
186
|
-
// Stop existing downloads
|
|
187
|
-
abortController.abort();
|
|
188
|
-
});
|
|
189
|
-
|
|
190
|
-
// Add entries to .gitignore file
|
|
191
|
-
await updateGitignore(gitRepoRoot);
|
|
192
|
-
|
|
193
|
-
// Print out a message
|
|
194
|
-
const commands = [];
|
|
195
|
-
commands.push('set -eEuo pipefail');
|
|
196
|
-
commands.push(
|
|
197
|
-
`echo "Successfully downloaded ${GITHUB_WORKFLOW_PATHS.length} workflows and updated .gitignore. Follow the steps in ${readmeUrl} (skipping the /setup-github step) to complete setup."`,
|
|
198
|
-
);
|
|
199
|
-
commands.push(...getOpenUrlsCommands(readmeUrl));
|
|
200
|
-
|
|
201
|
-
const command = `(${commands.join(' && ')})`;
|
|
202
|
-
return {
|
|
203
|
-
type: 'tool',
|
|
204
|
-
toolName: 'run_shell_command',
|
|
205
|
-
toolArgs: {
|
|
206
|
-
description:
|
|
207
|
-
'Setting up GitHub Actions to triage issues and review PRs with Gemini.',
|
|
208
|
-
command,
|
|
209
|
-
},
|
|
210
|
-
};
|
|
211
|
-
},
|
|
212
|
-
};
|
|
@@ -1,175 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { CommandKind, SlashCommand } from './types.js';
|
|
8
|
-
import { MessageType } from '../types.js';
|
|
9
|
-
|
|
10
|
-
// Simple text cleaning for speech synthesis
|
|
11
|
-
function cleanTextForSpeech(text: string): string {
|
|
12
|
-
let cleaned = text.trim();
|
|
13
|
-
|
|
14
|
-
// Conservative approach - only remove obvious machine-generated garbage
|
|
15
|
-
cleaned = cleaned.replace(/\b[a-f0-9]{16,}\b/gi, 'hash'); // Only very long hashes
|
|
16
|
-
cleaned = cleaned.replace(/\b[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}\b/gi, 'UUID'); // UUIDs
|
|
17
|
-
|
|
18
|
-
// Clean up markdown formatting for speech
|
|
19
|
-
cleaned = cleaned.replace(/```[\s\S]*?```/g, 'code block');
|
|
20
|
-
cleaned = cleaned.replace(/`([^`]+)`/g, '$1');
|
|
21
|
-
cleaned = cleaned.replace(/\*\*([^*]+)\*\*/g, '$1');
|
|
22
|
-
cleaned = cleaned.replace(/\*([^*]+)\*/g, '$1');
|
|
23
|
-
|
|
24
|
-
// Clean excessive whitespace
|
|
25
|
-
cleaned = cleaned.replace(/\s+/g, ' ').trim();
|
|
26
|
-
|
|
27
|
-
return cleaned;
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
interface TTSProgress {
|
|
31
|
-
setPhase: (phase: 'cleaning' | 'sending' | 'processing' | 'speaking' | 'completed' | 'error') => void;
|
|
32
|
-
setError: (error: string) => void;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
async function executeSpeech(text: string, voice?: string, progress?: TTSProgress): Promise<void> {
|
|
36
|
-
try {
|
|
37
|
-
// Phase 1: Text cleaning
|
|
38
|
-
progress?.setPhase('cleaning');
|
|
39
|
-
await new Promise(resolve => setTimeout(resolve, 300)); // Brief delay to show animation
|
|
40
|
-
const cleanedText = cleanTextForSpeech(text);
|
|
41
|
-
|
|
42
|
-
// Phase 2: Sending to server
|
|
43
|
-
progress?.setPhase('sending');
|
|
44
|
-
await new Promise(resolve => setTimeout(resolve, 200));
|
|
45
|
-
|
|
46
|
-
const response = await fetch('http://192.168.1.5:11450/api/v1/synthesize', {
|
|
47
|
-
method: 'POST',
|
|
48
|
-
headers: {
|
|
49
|
-
'Content-Type': 'application/json',
|
|
50
|
-
},
|
|
51
|
-
body: JSON.stringify({
|
|
52
|
-
text: cleanedText,
|
|
53
|
-
voice: voice || 'Bella',
|
|
54
|
-
format: 'wav',
|
|
55
|
-
output_mode: 'computer'
|
|
56
|
-
}),
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
if (!response.ok) {
|
|
60
|
-
progress?.setError(`Server returned ${response.status}: ${response.statusText}`);
|
|
61
|
-
progress?.setPhase('error');
|
|
62
|
-
return;
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
// Phase 3: Processing response
|
|
66
|
-
progress?.setPhase('processing');
|
|
67
|
-
await response.json(); // Process the response
|
|
68
|
-
await new Promise(resolve => setTimeout(resolve, 400));
|
|
69
|
-
|
|
70
|
-
// Phase 4: Speaking (simulate audio duration)
|
|
71
|
-
progress?.setPhase('speaking');
|
|
72
|
-
const estimatedDuration = Math.max(1000, Math.min(text.length * 50, 5000));
|
|
73
|
-
await new Promise(resolve => setTimeout(resolve, estimatedDuration));
|
|
74
|
-
|
|
75
|
-
// Phase 5: Completed
|
|
76
|
-
progress?.setPhase('completed');
|
|
77
|
-
|
|
78
|
-
} catch (error) {
|
|
79
|
-
progress?.setError(error instanceof Error ? error.message : 'Unknown error');
|
|
80
|
-
progress?.setPhase('error');
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
// Voice personality emojis
|
|
85
|
-
const VOICE_EMOJIS: Record<string, string> = {
|
|
86
|
-
'Bella': '👩💼',
|
|
87
|
-
'Emma': '👩🎨',
|
|
88
|
-
'George': '👨🔧',
|
|
89
|
-
'Lewis': '👨💻',
|
|
90
|
-
'Sarah': '👩🏫',
|
|
91
|
-
'Michael': '👨⚖️'
|
|
92
|
-
};
|
|
93
|
-
|
|
94
|
-
export const speakCommand: SlashCommand = {
|
|
95
|
-
name: 'speak',
|
|
96
|
-
description: 'speak text using text-to-speech',
|
|
97
|
-
kind: CommandKind.BUILT_IN,
|
|
98
|
-
action: async (context, args) => {
|
|
99
|
-
const text = args.trim();
|
|
100
|
-
|
|
101
|
-
// Basic validation
|
|
102
|
-
if (!text) {
|
|
103
|
-
context.ui.addItem(
|
|
104
|
-
{
|
|
105
|
-
type: MessageType.ERROR,
|
|
106
|
-
text: '❌ Usage: /speak "your message here"',
|
|
107
|
-
},
|
|
108
|
-
Date.now()
|
|
109
|
-
);
|
|
110
|
-
return;
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
if (text.length > 1000) {
|
|
114
|
-
context.ui.addItem(
|
|
115
|
-
{
|
|
116
|
-
type: MessageType.INFO,
|
|
117
|
-
text: '⚠️ Text too long (max 1000 chars), will be truncated',
|
|
118
|
-
},
|
|
119
|
-
Date.now()
|
|
120
|
-
);
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
const voice = context.session.ttsVoice || 'Bella';
|
|
124
|
-
const voiceEmoji = VOICE_EMOJIS[voice] || '🗣️';
|
|
125
|
-
const truncatedText = text.length > 1000 ? text.substring(0, 1000) : text;
|
|
126
|
-
|
|
127
|
-
// Create progress tracker
|
|
128
|
-
let errorMessage = '';
|
|
129
|
-
|
|
130
|
-
// Simple progress indicators - just loading bar and icons
|
|
131
|
-
const updateProgress = (phase: string, error?: string) => {
|
|
132
|
-
const phaseConfig: Record<string, {icon: string, bar: string}> = {
|
|
133
|
-
cleaning: { icon: '🧹', bar: '▓░░░░' },
|
|
134
|
-
sending: { icon: '📡', bar: '▓▓░░░' },
|
|
135
|
-
processing: { icon: '⚙️', bar: '▓▓▓░░' },
|
|
136
|
-
speaking: { icon: '🎵', bar: '▓▓▓▓░' },
|
|
137
|
-
completed: { icon: '✅', bar: '▓▓▓▓▓' },
|
|
138
|
-
error: { icon: '❌', bar: '░░░░░' }
|
|
139
|
-
};
|
|
140
|
-
|
|
141
|
-
const config = phaseConfig[phase] || phaseConfig['cleaning'];
|
|
142
|
-
let statusText = `${config.icon} ${config.bar} ${voiceEmoji}`;
|
|
143
|
-
|
|
144
|
-
if (error && phase === 'error') {
|
|
145
|
-
statusText += ` ${error}`;
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
context.ui.addItem(
|
|
149
|
-
{
|
|
150
|
-
type: phase === 'error' ? MessageType.ERROR : MessageType.INFO,
|
|
151
|
-
text: statusText,
|
|
152
|
-
},
|
|
153
|
-
Date.now()
|
|
154
|
-
);
|
|
155
|
-
};
|
|
156
|
-
|
|
157
|
-
// Initial progress display
|
|
158
|
-
updateProgress('cleaning');
|
|
159
|
-
|
|
160
|
-
// Execute speech with rich progress tracking
|
|
161
|
-
const progress: TTSProgress = {
|
|
162
|
-
setPhase: (phase) => {
|
|
163
|
-
updateProgress(phase, errorMessage);
|
|
164
|
-
},
|
|
165
|
-
setError: (error) => {
|
|
166
|
-
errorMessage = error;
|
|
167
|
-
}
|
|
168
|
-
};
|
|
169
|
-
|
|
170
|
-
// Execute in background with progress updates
|
|
171
|
-
executeSpeech(truncatedText, voice, progress).catch(() => {
|
|
172
|
-
// Error handling is done through progress callbacks
|
|
173
|
-
});
|
|
174
|
-
},
|
|
175
|
-
};
|
|
@@ -1,78 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { vi, describe, it, expect, beforeEach } from 'vitest';
|
|
8
|
-
import { statsCommand } from './statsCommand.js';
|
|
9
|
-
import { type CommandContext } from './types.js';
|
|
10
|
-
import { createMockCommandContext } from '../../test-utils/mockCommandContext.js';
|
|
11
|
-
import { MessageType } from '../types.js';
|
|
12
|
-
import { formatDuration } from '../utils/formatters.js';
|
|
13
|
-
|
|
14
|
-
describe('statsCommand', () => {
|
|
15
|
-
let mockContext: CommandContext;
|
|
16
|
-
const startTime = new Date('2025-07-14T10:00:00.000Z');
|
|
17
|
-
const endTime = new Date('2025-07-14T10:00:30.000Z');
|
|
18
|
-
|
|
19
|
-
beforeEach(() => {
|
|
20
|
-
vi.useFakeTimers();
|
|
21
|
-
vi.setSystemTime(endTime);
|
|
22
|
-
|
|
23
|
-
// 1. Create the mock context with all default values
|
|
24
|
-
mockContext = createMockCommandContext();
|
|
25
|
-
|
|
26
|
-
// 2. Directly set the property on the created mock context
|
|
27
|
-
mockContext.session.stats.sessionStartTime = startTime;
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
it('should display general session stats when run with no subcommand', () => {
|
|
31
|
-
if (!statsCommand.action) throw new Error('Command has no action');
|
|
32
|
-
|
|
33
|
-
statsCommand.action(mockContext, '');
|
|
34
|
-
|
|
35
|
-
const expectedDuration = formatDuration(
|
|
36
|
-
endTime.getTime() - startTime.getTime(),
|
|
37
|
-
);
|
|
38
|
-
expect(mockContext.ui.addItem).toHaveBeenCalledWith(
|
|
39
|
-
{
|
|
40
|
-
type: MessageType.STATS,
|
|
41
|
-
duration: expectedDuration,
|
|
42
|
-
},
|
|
43
|
-
expect.any(Number),
|
|
44
|
-
);
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
it('should display model stats when using the "model" subcommand', () => {
|
|
48
|
-
const modelSubCommand = statsCommand.subCommands?.find(
|
|
49
|
-
(sc) => sc.name === 'model',
|
|
50
|
-
);
|
|
51
|
-
if (!modelSubCommand?.action) throw new Error('Subcommand has no action');
|
|
52
|
-
|
|
53
|
-
modelSubCommand.action(mockContext, '');
|
|
54
|
-
|
|
55
|
-
expect(mockContext.ui.addItem).toHaveBeenCalledWith(
|
|
56
|
-
{
|
|
57
|
-
type: MessageType.MODEL_STATS,
|
|
58
|
-
},
|
|
59
|
-
expect.any(Number),
|
|
60
|
-
);
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
it('should display tool stats when using the "tools" subcommand', () => {
|
|
64
|
-
const toolsSubCommand = statsCommand.subCommands?.find(
|
|
65
|
-
(sc) => sc.name === 'tools',
|
|
66
|
-
);
|
|
67
|
-
if (!toolsSubCommand?.action) throw new Error('Subcommand has no action');
|
|
68
|
-
|
|
69
|
-
toolsSubCommand.action(mockContext, '');
|
|
70
|
-
|
|
71
|
-
expect(mockContext.ui.addItem).toHaveBeenCalledWith(
|
|
72
|
-
{
|
|
73
|
-
type: MessageType.TOOL_STATS,
|
|
74
|
-
},
|
|
75
|
-
expect.any(Number),
|
|
76
|
-
);
|
|
77
|
-
});
|
|
78
|
-
});
|