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.
Files changed (419) hide show
  1. package/dist/index.js +0 -0
  2. package/dist/package.json +2 -2
  3. package/dist/src/config/auth.js +8 -5
  4. package/dist/src/config/auth.js.map +1 -1
  5. package/dist/src/config/database.d.ts +103 -11
  6. package/dist/src/config/database.js +301 -59
  7. package/dist/src/config/database.js.map +1 -1
  8. package/dist/src/config/databaseBackup.d.ts +114 -0
  9. package/dist/src/config/databaseBackup.js +334 -0
  10. package/dist/src/config/databaseBackup.js.map +1 -0
  11. package/dist/src/config/databaseMigrations.d.ts +63 -0
  12. package/dist/src/config/databaseMigrations.js +379 -0
  13. package/dist/src/config/databaseMigrations.js.map +1 -0
  14. package/dist/src/config/databasePool.d.ts +70 -0
  15. package/dist/src/config/databasePool.js +193 -0
  16. package/dist/src/config/databasePool.js.map +1 -0
  17. package/dist/src/config/queryOptimizer.d.ts +127 -0
  18. package/dist/src/config/queryOptimizer.js +309 -0
  19. package/dist/src/config/queryOptimizer.js.map +1 -0
  20. package/dist/src/utils/sandbox.js +2 -8
  21. package/dist/src/utils/sandbox.js.map +1 -1
  22. package/dist/src/validateNonInterActiveAuth.js +3 -7
  23. package/dist/src/validateNonInterActiveAuth.js.map +1 -1
  24. package/dist/tsconfig.tsbuildinfo +1 -1
  25. package/package.json +2 -2
  26. package/dist/commands/mcp/add.test.ts +0 -122
  27. package/dist/commands/mcp/add.ts +0 -222
  28. package/dist/commands/mcp/list.test.ts +0 -154
  29. package/dist/commands/mcp/list.ts +0 -139
  30. package/dist/commands/mcp/remove.test.ts +0 -69
  31. package/dist/commands/mcp/remove.ts +0 -60
  32. package/dist/commands/mcp.test.ts +0 -55
  33. package/dist/commands/mcp.ts +0 -27
  34. package/dist/config/apiValidation.test.ts +0 -118
  35. package/dist/config/auth.test.ts +0 -79
  36. package/dist/config/auth.ts +0 -100
  37. package/dist/config/config.integration.test.ts +0 -407
  38. package/dist/config/config.test.ts +0 -1952
  39. package/dist/config/config.ts +0 -690
  40. package/dist/config/database.test.ts +0 -96
  41. package/dist/config/database.ts +0 -824
  42. package/dist/config/extension.test.ts +0 -236
  43. package/dist/config/extension.ts +0 -180
  44. package/dist/config/keyBindings.test.ts +0 -62
  45. package/dist/config/keyBindings.ts +0 -184
  46. package/dist/config/modelManager.ts +0 -326
  47. package/dist/config/providerManager.ts +0 -244
  48. package/dist/config/providerPersistence.test.ts +0 -377
  49. package/dist/config/providerPersistence.ts +0 -105
  50. package/dist/config/sandboxConfig.ts +0 -107
  51. package/dist/config/settings.test.ts +0 -1424
  52. package/dist/config/settings.ts +0 -517
  53. package/dist/config/settingsSchema.test.ts +0 -252
  54. package/dist/config/settingsSchema.ts +0 -728
  55. package/dist/config/trustedFolders.test.ts +0 -208
  56. package/dist/config/trustedFolders.ts +0 -167
  57. package/dist/gemini.test.tsx +0 -252
  58. package/dist/gemini.tsx +0 -357
  59. package/dist/generated/git-commit.ts +0 -10
  60. package/dist/index.ts +0 -21
  61. package/dist/nonInteractiveCli.test.ts +0 -276
  62. package/dist/nonInteractiveCli.ts +0 -143
  63. package/dist/patches/is-in-ci.ts +0 -17
  64. package/dist/services/BuiltinCommandLoader.test.ts +0 -127
  65. package/dist/services/BuiltinCommandLoader.ts +0 -95
  66. package/dist/services/CommandService.test.ts +0 -352
  67. package/dist/services/CommandService.ts +0 -103
  68. package/dist/services/FileCommandLoader.test.ts +0 -1002
  69. package/dist/services/FileCommandLoader.ts +0 -289
  70. package/dist/services/McpPromptLoader.ts +0 -231
  71. package/dist/services/SearchEngineConfigProvider.ts +0 -100
  72. package/dist/services/prompt-processors/argumentProcessor.test.ts +0 -41
  73. package/dist/services/prompt-processors/argumentProcessor.ts +0 -23
  74. package/dist/services/prompt-processors/shellProcessor.test.ts +0 -709
  75. package/dist/services/prompt-processors/shellProcessor.ts +0 -248
  76. package/dist/services/prompt-processors/types.ts +0 -44
  77. package/dist/services/types.ts +0 -24
  78. package/dist/src/config/apiValidation.test.d.ts +0 -6
  79. package/dist/src/config/apiValidation.test.js +0 -99
  80. package/dist/src/config/apiValidation.test.js.map +0 -1
  81. package/dist/src/config/database.test.d.ts +0 -6
  82. package/dist/src/config/database.test.js +0 -80
  83. package/dist/src/config/database.test.js.map +0 -1
  84. package/dist/src/config/providerManager.d.ts +0 -74
  85. package/dist/src/config/providerManager.js +0 -203
  86. package/dist/src/config/providerManager.js.map +0 -1
  87. package/dist/src/config/providerPersistence.test.d.ts +0 -6
  88. package/dist/src/config/providerPersistence.test.js +0 -283
  89. package/dist/src/config/providerPersistence.test.js.map +0 -1
  90. package/dist/src/ui/components/GeminiKeyDialog.d.ts +0 -11
  91. package/dist/src/ui/components/GeminiKeyDialog.js +0 -156
  92. package/dist/src/ui/components/GeminiKeyDialog.js.map +0 -1
  93. package/dist/src/ui/components/OpenAIEndpointDialog.d.ts +0 -19
  94. package/dist/src/ui/components/OpenAIEndpointDialog.js +0 -163
  95. package/dist/src/ui/components/OpenAIEndpointDialog.js.map +0 -1
  96. package/dist/test-setup.ts +0 -12
  97. package/dist/test-utils/customMatchers.ts +0 -65
  98. package/dist/test-utils/mockCommandContext.test.ts +0 -62
  99. package/dist/test-utils/mockCommandContext.ts +0 -105
  100. package/dist/test-utils/render.tsx +0 -18
  101. package/dist/ui/App.test.tsx +0 -2181
  102. package/dist/ui/App.tsx +0 -1344
  103. package/dist/ui/IdeIntegrationNudge.tsx +0 -98
  104. package/dist/ui/__snapshots__/App.test.tsx.snap +0 -124
  105. package/dist/ui/colors.ts +0 -56
  106. package/dist/ui/commands/aboutCommand.test.ts +0 -153
  107. package/dist/ui/commands/aboutCommand.ts +0 -49
  108. package/dist/ui/commands/authCommand.test.ts +0 -36
  109. package/dist/ui/commands/authCommand.ts +0 -17
  110. package/dist/ui/commands/bugCommand.test.ts +0 -114
  111. package/dist/ui/commands/bugCommand.ts +0 -92
  112. package/dist/ui/commands/chatCommand.test.ts +0 -414
  113. package/dist/ui/commands/chatCommand.ts +0 -280
  114. package/dist/ui/commands/clearCommand.test.ts +0 -100
  115. package/dist/ui/commands/clearCommand.ts +0 -29
  116. package/dist/ui/commands/compressCommand.test.ts +0 -129
  117. package/dist/ui/commands/compressCommand.ts +0 -78
  118. package/dist/ui/commands/contextCommand.ts +0 -132
  119. package/dist/ui/commands/copyCommand.test.ts +0 -296
  120. package/dist/ui/commands/copyCommand.ts +0 -67
  121. package/dist/ui/commands/corgiCommand.test.ts +0 -34
  122. package/dist/ui/commands/corgiCommand.ts +0 -16
  123. package/dist/ui/commands/directoryCommand.test.tsx +0 -185
  124. package/dist/ui/commands/directoryCommand.tsx +0 -179
  125. package/dist/ui/commands/docsCommand.test.ts +0 -99
  126. package/dist/ui/commands/docsCommand.ts +0 -42
  127. package/dist/ui/commands/editorCommand.test.ts +0 -30
  128. package/dist/ui/commands/editorCommand.ts +0 -21
  129. package/dist/ui/commands/extensionsCommand.test.ts +0 -67
  130. package/dist/ui/commands/extensionsCommand.ts +0 -46
  131. package/dist/ui/commands/helpCommand.test.ts +0 -52
  132. package/dist/ui/commands/helpCommand.ts +0 -23
  133. package/dist/ui/commands/ideCommand.test.ts +0 -255
  134. package/dist/ui/commands/ideCommand.ts +0 -283
  135. package/dist/ui/commands/initCommand.test.ts +0 -127
  136. package/dist/ui/commands/initCommand.ts +0 -117
  137. package/dist/ui/commands/mcpCommand.test.ts +0 -1057
  138. package/dist/ui/commands/mcpCommand.ts +0 -531
  139. package/dist/ui/commands/memoryCommand.test.ts +0 -344
  140. package/dist/ui/commands/memoryCommand.ts +0 -305
  141. package/dist/ui/commands/privacyCommand.test.ts +0 -38
  142. package/dist/ui/commands/privacyCommand.ts +0 -17
  143. package/dist/ui/commands/quitCommand.test.ts +0 -55
  144. package/dist/ui/commands/quitCommand.ts +0 -36
  145. package/dist/ui/commands/restoreCommand.test.ts +0 -250
  146. package/dist/ui/commands/restoreCommand.ts +0 -157
  147. package/dist/ui/commands/searchEngineSetupCommand.ts +0 -18
  148. package/dist/ui/commands/settingsCommand.test.ts +0 -36
  149. package/dist/ui/commands/settingsCommand.ts +0 -17
  150. package/dist/ui/commands/setupGithubCommand.test.ts +0 -238
  151. package/dist/ui/commands/setupGithubCommand.ts +0 -212
  152. package/dist/ui/commands/speakCommand.ts +0 -175
  153. package/dist/ui/commands/statsCommand.test.ts +0 -78
  154. package/dist/ui/commands/statsCommand.ts +0 -70
  155. package/dist/ui/commands/terminalSetupCommand.test.ts +0 -85
  156. package/dist/ui/commands/terminalSetupCommand.ts +0 -45
  157. package/dist/ui/commands/themeCommand.test.ts +0 -38
  158. package/dist/ui/commands/themeCommand.ts +0 -17
  159. package/dist/ui/commands/toolsCommand.test.ts +0 -105
  160. package/dist/ui/commands/toolsCommand.ts +0 -71
  161. package/dist/ui/commands/ttsCommand.ts +0 -143
  162. package/dist/ui/commands/types.ts +0 -204
  163. package/dist/ui/commands/vimCommand.ts +0 -25
  164. package/dist/ui/commands/voiceCommand.ts +0 -125
  165. package/dist/ui/components/AboutBox.tsx +0 -133
  166. package/dist/ui/components/AsciiArt.ts +0 -54
  167. package/dist/ui/components/AuthDialog.test.tsx +0 -334
  168. package/dist/ui/components/AuthDialog.tsx +0 -289
  169. package/dist/ui/components/AuthInProgress.tsx +0 -62
  170. package/dist/ui/components/AutoAcceptIndicator.tsx +0 -47
  171. package/dist/ui/components/ConsoleSummaryDisplay.tsx +0 -35
  172. package/dist/ui/components/ContextSummaryDisplay.test.tsx +0 -85
  173. package/dist/ui/components/ContextSummaryDisplay.tsx +0 -120
  174. package/dist/ui/components/ContextUsageDisplay.tsx +0 -77
  175. package/dist/ui/components/DebugProfiler.tsx +0 -36
  176. package/dist/ui/components/DetailedMessagesDisplay.tsx +0 -82
  177. package/dist/ui/components/EditorSettingsDialog.tsx +0 -172
  178. package/dist/ui/components/FolderTrustDialog.test.tsx +0 -36
  179. package/dist/ui/components/FolderTrustDialog.tsx +0 -74
  180. package/dist/ui/components/Footer.test.tsx +0 -159
  181. package/dist/ui/components/Footer.tsx +0 -158
  182. package/dist/ui/components/GeminiKeyDialog.tsx +0 -252
  183. package/dist/ui/components/GeminiRespondingSpinner.tsx +0 -34
  184. package/dist/ui/components/Header.test.tsx +0 -44
  185. package/dist/ui/components/Header.tsx +0 -70
  186. package/dist/ui/components/Help.tsx +0 -174
  187. package/dist/ui/components/HistoryItemDisplay.test.tsx +0 -125
  188. package/dist/ui/components/HistoryItemDisplay.tsx +0 -98
  189. package/dist/ui/components/InputPrompt.test.tsx +0 -1467
  190. package/dist/ui/components/InputPrompt.tsx +0 -641
  191. package/dist/ui/components/LMStudioModelPrompt.tsx +0 -215
  192. package/dist/ui/components/LoadingIndicator.test.tsx +0 -296
  193. package/dist/ui/components/LoadingIndicator.tsx +0 -82
  194. package/dist/ui/components/MemoryUsageDisplay.tsx +0 -36
  195. package/dist/ui/components/ModelStatsDisplay.test.tsx +0 -252
  196. package/dist/ui/components/ModelStatsDisplay.tsx +0 -197
  197. package/dist/ui/components/OllamaModelPrompt.tsx +0 -206
  198. package/dist/ui/components/OpenAIEndpointDialog.tsx +0 -261
  199. package/dist/ui/components/OpenAIKeyPrompt.test.tsx +0 -64
  200. package/dist/ui/components/OpenAIKeyPrompt.tsx +0 -197
  201. package/dist/ui/components/PrepareLabel.tsx +0 -48
  202. package/dist/ui/components/SearchEngineConfigDialog.tsx +0 -280
  203. package/dist/ui/components/SessionSummaryDisplay.test.tsx +0 -75
  204. package/dist/ui/components/SessionSummaryDisplay.tsx +0 -18
  205. package/dist/ui/components/SettingsDialog.test.tsx +0 -865
  206. package/dist/ui/components/SettingsDialog.tsx +0 -753
  207. package/dist/ui/components/ShellConfirmationDialog.test.tsx +0 -53
  208. package/dist/ui/components/ShellConfirmationDialog.tsx +0 -103
  209. package/dist/ui/components/ShellModeIndicator.tsx +0 -18
  210. package/dist/ui/components/ShowMoreLines.tsx +0 -40
  211. package/dist/ui/components/StatsDisplay.test.tsx +0 -401
  212. package/dist/ui/components/StatsDisplay.tsx +0 -273
  213. package/dist/ui/components/SuggestionsDisplay.tsx +0 -102
  214. package/dist/ui/components/ThemeDialog.tsx +0 -310
  215. package/dist/ui/components/Tips.tsx +0 -45
  216. package/dist/ui/components/TodoDisplay.test.tsx +0 -97
  217. package/dist/ui/components/TodoDisplay.tsx +0 -72
  218. package/dist/ui/components/ToolStatsDisplay.test.tsx +0 -180
  219. package/dist/ui/components/ToolStatsDisplay.tsx +0 -208
  220. package/dist/ui/components/UpdateNotification.tsx +0 -23
  221. package/dist/ui/components/WelcomeBackDialog.tsx +0 -290
  222. package/dist/ui/components/__snapshots__/IDEContextDetailDisplay.test.tsx.snap +0 -24
  223. package/dist/ui/components/__snapshots__/ModelStatsDisplay.test.tsx.snap +0 -121
  224. package/dist/ui/components/__snapshots__/SessionSummaryDisplay.test.tsx.snap +0 -30
  225. package/dist/ui/components/__snapshots__/ShellConfirmationDialog.test.tsx.snap +0 -21
  226. package/dist/ui/components/__snapshots__/StatsDisplay.test.tsx.snap +0 -264
  227. package/dist/ui/components/__snapshots__/ToolStatsDisplay.test.tsx.snap +0 -91
  228. package/dist/ui/components/messages/CompressionMessage.tsx +0 -49
  229. package/dist/ui/components/messages/DiffRenderer.test.tsx +0 -365
  230. package/dist/ui/components/messages/DiffRenderer.tsx +0 -358
  231. package/dist/ui/components/messages/ErrorMessage.tsx +0 -31
  232. package/dist/ui/components/messages/GeminiMessage.tsx +0 -43
  233. package/dist/ui/components/messages/GeminiMessageContent.tsx +0 -43
  234. package/dist/ui/components/messages/InfoMessage.tsx +0 -32
  235. package/dist/ui/components/messages/ToolConfirmationMessage.test.tsx +0 -58
  236. package/dist/ui/components/messages/ToolConfirmationMessage.tsx +0 -297
  237. package/dist/ui/components/messages/ToolGroupMessage.tsx +0 -126
  238. package/dist/ui/components/messages/ToolMessage.test.tsx +0 -183
  239. package/dist/ui/components/messages/ToolMessage.tsx +0 -296
  240. package/dist/ui/components/messages/UserMessage.tsx +0 -43
  241. package/dist/ui/components/messages/UserShellMessage.tsx +0 -25
  242. package/dist/ui/components/shared/MaxSizedBox.test.tsx +0 -425
  243. package/dist/ui/components/shared/MaxSizedBox.tsx +0 -624
  244. package/dist/ui/components/shared/RadioButtonSelect.test.tsx +0 -181
  245. package/dist/ui/components/shared/RadioButtonSelect.tsx +0 -234
  246. package/dist/ui/components/shared/__snapshots__/RadioButtonSelect.test.tsx.snap +0 -47
  247. package/dist/ui/components/shared/text-buffer.test.ts +0 -1728
  248. package/dist/ui/components/shared/text-buffer.ts +0 -2227
  249. package/dist/ui/components/shared/vim-buffer-actions.test.ts +0 -1119
  250. package/dist/ui/components/shared/vim-buffer-actions.ts +0 -814
  251. package/dist/ui/constants.ts +0 -17
  252. package/dist/ui/contexts/KeypressContext.test.tsx +0 -391
  253. package/dist/ui/contexts/KeypressContext.tsx +0 -440
  254. package/dist/ui/contexts/OverflowContext.tsx +0 -87
  255. package/dist/ui/contexts/SessionContext.test.tsx +0 -132
  256. package/dist/ui/contexts/SessionContext.tsx +0 -143
  257. package/dist/ui/contexts/SettingsContext.tsx +0 -20
  258. package/dist/ui/contexts/StreamingContext.tsx +0 -22
  259. package/dist/ui/contexts/VimModeContext.tsx +0 -79
  260. package/dist/ui/editors/editorSettingsManager.ts +0 -66
  261. package/dist/ui/hooks/atCommandProcessor.test.ts +0 -1102
  262. package/dist/ui/hooks/atCommandProcessor.ts +0 -485
  263. package/dist/ui/hooks/shellCommandProcessor.test.ts +0 -481
  264. package/dist/ui/hooks/shellCommandProcessor.ts +0 -314
  265. package/dist/ui/hooks/slashCommandProcessor.test.ts +0 -1044
  266. package/dist/ui/hooks/slashCommandProcessor.ts +0 -595
  267. package/dist/ui/hooks/useAtCompletion.test.ts +0 -497
  268. package/dist/ui/hooks/useAtCompletion.ts +0 -244
  269. package/dist/ui/hooks/useAuthCommand.ts +0 -129
  270. package/dist/ui/hooks/useAutoAcceptIndicator.test.ts +0 -300
  271. package/dist/ui/hooks/useAutoAcceptIndicator.ts +0 -52
  272. package/dist/ui/hooks/useBracketedPaste.ts +0 -37
  273. package/dist/ui/hooks/useCommandCompletion.test.ts +0 -518
  274. package/dist/ui/hooks/useCommandCompletion.tsx +0 -238
  275. package/dist/ui/hooks/useCompletion.ts +0 -128
  276. package/dist/ui/hooks/useConsoleMessages.test.ts +0 -147
  277. package/dist/ui/hooks/useConsoleMessages.ts +0 -110
  278. package/dist/ui/hooks/useEditorSettings.test.ts +0 -283
  279. package/dist/ui/hooks/useEditorSettings.ts +0 -75
  280. package/dist/ui/hooks/useFocus.test.ts +0 -119
  281. package/dist/ui/hooks/useFocus.ts +0 -48
  282. package/dist/ui/hooks/useFolderTrust.test.ts +0 -159
  283. package/dist/ui/hooks/useFolderTrust.ts +0 -72
  284. package/dist/ui/hooks/useGeminiStream.test.tsx +0 -1998
  285. package/dist/ui/hooks/useGeminiStream.ts +0 -1017
  286. package/dist/ui/hooks/useGitBranchName.test.ts +0 -280
  287. package/dist/ui/hooks/useGitBranchName.ts +0 -79
  288. package/dist/ui/hooks/useHistoryManager.test.ts +0 -202
  289. package/dist/ui/hooks/useHistoryManager.ts +0 -111
  290. package/dist/ui/hooks/useInputHistory.test.ts +0 -261
  291. package/dist/ui/hooks/useInputHistory.ts +0 -111
  292. package/dist/ui/hooks/useKeypress.test.ts +0 -280
  293. package/dist/ui/hooks/useKeypress.ts +0 -39
  294. package/dist/ui/hooks/useKittyKeyboardProtocol.ts +0 -31
  295. package/dist/ui/hooks/useLoadingIndicator.test.ts +0 -139
  296. package/dist/ui/hooks/useLoadingIndicator.ts +0 -57
  297. package/dist/ui/hooks/useLogger.ts +0 -32
  298. package/dist/ui/hooks/useMessageQueue.test.ts +0 -226
  299. package/dist/ui/hooks/useMessageQueue.ts +0 -69
  300. package/dist/ui/hooks/usePhraseCycler.test.ts +0 -145
  301. package/dist/ui/hooks/usePhraseCycler.ts +0 -198
  302. package/dist/ui/hooks/usePrivacySettings.test.ts +0 -242
  303. package/dist/ui/hooks/usePrivacySettings.ts +0 -150
  304. package/dist/ui/hooks/useReactToolScheduler.ts +0 -309
  305. package/dist/ui/hooks/useRefreshMemoryCommand.ts +0 -7
  306. package/dist/ui/hooks/useReverseSearchCompletion.test.tsx +0 -260
  307. package/dist/ui/hooks/useReverseSearchCompletion.tsx +0 -95
  308. package/dist/ui/hooks/useSettingsCommand.ts +0 -25
  309. package/dist/ui/hooks/useShellHistory.test.ts +0 -219
  310. package/dist/ui/hooks/useShellHistory.ts +0 -133
  311. package/dist/ui/hooks/useShowMemoryCommand.ts +0 -75
  312. package/dist/ui/hooks/useSlashCompletion.test.ts +0 -434
  313. package/dist/ui/hooks/useSlashCompletion.ts +0 -187
  314. package/dist/ui/hooks/useStateAndRef.ts +0 -36
  315. package/dist/ui/hooks/useTerminalSize.ts +0 -32
  316. package/dist/ui/hooks/useThemeCommand.ts +0 -110
  317. package/dist/ui/hooks/useTimer.test.ts +0 -120
  318. package/dist/ui/hooks/useTimer.ts +0 -65
  319. package/dist/ui/hooks/useToolScheduler.test.ts +0 -1123
  320. package/dist/ui/hooks/useWelcomeBack.ts +0 -253
  321. package/dist/ui/hooks/vim.test.ts +0 -1691
  322. package/dist/ui/hooks/vim.ts +0 -784
  323. package/dist/ui/keyMatchers.test.ts +0 -337
  324. package/dist/ui/keyMatchers.ts +0 -105
  325. package/dist/ui/privacy/CloudFreePrivacyNotice.tsx +0 -117
  326. package/dist/ui/privacy/CloudPaidPrivacyNotice.tsx +0 -59
  327. package/dist/ui/privacy/GeminiPrivacyNotice.tsx +0 -62
  328. package/dist/ui/privacy/PrivacyNotice.tsx +0 -42
  329. package/dist/ui/semantic-colors.ts +0 -26
  330. package/dist/ui/themes/ansi-light.ts +0 -150
  331. package/dist/ui/themes/ansi.ts +0 -159
  332. package/dist/ui/themes/atom-one-dark.ts +0 -147
  333. package/dist/ui/themes/ayu-light.ts +0 -139
  334. package/dist/ui/themes/ayu.ts +0 -113
  335. package/dist/ui/themes/color-utils.test.ts +0 -221
  336. package/dist/ui/themes/color-utils.ts +0 -231
  337. package/dist/ui/themes/default-light.ts +0 -108
  338. package/dist/ui/themes/default.ts +0 -151
  339. package/dist/ui/themes/dracula.ts +0 -124
  340. package/dist/ui/themes/fss-code-dark.ts +0 -156
  341. package/dist/ui/themes/fss-dark.ts +0 -113
  342. package/dist/ui/themes/fss-light.ts +0 -139
  343. package/dist/ui/themes/github-dark.ts +0 -147
  344. package/dist/ui/themes/github-light.ts +0 -149
  345. package/dist/ui/themes/googlecode.ts +0 -146
  346. package/dist/ui/themes/no-color.ts +0 -125
  347. package/dist/ui/themes/qwen-dark.ts +0 -118
  348. package/dist/ui/themes/qwen-light.ts +0 -144
  349. package/dist/ui/themes/semantic-tokens.ts +0 -127
  350. package/dist/ui/themes/shades-of-purple.ts +0 -352
  351. package/dist/ui/themes/theme-manager.test.ts +0 -99
  352. package/dist/ui/themes/theme-manager.ts +0 -257
  353. package/dist/ui/themes/theme.test.ts +0 -97
  354. package/dist/ui/themes/theme.ts +0 -451
  355. package/dist/ui/themes/xcode.ts +0 -154
  356. package/dist/ui/types.ts +0 -255
  357. package/dist/ui/utils/CodeColorizer.tsx +0 -217
  358. package/dist/ui/utils/ConsolePatcher.ts +0 -71
  359. package/dist/ui/utils/InlineMarkdownRenderer.tsx +0 -173
  360. package/dist/ui/utils/MarkdownDisplay.test.tsx +0 -244
  361. package/dist/ui/utils/MarkdownDisplay.tsx +0 -415
  362. package/dist/ui/utils/TableRenderer.tsx +0 -159
  363. package/dist/ui/utils/__snapshots__/MarkdownDisplay.test.tsx.snap +0 -93
  364. package/dist/ui/utils/clipboardUtils.test.ts +0 -76
  365. package/dist/ui/utils/clipboardUtils.ts +0 -149
  366. package/dist/ui/utils/commandUtils.test.ts +0 -384
  367. package/dist/ui/utils/commandUtils.ts +0 -106
  368. package/dist/ui/utils/computeStats.test.ts +0 -292
  369. package/dist/ui/utils/computeStats.ts +0 -86
  370. package/dist/ui/utils/displayUtils.test.ts +0 -58
  371. package/dist/ui/utils/displayUtils.ts +0 -32
  372. package/dist/ui/utils/formatters.test.ts +0 -72
  373. package/dist/ui/utils/formatters.ts +0 -63
  374. package/dist/ui/utils/isNarrowWidth.ts +0 -9
  375. package/dist/ui/utils/kittyProtocolDetector.ts +0 -105
  376. package/dist/ui/utils/markdownUtilities.test.ts +0 -50
  377. package/dist/ui/utils/markdownUtilities.ts +0 -125
  378. package/dist/ui/utils/platformConstants.ts +0 -52
  379. package/dist/ui/utils/terminalSetup.ts +0 -342
  380. package/dist/ui/utils/textUtils.ts +0 -40
  381. package/dist/ui/utils/updateCheck.test.ts +0 -163
  382. package/dist/ui/utils/updateCheck.ts +0 -100
  383. package/dist/utils/checks.ts +0 -28
  384. package/dist/utils/cleanup.test.ts +0 -68
  385. package/dist/utils/cleanup.ts +0 -36
  386. package/dist/utils/dialogScopeUtils.ts +0 -64
  387. package/dist/utils/events.ts +0 -14
  388. package/dist/utils/gitUtils.test.ts +0 -149
  389. package/dist/utils/gitUtils.ts +0 -116
  390. package/dist/utils/handleAutoUpdate.test.ts +0 -272
  391. package/dist/utils/handleAutoUpdate.ts +0 -145
  392. package/dist/utils/installationInfo.test.ts +0 -315
  393. package/dist/utils/installationInfo.ts +0 -176
  394. package/dist/utils/package.ts +0 -38
  395. package/dist/utils/readStdin.ts +0 -51
  396. package/dist/utils/resolvePath.ts +0 -21
  397. package/dist/utils/sandbox-macos-permissive-closed.sb +0 -32
  398. package/dist/utils/sandbox-macos-permissive-open.sb +0 -25
  399. package/dist/utils/sandbox-macos-permissive-proxied.sb +0 -37
  400. package/dist/utils/sandbox-macos-restrictive-closed.sb +0 -93
  401. package/dist/utils/sandbox-macos-restrictive-open.sb +0 -96
  402. package/dist/utils/sandbox-macos-restrictive-proxied.sb +0 -98
  403. package/dist/utils/sandbox.ts +0 -962
  404. package/dist/utils/settingsUtils.test.ts +0 -797
  405. package/dist/utils/settingsUtils.ts +0 -489
  406. package/dist/utils/spawnWrapper.ts +0 -9
  407. package/dist/utils/startupWarnings.test.ts +0 -83
  408. package/dist/utils/startupWarnings.ts +0 -40
  409. package/dist/utils/updateEventEmitter.ts +0 -13
  410. package/dist/utils/userStartupWarnings.test.ts +0 -87
  411. package/dist/utils/userStartupWarnings.ts +0 -69
  412. package/dist/utils/version.ts +0 -12
  413. package/dist/validateNonInterActiveAuth.test.ts +0 -260
  414. package/dist/validateNonInterActiveAuth.ts +0 -51
  415. package/dist/vitest.config.ts +0 -37
  416. package/dist/zed-integration/acp.ts +0 -366
  417. package/dist/zed-integration/fileSystemService.ts +0 -47
  418. package/dist/zed-integration/schema.ts +0 -466
  419. 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
- });