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