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,1102 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
-
7
- import { describe, it, expect, vi, beforeEach, afterEach, Mock } from 'vitest';
8
- import { handleAtCommand } from './atCommandProcessor.js';
9
- import {
10
- Config,
11
- FileDiscoveryService,
12
- GlobTool,
13
- ReadManyFilesTool,
14
- StandardFileSystemService,
15
- ToolRegistry,
16
- } from 'fss-link-core';
17
- import * as os from 'os';
18
- import { ToolCallStatus } from '../types.js';
19
- import { UseHistoryManagerReturn } from './useHistoryManager.js';
20
- import * as fsPromises from 'fs/promises';
21
- import * as path from 'path';
22
-
23
- describe('handleAtCommand', () => {
24
- let testRootDir: string;
25
- let mockConfig: Config;
26
-
27
- const mockAddItem: Mock<UseHistoryManagerReturn['addItem']> = vi.fn();
28
- const mockOnDebugMessage: Mock<(message: string) => void> = vi.fn();
29
-
30
- let abortController: AbortController;
31
-
32
- async function createTestFile(fullPath: string, fileContents: string) {
33
- await fsPromises.mkdir(path.dirname(fullPath), { recursive: true });
34
- await fsPromises.writeFile(fullPath, fileContents);
35
- return path.resolve(testRootDir, fullPath);
36
- }
37
-
38
- beforeEach(async () => {
39
- vi.resetAllMocks();
40
-
41
- testRootDir = await fsPromises.mkdtemp(
42
- path.join(os.tmpdir(), 'folder-structure-test-'),
43
- );
44
-
45
- abortController = new AbortController();
46
-
47
- const getToolRegistry = vi.fn();
48
-
49
- mockConfig = {
50
- getToolRegistry,
51
- getTargetDir: () => testRootDir,
52
- isSandboxed: () => false,
53
- getFileService: () => new FileDiscoveryService(testRootDir),
54
- getFileFilteringRespectGitIgnore: () => true,
55
- getFileFilteringRespectGeminiIgnore: () => true,
56
- getFileFilteringOptions: () => ({
57
- respectGitIgnore: true,
58
- respectFssLinkIgnore: true,
59
- }),
60
- getFileSystemService: () => new StandardFileSystemService(),
61
- getEnableRecursiveFileSearch: vi.fn(() => true),
62
- getWorkspaceContext: () => ({
63
- isPathWithinWorkspace: () => true,
64
- getDirectories: () => [testRootDir],
65
- }),
66
- getMcpServers: () => ({}),
67
- getMcpServerCommand: () => undefined,
68
- getPromptRegistry: () => ({
69
- getPromptsByServer: () => [],
70
- }),
71
- getDebugMode: () => false,
72
- } as unknown as Config;
73
-
74
- const registry = new ToolRegistry(mockConfig);
75
- registry.registerTool(new ReadManyFilesTool(mockConfig));
76
- registry.registerTool(new GlobTool(mockConfig));
77
- getToolRegistry.mockReturnValue(registry);
78
- });
79
-
80
- afterEach(async () => {
81
- abortController.abort();
82
- await fsPromises.rm(testRootDir, { recursive: true, force: true });
83
- });
84
-
85
- it('should pass through query if no @ command is present', async () => {
86
- const query = 'regular user query';
87
-
88
- const result = await handleAtCommand({
89
- query,
90
- config: mockConfig,
91
- addItem: mockAddItem,
92
- onDebugMessage: mockOnDebugMessage,
93
- messageId: 123,
94
- signal: abortController.signal,
95
- });
96
-
97
- expect(result).toEqual({
98
- processedQuery: [{ text: query }],
99
- shouldProceed: true,
100
- });
101
- });
102
-
103
- it('should pass through original query if only a lone @ symbol is present', async () => {
104
- const queryWithSpaces = ' @ ';
105
-
106
- const result = await handleAtCommand({
107
- query: queryWithSpaces,
108
- config: mockConfig,
109
- addItem: mockAddItem,
110
- onDebugMessage: mockOnDebugMessage,
111
- messageId: 124,
112
- signal: abortController.signal,
113
- });
114
-
115
- expect(result).toEqual({
116
- processedQuery: [{ text: queryWithSpaces }],
117
- shouldProceed: true,
118
- });
119
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
120
- 'Lone @ detected, will be treated as text in the modified query.',
121
- );
122
- });
123
-
124
- it('should process a valid text file path', async () => {
125
- const fileContent = 'This is the file content.';
126
- const filePath = await createTestFile(
127
- path.join(testRootDir, 'path', 'to', 'file.txt'),
128
- fileContent,
129
- );
130
- const query = `@${filePath}`;
131
-
132
- const result = await handleAtCommand({
133
- query,
134
- config: mockConfig,
135
- addItem: mockAddItem,
136
- onDebugMessage: mockOnDebugMessage,
137
- messageId: 125,
138
- signal: abortController.signal,
139
- });
140
-
141
- expect(result).toEqual({
142
- processedQuery: [
143
- { text: `@${filePath}` },
144
- { text: '\n--- Content from referenced files ---' },
145
- { text: `\nContent from @${filePath}:\n` },
146
- { text: fileContent },
147
- { text: '\n--- End of content ---' },
148
- ],
149
- shouldProceed: true,
150
- });
151
- expect(mockAddItem).toHaveBeenCalledWith(
152
- expect.objectContaining({
153
- type: 'tool_group',
154
- tools: [expect.objectContaining({ status: ToolCallStatus.Success })],
155
- }),
156
- 125,
157
- );
158
- });
159
-
160
- it('should process a valid directory path and convert to glob', async () => {
161
- const fileContent = 'This is the file content.';
162
- const filePath = await createTestFile(
163
- path.join(testRootDir, 'path', 'to', 'file.txt'),
164
- fileContent,
165
- );
166
- const dirPath = path.dirname(filePath);
167
- const query = `@${dirPath}`;
168
- const resolvedGlob = `${dirPath}/**`;
169
-
170
- const result = await handleAtCommand({
171
- query,
172
- config: mockConfig,
173
- addItem: mockAddItem,
174
- onDebugMessage: mockOnDebugMessage,
175
- messageId: 126,
176
- signal: abortController.signal,
177
- });
178
-
179
- expect(result).toEqual({
180
- processedQuery: [
181
- { text: `@${resolvedGlob}` },
182
- { text: '\n--- Content from referenced files ---' },
183
- { text: `\nContent from @${filePath}:\n` },
184
- { text: fileContent },
185
- { text: '\n--- End of content ---' },
186
- ],
187
- shouldProceed: true,
188
- });
189
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
190
- `Path ${dirPath} resolved to directory, using glob: ${resolvedGlob}`,
191
- );
192
- });
193
-
194
- it('should handle query with text before and after @command', async () => {
195
- const fileContent = 'Markdown content.';
196
- const filePath = await createTestFile(
197
- path.join(testRootDir, 'doc.md'),
198
- fileContent,
199
- );
200
- const textBefore = 'Explain this: ';
201
- const textAfter = ' in detail.';
202
- const query = `${textBefore}@${filePath}${textAfter}`;
203
-
204
- const result = await handleAtCommand({
205
- query,
206
- config: mockConfig,
207
- addItem: mockAddItem,
208
- onDebugMessage: mockOnDebugMessage,
209
- messageId: 128,
210
- signal: abortController.signal,
211
- });
212
-
213
- expect(result).toEqual({
214
- processedQuery: [
215
- { text: `${textBefore}@${filePath}${textAfter}` },
216
- { text: '\n--- Content from referenced files ---' },
217
- { text: `\nContent from @${filePath}:\n` },
218
- { text: fileContent },
219
- { text: '\n--- End of content ---' },
220
- ],
221
- shouldProceed: true,
222
- });
223
- });
224
-
225
- it('should correctly unescape paths with escaped spaces', async () => {
226
- const fileContent = 'This is the file content.';
227
- const filePath = await createTestFile(
228
- path.join(testRootDir, 'path', 'to', 'my file.txt'),
229
- fileContent,
230
- );
231
- const escapedpath = path.join(testRootDir, 'path', 'to', 'my\\ file.txt');
232
- const query = `@${escapedpath}`;
233
-
234
- const result = await handleAtCommand({
235
- query,
236
- config: mockConfig,
237
- addItem: mockAddItem,
238
- onDebugMessage: mockOnDebugMessage,
239
- messageId: 125,
240
- signal: abortController.signal,
241
- });
242
-
243
- expect(result).toEqual({
244
- processedQuery: [
245
- { text: `@${filePath}` },
246
- { text: '\n--- Content from referenced files ---' },
247
- { text: `\nContent from @${filePath}:\n` },
248
- { text: fileContent },
249
- { text: '\n--- End of content ---' },
250
- ],
251
- shouldProceed: true,
252
- });
253
- expect(mockAddItem).toHaveBeenCalledWith(
254
- expect.objectContaining({
255
- type: 'tool_group',
256
- tools: [expect.objectContaining({ status: ToolCallStatus.Success })],
257
- }),
258
- 125,
259
- );
260
- });
261
-
262
- it('should handle multiple @file references', async () => {
263
- const content1 = 'Content file1';
264
- const file1Path = await createTestFile(
265
- path.join(testRootDir, 'file1.txt'),
266
- content1,
267
- );
268
- const content2 = 'Content file2';
269
- const file2Path = await createTestFile(
270
- path.join(testRootDir, 'file2.md'),
271
- content2,
272
- );
273
- const query = `@${file1Path} @${file2Path}`;
274
-
275
- const result = await handleAtCommand({
276
- query,
277
- config: mockConfig,
278
- addItem: mockAddItem,
279
- onDebugMessage: mockOnDebugMessage,
280
- messageId: 130,
281
- signal: abortController.signal,
282
- });
283
-
284
- expect(result).toEqual({
285
- processedQuery: [
286
- { text: query },
287
- { text: '\n--- Content from referenced files ---' },
288
- { text: `\nContent from @${file1Path}:\n` },
289
- { text: content1 },
290
- { text: `\nContent from @${file2Path}:\n` },
291
- { text: content2 },
292
- { text: '\n--- End of content ---' },
293
- ],
294
- shouldProceed: true,
295
- });
296
- });
297
-
298
- it('should handle multiple @file references with interleaved text', async () => {
299
- const text1 = 'Check ';
300
- const content1 = 'C1';
301
- const file1Path = await createTestFile(
302
- path.join(testRootDir, 'f1.txt'),
303
- content1,
304
- );
305
- const text2 = ' and ';
306
- const content2 = 'C2';
307
- const file2Path = await createTestFile(
308
- path.join(testRootDir, 'f2.md'),
309
- content2,
310
- );
311
- const text3 = ' please.';
312
- const query = `${text1}@${file1Path}${text2}@${file2Path}${text3}`;
313
-
314
- const result = await handleAtCommand({
315
- query,
316
- config: mockConfig,
317
- addItem: mockAddItem,
318
- onDebugMessage: mockOnDebugMessage,
319
- messageId: 131,
320
- signal: abortController.signal,
321
- });
322
-
323
- expect(result).toEqual({
324
- processedQuery: [
325
- { text: query },
326
- { text: '\n--- Content from referenced files ---' },
327
- { text: `\nContent from @${file1Path}:\n` },
328
- { text: content1 },
329
- { text: `\nContent from @${file2Path}:\n` },
330
- { text: content2 },
331
- { text: '\n--- End of content ---' },
332
- ],
333
- shouldProceed: true,
334
- });
335
- });
336
-
337
- it('should handle a mix of valid, invalid, and lone @ references', async () => {
338
- const content1 = 'Valid content 1';
339
- const file1Path = await createTestFile(
340
- path.join(testRootDir, 'valid1.txt'),
341
- content1,
342
- );
343
- const invalidFile = 'nonexistent.txt';
344
- const content2 = 'Globbed content';
345
- const file2Path = await createTestFile(
346
- path.join(testRootDir, 'resolved', 'valid2.actual'),
347
- content2,
348
- );
349
- const query = `Look at @${file1Path} then @${invalidFile} and also just @ symbol, then @${file2Path}`;
350
-
351
- const result = await handleAtCommand({
352
- query,
353
- config: mockConfig,
354
- addItem: mockAddItem,
355
- onDebugMessage: mockOnDebugMessage,
356
- messageId: 132,
357
- signal: abortController.signal,
358
- });
359
-
360
- expect(result).toEqual({
361
- processedQuery: [
362
- {
363
- text: `Look at @${file1Path} then @${invalidFile} and also just @ symbol, then @${file2Path}`,
364
- },
365
- { text: '\n--- Content from referenced files ---' },
366
- { text: `\nContent from @${file2Path}:\n` },
367
- { text: content2 },
368
- { text: `\nContent from @${file1Path}:\n` },
369
- { text: content1 },
370
- { text: '\n--- End of content ---' },
371
- ],
372
- shouldProceed: true,
373
- });
374
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
375
- `Path ${invalidFile} not found directly, attempting glob search.`,
376
- );
377
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
378
- `Glob search for '**/*${invalidFile}*' found no files or an error. Path ${invalidFile} will be skipped.`,
379
- );
380
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
381
- 'Lone @ detected, will be treated as text in the modified query.',
382
- );
383
- });
384
-
385
- it('should return original query if all @paths are invalid or lone @', async () => {
386
- const query = 'Check @nonexistent.txt and @ also';
387
-
388
- const result = await handleAtCommand({
389
- query,
390
- config: mockConfig,
391
- addItem: mockAddItem,
392
- onDebugMessage: mockOnDebugMessage,
393
- messageId: 133,
394
- signal: abortController.signal,
395
- });
396
-
397
- expect(result).toEqual({
398
- processedQuery: [{ text: 'Check @nonexistent.txt and @ also' }],
399
- shouldProceed: true,
400
- });
401
- });
402
-
403
- describe('git-aware filtering', () => {
404
- beforeEach(async () => {
405
- await fsPromises.mkdir(path.join(testRootDir, '.git'), {
406
- recursive: true,
407
- });
408
- });
409
-
410
- it('should skip git-ignored files in @ commands', async () => {
411
- await createTestFile(
412
- path.join(testRootDir, '.gitignore'),
413
- 'node_modules/package.json',
414
- );
415
- const gitIgnoredFile = await createTestFile(
416
- path.join(testRootDir, 'node_modules', 'package.json'),
417
- 'the file contents',
418
- );
419
-
420
- const query = `@${gitIgnoredFile}`;
421
-
422
- const result = await handleAtCommand({
423
- query,
424
- config: mockConfig,
425
- addItem: mockAddItem,
426
- onDebugMessage: mockOnDebugMessage,
427
- messageId: 200,
428
- signal: abortController.signal,
429
- });
430
-
431
- expect(result).toEqual({
432
- processedQuery: [{ text: query }],
433
- shouldProceed: true,
434
- });
435
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
436
- `Path ${gitIgnoredFile} is git-ignored and will be skipped.`,
437
- );
438
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
439
- `Ignored 1 files:\nGit-ignored: ${gitIgnoredFile}`,
440
- );
441
- });
442
-
443
- it('should process non-git-ignored files normally', async () => {
444
- await createTestFile(
445
- path.join(testRootDir, '.gitignore'),
446
- 'node_modules/package.json',
447
- );
448
-
449
- const validFile = await createTestFile(
450
- path.join(testRootDir, 'src', 'index.ts'),
451
- 'console.log("Hello world");',
452
- );
453
- const query = `@${validFile}`;
454
-
455
- const result = await handleAtCommand({
456
- query,
457
- config: mockConfig,
458
- addItem: mockAddItem,
459
- onDebugMessage: mockOnDebugMessage,
460
- messageId: 201,
461
- signal: abortController.signal,
462
- });
463
-
464
- expect(result).toEqual({
465
- processedQuery: [
466
- { text: `@${validFile}` },
467
- { text: '\n--- Content from referenced files ---' },
468
- { text: `\nContent from @${validFile}:\n` },
469
- { text: 'console.log("Hello world");' },
470
- { text: '\n--- End of content ---' },
471
- ],
472
- shouldProceed: true,
473
- });
474
- });
475
-
476
- it('should handle mixed git-ignored and valid files', async () => {
477
- await createTestFile(path.join(testRootDir, '.gitignore'), '.env');
478
- const validFile = await createTestFile(
479
- path.join(testRootDir, 'README.md'),
480
- '# Project README',
481
- );
482
- const gitIgnoredFile = await createTestFile(
483
- path.join(testRootDir, '.env'),
484
- 'SECRET=123',
485
- );
486
- const query = `@${validFile} @${gitIgnoredFile}`;
487
-
488
- const result = await handleAtCommand({
489
- query,
490
- config: mockConfig,
491
- addItem: mockAddItem,
492
- onDebugMessage: mockOnDebugMessage,
493
- messageId: 202,
494
- signal: abortController.signal,
495
- });
496
-
497
- expect(result).toEqual({
498
- processedQuery: [
499
- { text: `@${validFile} @${gitIgnoredFile}` },
500
- { text: '\n--- Content from referenced files ---' },
501
- { text: `\nContent from @${validFile}:\n` },
502
- { text: '# Project README' },
503
- { text: '\n--- End of content ---' },
504
- ],
505
- shouldProceed: true,
506
- });
507
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
508
- `Path ${gitIgnoredFile} is git-ignored and will be skipped.`,
509
- );
510
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
511
- `Ignored 1 files:\nGit-ignored: ${gitIgnoredFile}`,
512
- );
513
- });
514
-
515
- it('should always ignore .git directory files', async () => {
516
- const gitFile = await createTestFile(
517
- path.join(testRootDir, '.git', 'config'),
518
- '[core]\n\trepositoryformatversion = 0\n',
519
- );
520
- const query = `@${gitFile}`;
521
-
522
- const result = await handleAtCommand({
523
- query,
524
- config: mockConfig,
525
- addItem: mockAddItem,
526
- onDebugMessage: mockOnDebugMessage,
527
- messageId: 203,
528
- signal: abortController.signal,
529
- });
530
-
531
- expect(result).toEqual({
532
- processedQuery: [{ text: query }],
533
- shouldProceed: true,
534
- });
535
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
536
- `Path ${gitFile} is git-ignored and will be skipped.`,
537
- );
538
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
539
- `Ignored 1 files:\nGit-ignored: ${gitFile}`,
540
- );
541
- });
542
- });
543
-
544
- describe('when recursive file search is disabled', () => {
545
- beforeEach(() => {
546
- vi.mocked(mockConfig.getEnableRecursiveFileSearch).mockReturnValue(false);
547
- });
548
-
549
- it('should not use glob search for a nonexistent file', async () => {
550
- const invalidFile = 'nonexistent.txt';
551
- const query = `@${invalidFile}`;
552
-
553
- const result = await handleAtCommand({
554
- query,
555
- config: mockConfig,
556
- addItem: mockAddItem,
557
- onDebugMessage: mockOnDebugMessage,
558
- messageId: 300,
559
- signal: abortController.signal,
560
- });
561
-
562
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
563
- `Glob tool not found. Path ${invalidFile} will be skipped.`,
564
- );
565
- expect(result.processedQuery).toEqual([{ text: query }]);
566
- expect(result.shouldProceed).toBe(true);
567
- });
568
- });
569
-
570
- describe('fss-link-ignore filtering', () => {
571
- it('should skip fss-link-ignored files in @ commands', async () => {
572
- await createTestFile(
573
- path.join(testRootDir, '.fss-linkignore'),
574
- 'build/output.js',
575
- );
576
- const fssLinkIgnoredFile = await createTestFile(
577
- path.join(testRootDir, 'build', 'output.js'),
578
- 'console.log("Hello");',
579
- );
580
- const query = `@${fssLinkIgnoredFile}`;
581
-
582
- const result = await handleAtCommand({
583
- query,
584
- config: mockConfig,
585
- addItem: mockAddItem,
586
- onDebugMessage: mockOnDebugMessage,
587
- messageId: 204,
588
- signal: abortController.signal,
589
- });
590
-
591
- expect(result).toEqual({
592
- processedQuery: [{ text: query }],
593
- shouldProceed: true,
594
- });
595
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
596
- `Path ${fssLinkIgnoredFile} is fss-link-ignored and will be skipped.`,
597
- );
598
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
599
- `Ignored 1 files:\nFss-Link-ignored: ${fssLinkIgnoredFile}`,
600
- );
601
- });
602
- });
603
- it('should process non-ignored files when .fss-linkignore is present', async () => {
604
- await createTestFile(
605
- path.join(testRootDir, '.geminiignore'),
606
- 'build/output.js',
607
- );
608
- const validFile = await createTestFile(
609
- path.join(testRootDir, 'src', 'index.ts'),
610
- 'console.log("Hello world");',
611
- );
612
- const query = `@${validFile}`;
613
-
614
- const result = await handleAtCommand({
615
- query,
616
- config: mockConfig,
617
- addItem: mockAddItem,
618
- onDebugMessage: mockOnDebugMessage,
619
- messageId: 205,
620
- signal: abortController.signal,
621
- });
622
-
623
- expect(result).toEqual({
624
- processedQuery: [
625
- { text: `@${validFile}` },
626
- { text: '\n--- Content from referenced files ---' },
627
- { text: `\nContent from @${validFile}:\n` },
628
- { text: 'console.log("Hello world");' },
629
- { text: '\n--- End of content ---' },
630
- ],
631
- shouldProceed: true,
632
- });
633
- });
634
-
635
- it('should handle mixed gemini-ignored and valid files', async () => {
636
- await createTestFile(
637
- path.join(testRootDir, '.fss-linkignore'),
638
- 'dist/bundle.js',
639
- );
640
- const validFile = await createTestFile(
641
- path.join(testRootDir, 'src', 'main.ts'),
642
- '// Main application entry',
643
- );
644
- const fssLinkIgnoredFile = await createTestFile(
645
- path.join(testRootDir, 'dist', 'bundle.js'),
646
- 'console.log("bundle");',
647
- );
648
- const query = `@${validFile} @${fssLinkIgnoredFile}`;
649
-
650
- const result = await handleAtCommand({
651
- query,
652
- config: mockConfig,
653
- addItem: mockAddItem,
654
- onDebugMessage: mockOnDebugMessage,
655
- messageId: 206,
656
- signal: abortController.signal,
657
- });
658
-
659
- expect(result).toEqual({
660
- processedQuery: [
661
- { text: `@${validFile} @${fssLinkIgnoredFile}` },
662
- { text: '\n--- Content from referenced files ---' },
663
- { text: `\nContent from @${validFile}:\n` },
664
- { text: '// Main application entry' },
665
- { text: '\n--- End of content ---' },
666
- ],
667
- shouldProceed: true,
668
- });
669
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
670
- `Path ${fssLinkIgnoredFile} is fss-link-ignored and will be skipped.`,
671
- );
672
- expect(mockOnDebugMessage).toHaveBeenCalledWith(
673
- `Ignored 1 files:\nFss-Link-ignored: ${fssLinkIgnoredFile}`,
674
- );
675
- });
676
-
677
- describe('punctuation termination in @ commands', () => {
678
- const punctuationTestCases = [
679
- {
680
- name: 'comma',
681
- fileName: 'test.txt',
682
- fileContent: 'File content here',
683
- queryTemplate: (filePath: string) =>
684
- `Look at @${filePath}, then explain it.`,
685
- messageId: 400,
686
- },
687
- {
688
- name: 'period',
689
- fileName: 'readme.md',
690
- fileContent: 'File content here',
691
- queryTemplate: (filePath: string) =>
692
- `Check @${filePath}. What does it say?`,
693
- messageId: 401,
694
- },
695
- {
696
- name: 'semicolon',
697
- fileName: 'example.js',
698
- fileContent: 'Code example',
699
- queryTemplate: (filePath: string) =>
700
- `Review @${filePath}; check for bugs.`,
701
- messageId: 402,
702
- },
703
- {
704
- name: 'exclamation mark',
705
- fileName: 'important.txt',
706
- fileContent: 'Important content',
707
- queryTemplate: (filePath: string) =>
708
- `Look at @${filePath}! This is critical.`,
709
- messageId: 403,
710
- },
711
- {
712
- name: 'question mark',
713
- fileName: 'config.json',
714
- fileContent: 'Config settings',
715
- queryTemplate: (filePath: string) =>
716
- `What is in @${filePath}? Please explain.`,
717
- messageId: 404,
718
- },
719
- {
720
- name: 'opening parenthesis',
721
- fileName: 'func.ts',
722
- fileContent: 'Function definition',
723
- queryTemplate: (filePath: string) =>
724
- `Analyze @${filePath}(the main function).`,
725
- messageId: 405,
726
- },
727
- {
728
- name: 'closing parenthesis',
729
- fileName: 'data.json',
730
- fileContent: 'Test data',
731
- queryTemplate: (filePath: string) =>
732
- `Use data from @${filePath}) for testing.`,
733
- messageId: 406,
734
- },
735
- {
736
- name: 'opening square bracket',
737
- fileName: 'array.js',
738
- fileContent: 'Array data',
739
- queryTemplate: (filePath: string) =>
740
- `Check @${filePath}[0] for the first element.`,
741
- messageId: 407,
742
- },
743
- {
744
- name: 'closing square bracket',
745
- fileName: 'list.md',
746
- fileContent: 'List content',
747
- queryTemplate: (filePath: string) =>
748
- `Review item @${filePath}] from the list.`,
749
- messageId: 408,
750
- },
751
- {
752
- name: 'opening curly brace',
753
- fileName: 'object.ts',
754
- fileContent: 'Object definition',
755
- queryTemplate: (filePath: string) =>
756
- `Parse @${filePath}{prop1: value1}.`,
757
- messageId: 409,
758
- },
759
- {
760
- name: 'closing curly brace',
761
- fileName: 'config.yaml',
762
- fileContent: 'Configuration',
763
- queryTemplate: (filePath: string) =>
764
- `Use settings from @${filePath}} for deployment.`,
765
- messageId: 410,
766
- },
767
- ];
768
-
769
- it.each(punctuationTestCases)(
770
- 'should terminate @path at $name',
771
- async ({ fileName, fileContent, queryTemplate, messageId }) => {
772
- const filePath = await createTestFile(
773
- path.join(testRootDir, fileName),
774
- fileContent,
775
- );
776
- const query = queryTemplate(filePath);
777
-
778
- const result = await handleAtCommand({
779
- query,
780
- config: mockConfig,
781
- addItem: mockAddItem,
782
- onDebugMessage: mockOnDebugMessage,
783
- messageId,
784
- signal: abortController.signal,
785
- });
786
-
787
- expect(result).toEqual({
788
- processedQuery: [
789
- { text: query },
790
- { text: '\n--- Content from referenced files ---' },
791
- { text: `\nContent from @${filePath}:\n` },
792
- { text: fileContent },
793
- { text: '\n--- End of content ---' },
794
- ],
795
- shouldProceed: true,
796
- });
797
- },
798
- );
799
-
800
- it('should handle multiple @paths terminated by different punctuation', async () => {
801
- const content1 = 'First file';
802
- const file1Path = await createTestFile(
803
- path.join(testRootDir, 'first.txt'),
804
- content1,
805
- );
806
- const content2 = 'Second file';
807
- const file2Path = await createTestFile(
808
- path.join(testRootDir, 'second.txt'),
809
- content2,
810
- );
811
- const query = `Compare @${file1Path}, @${file2Path}; what's different?`;
812
-
813
- const result = await handleAtCommand({
814
- query,
815
- config: mockConfig,
816
- addItem: mockAddItem,
817
- onDebugMessage: mockOnDebugMessage,
818
- messageId: 411,
819
- signal: abortController.signal,
820
- });
821
-
822
- expect(result).toEqual({
823
- processedQuery: [
824
- { text: `Compare @${file1Path}, @${file2Path}; what's different?` },
825
- { text: '\n--- Content from referenced files ---' },
826
- { text: `\nContent from @${file1Path}:\n` },
827
- { text: content1 },
828
- { text: `\nContent from @${file2Path}:\n` },
829
- { text: content2 },
830
- { text: '\n--- End of content ---' },
831
- ],
832
- shouldProceed: true,
833
- });
834
- });
835
-
836
- it('should still handle escaped spaces in paths before punctuation', async () => {
837
- const fileContent = 'Spaced file content';
838
- const filePath = await createTestFile(
839
- path.join(testRootDir, 'spaced file.txt'),
840
- fileContent,
841
- );
842
- const escapedPath = path.join(testRootDir, 'spaced\\ file.txt');
843
- const query = `Check @${escapedPath}, it has spaces.`;
844
-
845
- const result = await handleAtCommand({
846
- query,
847
- config: mockConfig,
848
- addItem: mockAddItem,
849
- onDebugMessage: mockOnDebugMessage,
850
- messageId: 412,
851
- signal: abortController.signal,
852
- });
853
-
854
- expect(result).toEqual({
855
- processedQuery: [
856
- { text: `Check @${filePath}, it has spaces.` },
857
- { text: '\n--- Content from referenced files ---' },
858
- { text: `\nContent from @${filePath}:\n` },
859
- { text: fileContent },
860
- { text: '\n--- End of content ---' },
861
- ],
862
- shouldProceed: true,
863
- });
864
- });
865
-
866
- it('should not break file paths with periods in extensions', async () => {
867
- const fileContent = 'TypeScript content';
868
- const filePath = await createTestFile(
869
- path.join(testRootDir, 'example.d.ts'),
870
- fileContent,
871
- );
872
- const query = `Analyze @${filePath} for type definitions.`;
873
-
874
- const result = await handleAtCommand({
875
- query,
876
- config: mockConfig,
877
- addItem: mockAddItem,
878
- onDebugMessage: mockOnDebugMessage,
879
- messageId: 413,
880
- signal: abortController.signal,
881
- });
882
-
883
- expect(result).toEqual({
884
- processedQuery: [
885
- { text: `Analyze @${filePath} for type definitions.` },
886
- { text: '\n--- Content from referenced files ---' },
887
- { text: `\nContent from @${filePath}:\n` },
888
- { text: fileContent },
889
- { text: '\n--- End of content ---' },
890
- ],
891
- shouldProceed: true,
892
- });
893
- });
894
-
895
- it('should handle file paths ending with period followed by space', async () => {
896
- const fileContent = 'Config content';
897
- const filePath = await createTestFile(
898
- path.join(testRootDir, 'config.json'),
899
- fileContent,
900
- );
901
- const query = `Check @${filePath}. This file contains settings.`;
902
-
903
- const result = await handleAtCommand({
904
- query,
905
- config: mockConfig,
906
- addItem: mockAddItem,
907
- onDebugMessage: mockOnDebugMessage,
908
- messageId: 414,
909
- signal: abortController.signal,
910
- });
911
-
912
- expect(result).toEqual({
913
- processedQuery: [
914
- { text: `Check @${filePath}. This file contains settings.` },
915
- { text: '\n--- Content from referenced files ---' },
916
- { text: `\nContent from @${filePath}:\n` },
917
- { text: fileContent },
918
- { text: '\n--- End of content ---' },
919
- ],
920
- shouldProceed: true,
921
- });
922
- });
923
-
924
- it('should handle comma termination with complex file paths', async () => {
925
- const fileContent = 'Package info';
926
- const filePath = await createTestFile(
927
- path.join(testRootDir, 'package.json'),
928
- fileContent,
929
- );
930
- const query = `Review @${filePath}, then check dependencies.`;
931
-
932
- const result = await handleAtCommand({
933
- query,
934
- config: mockConfig,
935
- addItem: mockAddItem,
936
- onDebugMessage: mockOnDebugMessage,
937
- messageId: 415,
938
- signal: abortController.signal,
939
- });
940
-
941
- expect(result).toEqual({
942
- processedQuery: [
943
- { text: `Review @${filePath}, then check dependencies.` },
944
- { text: '\n--- Content from referenced files ---' },
945
- { text: `\nContent from @${filePath}:\n` },
946
- { text: fileContent },
947
- { text: '\n--- End of content ---' },
948
- ],
949
- shouldProceed: true,
950
- });
951
- });
952
-
953
- it('should not terminate at period within file name', async () => {
954
- const fileContent = 'Version info';
955
- const filePath = await createTestFile(
956
- path.join(testRootDir, 'version.1.2.3.txt'),
957
- fileContent,
958
- );
959
- const query = `Check @${filePath} contains version information.`;
960
-
961
- const result = await handleAtCommand({
962
- query,
963
- config: mockConfig,
964
- addItem: mockAddItem,
965
- onDebugMessage: mockOnDebugMessage,
966
- messageId: 416,
967
- signal: abortController.signal,
968
- });
969
-
970
- expect(result).toEqual({
971
- processedQuery: [
972
- { text: `Check @${filePath} contains version information.` },
973
- { text: '\n--- Content from referenced files ---' },
974
- { text: `\nContent from @${filePath}:\n` },
975
- { text: fileContent },
976
- { text: '\n--- End of content ---' },
977
- ],
978
- shouldProceed: true,
979
- });
980
- });
981
-
982
- it('should handle end of string termination for period and comma', async () => {
983
- const fileContent = 'End file content';
984
- const filePath = await createTestFile(
985
- path.join(testRootDir, 'end.txt'),
986
- fileContent,
987
- );
988
- const query = `Show me @${filePath}.`;
989
-
990
- const result = await handleAtCommand({
991
- query,
992
- config: mockConfig,
993
- addItem: mockAddItem,
994
- onDebugMessage: mockOnDebugMessage,
995
- messageId: 417,
996
- signal: abortController.signal,
997
- });
998
-
999
- expect(result).toEqual({
1000
- processedQuery: [
1001
- { text: `Show me @${filePath}.` },
1002
- { text: '\n--- Content from referenced files ---' },
1003
- { text: `\nContent from @${filePath}:\n` },
1004
- { text: fileContent },
1005
- { text: '\n--- End of content ---' },
1006
- ],
1007
- shouldProceed: true,
1008
- });
1009
- });
1010
-
1011
- it('should handle files with special characters in names', async () => {
1012
- const fileContent = 'File with special chars content';
1013
- const filePath = await createTestFile(
1014
- path.join(testRootDir, 'file$with&special#chars.txt'),
1015
- fileContent,
1016
- );
1017
- const query = `Check @${filePath} for content.`;
1018
-
1019
- const result = await handleAtCommand({
1020
- query,
1021
- config: mockConfig,
1022
- addItem: mockAddItem,
1023
- onDebugMessage: mockOnDebugMessage,
1024
- messageId: 418,
1025
- signal: abortController.signal,
1026
- });
1027
-
1028
- expect(result).toEqual({
1029
- processedQuery: [
1030
- { text: `Check @${filePath} for content.` },
1031
- { text: '\n--- Content from referenced files ---' },
1032
- { text: `\nContent from @${filePath}:\n` },
1033
- { text: fileContent },
1034
- { text: '\n--- End of content ---' },
1035
- ],
1036
- shouldProceed: true,
1037
- });
1038
- });
1039
-
1040
- it('should handle basic file names without special characters', async () => {
1041
- const fileContent = 'Basic file content';
1042
- const filePath = await createTestFile(
1043
- path.join(testRootDir, 'basicfile.txt'),
1044
- fileContent,
1045
- );
1046
- const query = `Check @${filePath} please.`;
1047
-
1048
- const result = await handleAtCommand({
1049
- query,
1050
- config: mockConfig,
1051
- addItem: mockAddItem,
1052
- onDebugMessage: mockOnDebugMessage,
1053
- messageId: 421,
1054
- signal: abortController.signal,
1055
- });
1056
-
1057
- expect(result).toEqual({
1058
- processedQuery: [
1059
- { text: `Check @${filePath} please.` },
1060
- { text: '\n--- Content from referenced files ---' },
1061
- { text: `\nContent from @${filePath}:\n` },
1062
- { text: fileContent },
1063
- { text: '\n--- End of content ---' },
1064
- ],
1065
- shouldProceed: true,
1066
- });
1067
- });
1068
- });
1069
-
1070
- it("should not add the user's turn to history, as that is the caller's responsibility", async () => {
1071
- // Arrange
1072
- const fileContent = 'This is the file content.';
1073
- const filePath = await createTestFile(
1074
- path.join(testRootDir, 'path', 'to', 'another-file.txt'),
1075
- fileContent,
1076
- );
1077
- const query = `A query with @${filePath}`;
1078
-
1079
- // Act
1080
- await handleAtCommand({
1081
- query,
1082
- config: mockConfig,
1083
- addItem: mockAddItem,
1084
- onDebugMessage: mockOnDebugMessage,
1085
- messageId: 999,
1086
- signal: abortController.signal,
1087
- });
1088
-
1089
- // Assert
1090
- // It SHOULD be called for the tool_group
1091
- expect(mockAddItem).toHaveBeenCalledWith(
1092
- expect.objectContaining({ type: 'tool_group' }),
1093
- 999,
1094
- );
1095
-
1096
- // It should NOT have been called for the user turn
1097
- const userTurnCalls = mockAddItem.mock.calls.filter(
1098
- (call) => call[0].type === 'user',
1099
- );
1100
- expect(userTurnCalls).toHaveLength(0);
1101
- });
1102
- });