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,1057 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
-
7
- import { vi, describe, it, expect, beforeEach } from 'vitest';
8
- import { mcpCommand } from './mcpCommand.js';
9
- import { createMockCommandContext } from '../../test-utils/mockCommandContext.js';
10
- import {
11
- MCPServerStatus,
12
- MCPDiscoveryState,
13
- getMCPServerStatus,
14
- getMCPDiscoveryState,
15
- DiscoveredMCPTool,
16
- } from 'fss-link-core';
17
-
18
- import { MessageActionReturn } from './types.js';
19
- import { Type, CallableTool } from '@google/genai';
20
-
21
- vi.mock('fss-link-core', async (importOriginal) => {
22
- const actual =
23
- await importOriginal<typeof import('fss-link-core')>();
24
- return {
25
- ...actual,
26
- getMCPServerStatus: vi.fn(),
27
- getMCPDiscoveryState: vi.fn(),
28
- MCPOAuthProvider: {
29
- authenticate: vi.fn(),
30
- },
31
- MCPOAuthTokenStorage: {
32
- getToken: vi.fn(),
33
- isTokenExpired: vi.fn(),
34
- },
35
- };
36
- });
37
-
38
- // Helper function to check if result is a message action
39
- const isMessageAction = (result: unknown): result is MessageActionReturn =>
40
- result !== null &&
41
- typeof result === 'object' &&
42
- 'type' in result &&
43
- result.type === 'message';
44
-
45
- // Helper function to create a mock DiscoveredMCPTool
46
- const createMockMCPTool = (
47
- name: string,
48
- serverName: string,
49
- description?: string,
50
- ) =>
51
- new DiscoveredMCPTool(
52
- {
53
- callTool: vi.fn(),
54
- tool: vi.fn(),
55
- } as unknown as CallableTool,
56
- serverName,
57
- name,
58
- description || `Description for ${name}`,
59
- { type: Type.OBJECT, properties: {} },
60
- name, // serverToolName same as name for simplicity
61
- );
62
-
63
- describe('mcpCommand', () => {
64
- let mockContext: ReturnType<typeof createMockCommandContext>;
65
- let mockConfig: {
66
- getToolRegistry: ReturnType<typeof vi.fn>;
67
- getMcpServers: ReturnType<typeof vi.fn>;
68
- getBlockedMcpServers: ReturnType<typeof vi.fn>;
69
- getPromptRegistry: ReturnType<typeof vi.fn>;
70
- };
71
-
72
- beforeEach(() => {
73
- vi.clearAllMocks();
74
-
75
- // Set up default mock environment
76
- vi.unstubAllEnvs();
77
-
78
- // Default mock implementations
79
- vi.mocked(getMCPServerStatus).mockReturnValue(MCPServerStatus.CONNECTED);
80
- vi.mocked(getMCPDiscoveryState).mockReturnValue(
81
- MCPDiscoveryState.COMPLETED,
82
- );
83
-
84
- // Create mock config with all necessary methods
85
- mockConfig = {
86
- getToolRegistry: vi.fn().mockReturnValue({
87
- getAllTools: vi.fn().mockReturnValue([]),
88
- }),
89
- getMcpServers: vi.fn().mockReturnValue({}),
90
- getBlockedMcpServers: vi.fn().mockReturnValue([]),
91
- getPromptRegistry: vi.fn().mockResolvedValue({
92
- getAllPrompts: vi.fn().mockReturnValue([]),
93
- getPromptsByServer: vi.fn().mockReturnValue([]),
94
- }),
95
- };
96
-
97
- mockContext = createMockCommandContext({
98
- services: {
99
- config: mockConfig,
100
- },
101
- });
102
- });
103
-
104
- describe('basic functionality', () => {
105
- it('should show an error if config is not available', async () => {
106
- const contextWithoutConfig = createMockCommandContext({
107
- services: {
108
- config: null,
109
- },
110
- });
111
-
112
- const result = await mcpCommand.action!(contextWithoutConfig, '');
113
-
114
- expect(result).toEqual({
115
- type: 'message',
116
- messageType: 'error',
117
- content: 'Config not loaded.',
118
- });
119
- });
120
-
121
- it('should show an error if tool registry is not available', async () => {
122
- mockConfig.getToolRegistry = vi.fn().mockReturnValue(undefined);
123
-
124
- const result = await mcpCommand.action!(mockContext, '');
125
-
126
- expect(result).toEqual({
127
- type: 'message',
128
- messageType: 'error',
129
- content: 'Could not retrieve tool registry.',
130
- });
131
- });
132
- });
133
-
134
- describe('no MCP servers configured', () => {
135
- beforeEach(() => {
136
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
137
- getAllTools: vi.fn().mockReturnValue([]),
138
- });
139
- mockConfig.getMcpServers = vi.fn().mockReturnValue({});
140
- });
141
-
142
- it('should display a message with a URL when no MCP servers are configured', async () => {
143
- const result = await mcpCommand.action!(mockContext, '');
144
-
145
- expect(result).toEqual({
146
- type: 'message',
147
- messageType: 'info',
148
- content:
149
- 'No MCP servers configured. Please view MCP documentation in your browser: https://docs.anthropic.com/en/docs/claude-code/tools/mcp-server/#how-to-set-up-your-mcp-server or use the cli /docs command',
150
- });
151
- });
152
- });
153
-
154
- describe('with configured MCP servers', () => {
155
- beforeEach(() => {
156
- const mockMcpServers = {
157
- server1: { command: 'cmd1' },
158
- server2: { command: 'cmd2' },
159
- server3: { command: 'cmd3' },
160
- };
161
-
162
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
163
- });
164
-
165
- it('should display configured MCP servers with status indicators and their tools', async () => {
166
- // Setup getMCPServerStatus mock implementation
167
- vi.mocked(getMCPServerStatus).mockImplementation((serverName) => {
168
- if (serverName === 'server1') return MCPServerStatus.CONNECTED;
169
- if (serverName === 'server2') return MCPServerStatus.CONNECTED;
170
- return MCPServerStatus.DISCONNECTED; // server3
171
- });
172
-
173
- // Mock tools from each server using actual DiscoveredMCPTool instances
174
- const mockServer1Tools = [
175
- createMockMCPTool('server1_tool1', 'server1'),
176
- createMockMCPTool('server1_tool2', 'server1'),
177
- ];
178
- const mockServer2Tools = [createMockMCPTool('server2_tool1', 'server2')];
179
- const mockServer3Tools = [createMockMCPTool('server3_tool1', 'server3')];
180
-
181
- const allTools = [
182
- ...mockServer1Tools,
183
- ...mockServer2Tools,
184
- ...mockServer3Tools,
185
- ];
186
-
187
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
188
- getAllTools: vi.fn().mockReturnValue(allTools),
189
- });
190
-
191
- const result = await mcpCommand.action!(mockContext, '');
192
-
193
- expect(result).toEqual({
194
- type: 'message',
195
- messageType: 'info',
196
- content: expect.stringContaining('Configured MCP servers:'),
197
- });
198
-
199
- expect(isMessageAction(result)).toBe(true);
200
- if (isMessageAction(result)) {
201
- const message = result.content;
202
- // Server 1 - Connected
203
- expect(message).toContain(
204
- '🟢 \u001b[1mserver1\u001b[0m - Ready (2 tools)',
205
- );
206
- expect(message).toContain('server1_tool1');
207
- expect(message).toContain('server1_tool2');
208
-
209
- // Server 2 - Connected
210
- expect(message).toContain(
211
- '🟢 \u001b[1mserver2\u001b[0m - Ready (1 tool)',
212
- );
213
- expect(message).toContain('server2_tool1');
214
-
215
- // Server 3 - Disconnected but with cached tools, so shows as Ready
216
- expect(message).toContain(
217
- '🟢 \u001b[1mserver3\u001b[0m - Ready (1 tool)',
218
- );
219
- expect(message).toContain('server3_tool1');
220
-
221
- // Check that helpful tips are displayed when no arguments are provided
222
- expect(message).toContain('💡 Tips:');
223
- expect(message).toContain('/mcp desc');
224
- expect(message).toContain('/mcp schema');
225
- expect(message).toContain('/mcp nodesc');
226
- expect(message).toContain('Ctrl+T');
227
- }
228
- });
229
-
230
- it('should display tool descriptions when desc argument is used', async () => {
231
- const mockMcpServers = {
232
- server1: {
233
- command: 'cmd1',
234
- description: 'This is a server description',
235
- },
236
- };
237
-
238
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
239
-
240
- // Mock tools with descriptions using actual DiscoveredMCPTool instances
241
- const mockServerTools = [
242
- createMockMCPTool('tool1', 'server1', 'This is tool 1 description'),
243
- createMockMCPTool('tool2', 'server1', 'This is tool 2 description'),
244
- ];
245
-
246
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
247
- getAllTools: vi.fn().mockReturnValue(mockServerTools),
248
- });
249
-
250
- const result = await mcpCommand.action!(mockContext, 'desc');
251
-
252
- expect(result).toEqual({
253
- type: 'message',
254
- messageType: 'info',
255
- content: expect.stringContaining('Configured MCP servers:'),
256
- });
257
-
258
- expect(isMessageAction(result)).toBe(true);
259
- if (isMessageAction(result)) {
260
- const message = result.content;
261
-
262
- // Check that server description is included
263
- expect(message).toContain(
264
- '\u001b[1mserver1\u001b[0m - Ready (2 tools)',
265
- );
266
- expect(message).toContain(
267
- '\u001b[32mThis is a server description\u001b[0m',
268
- );
269
-
270
- // Check that tool descriptions are included
271
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
272
- expect(message).toContain(
273
- '\u001b[32mThis is tool 1 description\u001b[0m',
274
- );
275
- expect(message).toContain('\u001b[36mtool2\u001b[0m');
276
- expect(message).toContain(
277
- '\u001b[32mThis is tool 2 description\u001b[0m',
278
- );
279
-
280
- // Check that tips are NOT displayed when arguments are provided
281
- expect(message).not.toContain('💡 Tips:');
282
- }
283
- });
284
-
285
- it('should not display descriptions when nodesc argument is used', async () => {
286
- const mockMcpServers = {
287
- server1: {
288
- command: 'cmd1',
289
- description: 'This is a server description',
290
- },
291
- };
292
-
293
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
294
-
295
- const mockServerTools = [
296
- createMockMCPTool('tool1', 'server1', 'This is tool 1 description'),
297
- ];
298
-
299
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
300
- getAllTools: vi.fn().mockReturnValue(mockServerTools),
301
- });
302
-
303
- const result = await mcpCommand.action!(mockContext, 'nodesc');
304
-
305
- expect(result).toEqual({
306
- type: 'message',
307
- messageType: 'info',
308
- content: expect.stringContaining('Configured MCP servers:'),
309
- });
310
-
311
- expect(isMessageAction(result)).toBe(true);
312
- if (isMessageAction(result)) {
313
- const message = result.content;
314
-
315
- // Check that descriptions are not included
316
- expect(message).not.toContain('This is a server description');
317
- expect(message).not.toContain('This is tool 1 description');
318
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
319
-
320
- // Check that tips are NOT displayed when arguments are provided
321
- expect(message).not.toContain('💡 Tips:');
322
- }
323
- });
324
-
325
- it('should indicate when a server has no tools', async () => {
326
- const mockMcpServers = {
327
- server1: { command: 'cmd1' },
328
- server2: { command: 'cmd2' },
329
- };
330
-
331
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
332
-
333
- // Setup server statuses
334
- vi.mocked(getMCPServerStatus).mockImplementation((serverName) => {
335
- if (serverName === 'server1') return MCPServerStatus.CONNECTED;
336
- if (serverName === 'server2') return MCPServerStatus.DISCONNECTED;
337
- return MCPServerStatus.DISCONNECTED;
338
- });
339
-
340
- // Mock tools - only server1 has tools
341
- const mockServerTools = [createMockMCPTool('server1_tool1', 'server1')];
342
-
343
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
344
- getAllTools: vi.fn().mockReturnValue(mockServerTools),
345
- });
346
-
347
- const result = await mcpCommand.action!(mockContext, '');
348
-
349
- expect(isMessageAction(result)).toBe(true);
350
- if (isMessageAction(result)) {
351
- const message = result.content;
352
- expect(message).toContain(
353
- '🟢 \u001b[1mserver1\u001b[0m - Ready (1 tool)',
354
- );
355
- expect(message).toContain('\u001b[36mserver1_tool1\u001b[0m');
356
- expect(message).toContain(
357
- '🔴 \u001b[1mserver2\u001b[0m - Disconnected (0 tools cached)',
358
- );
359
- expect(message).toContain('No tools or prompts available');
360
- }
361
- });
362
-
363
- it('should show startup indicator when servers are connecting', async () => {
364
- const mockMcpServers = {
365
- server1: { command: 'cmd1' },
366
- server2: { command: 'cmd2' },
367
- };
368
-
369
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
370
-
371
- // Setup server statuses with one connecting
372
- vi.mocked(getMCPServerStatus).mockImplementation((serverName) => {
373
- if (serverName === 'server1') return MCPServerStatus.CONNECTED;
374
- if (serverName === 'server2') return MCPServerStatus.CONNECTING;
375
- return MCPServerStatus.DISCONNECTED;
376
- });
377
-
378
- // Setup discovery state as in progress
379
- vi.mocked(getMCPDiscoveryState).mockReturnValue(
380
- MCPDiscoveryState.IN_PROGRESS,
381
- );
382
-
383
- // Mock tools
384
- const mockServerTools = [
385
- createMockMCPTool('server1_tool1', 'server1'),
386
- createMockMCPTool('server2_tool1', 'server2'),
387
- ];
388
-
389
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
390
- getAllTools: vi.fn().mockReturnValue(mockServerTools),
391
- });
392
-
393
- const result = await mcpCommand.action!(mockContext, '');
394
-
395
- expect(isMessageAction(result)).toBe(true);
396
- if (isMessageAction(result)) {
397
- const message = result.content;
398
-
399
- // Check that startup indicator is shown
400
- expect(message).toContain(
401
- '⏳ MCP servers are starting up (1 initializing)...',
402
- );
403
- expect(message).toContain(
404
- 'Note: First startup may take longer. Tool availability will update automatically.',
405
- );
406
-
407
- // Check server statuses
408
- expect(message).toContain(
409
- '🟢 \u001b[1mserver1\u001b[0m - Ready (1 tool)',
410
- );
411
- expect(message).toContain(
412
- '🔄 \u001b[1mserver2\u001b[0m - Starting... (first startup may take longer) (tools and prompts will appear when ready)',
413
- );
414
- }
415
- });
416
-
417
- it('should display the extension name for servers from extensions', async () => {
418
- const mockMcpServers = {
419
- server1: { command: 'cmd1', extensionName: 'my-extension' },
420
- };
421
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
422
-
423
- const result = await mcpCommand.action!(mockContext, '');
424
-
425
- expect(isMessageAction(result)).toBe(true);
426
- if (isMessageAction(result)) {
427
- const message = result.content;
428
- expect(message).toContain('server1 (from my-extension)');
429
- }
430
- });
431
-
432
- it('should display blocked MCP servers', async () => {
433
- mockConfig.getMcpServers = vi.fn().mockReturnValue({});
434
- const blockedServers = [
435
- { name: 'blocked-server', extensionName: 'my-extension' },
436
- ];
437
- mockConfig.getBlockedMcpServers = vi.fn().mockReturnValue(blockedServers);
438
-
439
- const result = await mcpCommand.action!(mockContext, '');
440
-
441
- expect(isMessageAction(result)).toBe(true);
442
- if (isMessageAction(result)) {
443
- const message = result.content;
444
- expect(message).toContain(
445
- '🔴 \u001b[1mblocked-server (from my-extension)\u001b[0m - Blocked',
446
- );
447
- }
448
- });
449
-
450
- it('should display both active and blocked servers correctly', async () => {
451
- const mockMcpServers = {
452
- server1: { command: 'cmd1', extensionName: 'my-extension' },
453
- };
454
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
455
- const blockedServers = [
456
- { name: 'blocked-server', extensionName: 'another-extension' },
457
- ];
458
- mockConfig.getBlockedMcpServers = vi.fn().mockReturnValue(blockedServers);
459
-
460
- const result = await mcpCommand.action!(mockContext, '');
461
-
462
- expect(isMessageAction(result)).toBe(true);
463
- if (isMessageAction(result)) {
464
- const message = result.content;
465
- expect(message).toContain('server1 (from my-extension)');
466
- expect(message).toContain(
467
- '🔴 \u001b[1mblocked-server (from another-extension)\u001b[0m - Blocked',
468
- );
469
- }
470
- });
471
- });
472
-
473
- describe('schema functionality', () => {
474
- it('should display tool schemas when schema argument is used', async () => {
475
- const mockMcpServers = {
476
- server1: {
477
- command: 'cmd1',
478
- description: 'This is a server description',
479
- },
480
- };
481
-
482
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
483
-
484
- // Create tools with parameter schemas
485
- const mockCallableTool1: CallableTool = {
486
- callTool: vi.fn(),
487
- tool: vi.fn(),
488
- } as unknown as CallableTool;
489
- const mockCallableTool2: CallableTool = {
490
- callTool: vi.fn(),
491
- tool: vi.fn(),
492
- } as unknown as CallableTool;
493
-
494
- const tool1 = new DiscoveredMCPTool(
495
- mockCallableTool1,
496
- 'server1',
497
- 'tool1',
498
- 'This is tool 1 description',
499
- {
500
- type: Type.OBJECT,
501
- properties: {
502
- param1: { type: Type.STRING, description: 'First parameter' },
503
- },
504
- required: ['param1'],
505
- },
506
- 'tool1',
507
- );
508
-
509
- const tool2 = new DiscoveredMCPTool(
510
- mockCallableTool2,
511
- 'server1',
512
- 'tool2',
513
- 'This is tool 2 description',
514
- {
515
- type: Type.OBJECT,
516
- properties: {
517
- param2: { type: Type.NUMBER, description: 'Second parameter' },
518
- },
519
- required: ['param2'],
520
- },
521
- 'tool2',
522
- );
523
-
524
- const mockServerTools = [tool1, tool2];
525
-
526
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
527
- getAllTools: vi.fn().mockReturnValue(mockServerTools),
528
- });
529
-
530
- const result = await mcpCommand.action!(mockContext, 'schema');
531
-
532
- expect(result).toEqual({
533
- type: 'message',
534
- messageType: 'info',
535
- content: expect.stringContaining('Configured MCP servers:'),
536
- });
537
-
538
- expect(isMessageAction(result)).toBe(true);
539
- if (isMessageAction(result)) {
540
- const message = result.content;
541
-
542
- // Check that server description is included
543
- expect(message).toContain('Ready (2 tools)');
544
- expect(message).toContain('This is a server description');
545
-
546
- // Check that tool descriptions and schemas are included
547
- expect(message).toContain('This is tool 1 description');
548
- expect(message).toContain('Parameters:');
549
- expect(message).toContain('param1');
550
- expect(message).toContain('STRING');
551
- expect(message).toContain('This is tool 2 description');
552
- expect(message).toContain('param2');
553
- expect(message).toContain('NUMBER');
554
- }
555
- });
556
-
557
- it('should handle tools without parameter schemas gracefully', async () => {
558
- const mockMcpServers = {
559
- server1: { command: 'cmd1' },
560
- };
561
-
562
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
563
-
564
- // Mock tools without parameter schemas
565
- const mockServerTools = [
566
- createMockMCPTool('tool1', 'server1', 'Tool without schema'),
567
- ];
568
-
569
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
570
- getAllTools: vi.fn().mockReturnValue(mockServerTools),
571
- });
572
-
573
- const result = await mcpCommand.action!(mockContext, 'schema');
574
-
575
- expect(result).toEqual({
576
- type: 'message',
577
- messageType: 'info',
578
- content: expect.stringContaining('Configured MCP servers:'),
579
- });
580
-
581
- expect(isMessageAction(result)).toBe(true);
582
- if (isMessageAction(result)) {
583
- const message = result.content;
584
- expect(message).toContain('tool1');
585
- expect(message).toContain('Tool without schema');
586
- // Should not crash when parameterSchema is undefined
587
- }
588
- });
589
- });
590
-
591
- describe('argument parsing', () => {
592
- beforeEach(() => {
593
- const mockMcpServers = {
594
- server1: {
595
- command: 'cmd1',
596
- description: 'Server description',
597
- },
598
- };
599
-
600
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
601
-
602
- const mockServerTools = [
603
- createMockMCPTool('tool1', 'server1', 'Test tool'),
604
- ];
605
-
606
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
607
- getAllTools: vi.fn().mockReturnValue(mockServerTools),
608
- });
609
- });
610
-
611
- it('should handle "descriptions" as alias for "desc"', async () => {
612
- const result = await mcpCommand.action!(mockContext, 'descriptions');
613
-
614
- expect(isMessageAction(result)).toBe(true);
615
- if (isMessageAction(result)) {
616
- const message = result.content;
617
- expect(message).toContain('Test tool');
618
- expect(message).toContain('Server description');
619
- }
620
- });
621
-
622
- it('should handle "nodescriptions" as alias for "nodesc"', async () => {
623
- const result = await mcpCommand.action!(mockContext, 'nodescriptions');
624
-
625
- expect(isMessageAction(result)).toBe(true);
626
- if (isMessageAction(result)) {
627
- const message = result.content;
628
- expect(message).not.toContain('Test tool');
629
- expect(message).not.toContain('Server description');
630
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
631
- }
632
- });
633
-
634
- it('should handle mixed case arguments', async () => {
635
- const result = await mcpCommand.action!(mockContext, 'DESC');
636
-
637
- expect(isMessageAction(result)).toBe(true);
638
- if (isMessageAction(result)) {
639
- const message = result.content;
640
- expect(message).toContain('Test tool');
641
- expect(message).toContain('Server description');
642
- }
643
- });
644
-
645
- it('should handle multiple arguments - "schema desc"', async () => {
646
- const result = await mcpCommand.action!(mockContext, 'schema desc');
647
-
648
- expect(isMessageAction(result)).toBe(true);
649
- if (isMessageAction(result)) {
650
- const message = result.content;
651
- expect(message).toContain('Test tool');
652
- expect(message).toContain('Server description');
653
- expect(message).toContain('Parameters:');
654
- }
655
- });
656
-
657
- it('should handle multiple arguments - "desc schema"', async () => {
658
- const result = await mcpCommand.action!(mockContext, 'desc schema');
659
-
660
- expect(isMessageAction(result)).toBe(true);
661
- if (isMessageAction(result)) {
662
- const message = result.content;
663
- expect(message).toContain('Test tool');
664
- expect(message).toContain('Server description');
665
- expect(message).toContain('Parameters:');
666
- }
667
- });
668
-
669
- it('should handle "schema" alone showing descriptions', async () => {
670
- const result = await mcpCommand.action!(mockContext, 'schema');
671
-
672
- expect(isMessageAction(result)).toBe(true);
673
- if (isMessageAction(result)) {
674
- const message = result.content;
675
- expect(message).toContain('Test tool');
676
- expect(message).toContain('Server description');
677
- expect(message).toContain('Parameters:');
678
- }
679
- });
680
-
681
- it('should handle "nodesc" overriding "schema" - "schema nodesc"', async () => {
682
- const result = await mcpCommand.action!(mockContext, 'schema nodesc');
683
-
684
- expect(isMessageAction(result)).toBe(true);
685
- if (isMessageAction(result)) {
686
- const message = result.content;
687
- expect(message).not.toContain('Test tool');
688
- expect(message).not.toContain('Server description');
689
- expect(message).toContain('Parameters:'); // Schema should still show
690
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
691
- }
692
- });
693
-
694
- it('should handle "nodesc" overriding "desc" - "desc nodesc"', async () => {
695
- const result = await mcpCommand.action!(mockContext, 'desc nodesc');
696
-
697
- expect(isMessageAction(result)).toBe(true);
698
- if (isMessageAction(result)) {
699
- const message = result.content;
700
- expect(message).not.toContain('Test tool');
701
- expect(message).not.toContain('Server description');
702
- expect(message).not.toContain('Parameters:');
703
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
704
- }
705
- });
706
-
707
- it('should handle "nodesc" overriding both "desc" and "schema" - "desc schema nodesc"', async () => {
708
- const result = await mcpCommand.action!(
709
- mockContext,
710
- 'desc schema nodesc',
711
- );
712
-
713
- expect(isMessageAction(result)).toBe(true);
714
- if (isMessageAction(result)) {
715
- const message = result.content;
716
- expect(message).not.toContain('Test tool');
717
- expect(message).not.toContain('Server description');
718
- expect(message).toContain('Parameters:'); // Schema should still show
719
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
720
- }
721
- });
722
-
723
- it('should handle extra whitespace in arguments', async () => {
724
- const result = await mcpCommand.action!(mockContext, ' desc schema ');
725
-
726
- expect(isMessageAction(result)).toBe(true);
727
- if (isMessageAction(result)) {
728
- const message = result.content;
729
- expect(message).toContain('Test tool');
730
- expect(message).toContain('Server description');
731
- expect(message).toContain('Parameters:');
732
- }
733
- });
734
-
735
- it('should handle empty arguments gracefully', async () => {
736
- const result = await mcpCommand.action!(mockContext, '');
737
-
738
- expect(isMessageAction(result)).toBe(true);
739
- if (isMessageAction(result)) {
740
- const message = result.content;
741
- expect(message).not.toContain('Test tool');
742
- expect(message).not.toContain('Server description');
743
- expect(message).not.toContain('Parameters:');
744
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
745
- }
746
- });
747
-
748
- it('should handle unknown arguments gracefully', async () => {
749
- const result = await mcpCommand.action!(mockContext, 'unknown arg');
750
-
751
- expect(isMessageAction(result)).toBe(true);
752
- if (isMessageAction(result)) {
753
- const message = result.content;
754
- expect(message).not.toContain('Test tool');
755
- expect(message).not.toContain('Server description');
756
- expect(message).not.toContain('Parameters:');
757
- expect(message).toContain('\u001b[36mtool1\u001b[0m');
758
- }
759
- });
760
- });
761
-
762
- describe('edge cases', () => {
763
- it('should handle empty server names gracefully', async () => {
764
- const mockMcpServers = {
765
- '': { command: 'cmd1' }, // Empty server name
766
- };
767
-
768
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
769
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
770
- getAllTools: vi.fn().mockReturnValue([]),
771
- });
772
-
773
- const result = await mcpCommand.action!(mockContext, '');
774
-
775
- expect(result).toEqual({
776
- type: 'message',
777
- messageType: 'info',
778
- content: expect.stringContaining('Configured MCP servers:'),
779
- });
780
- });
781
-
782
- it('should handle servers with special characters in names', async () => {
783
- const mockMcpServers = {
784
- 'server-with-dashes': { command: 'cmd1' },
785
- server_with_underscores: { command: 'cmd2' },
786
- 'server.with.dots': { command: 'cmd3' },
787
- };
788
-
789
- mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
790
- mockConfig.getToolRegistry = vi.fn().mockReturnValue({
791
- getAllTools: vi.fn().mockReturnValue([]),
792
- });
793
-
794
- const result = await mcpCommand.action!(mockContext, '');
795
-
796
- expect(isMessageAction(result)).toBe(true);
797
- if (isMessageAction(result)) {
798
- const message = result.content;
799
- expect(message).toContain('server-with-dashes');
800
- expect(message).toContain('server_with_underscores');
801
- expect(message).toContain('server.with.dots');
802
- }
803
- });
804
- });
805
-
806
- describe('auth subcommand', () => {
807
- beforeEach(() => {
808
- vi.clearAllMocks();
809
- });
810
-
811
- it('should list OAuth-enabled servers when no server name is provided', async () => {
812
- const context = createMockCommandContext({
813
- services: {
814
- config: {
815
- getMcpServers: vi.fn().mockReturnValue({
816
- 'oauth-server': { oauth: { enabled: true } },
817
- 'regular-server': {},
818
- 'another-oauth': { oauth: { enabled: true } },
819
- }),
820
- },
821
- },
822
- });
823
-
824
- const authCommand = mcpCommand.subCommands?.find(
825
- (cmd) => cmd.name === 'auth',
826
- );
827
- expect(authCommand).toBeDefined();
828
-
829
- const result = await authCommand!.action!(context, '');
830
- expect(isMessageAction(result)).toBe(true);
831
- if (isMessageAction(result)) {
832
- expect(result.messageType).toBe('info');
833
- expect(result.content).toContain('oauth-server');
834
- expect(result.content).toContain('another-oauth');
835
- expect(result.content).not.toContain('regular-server');
836
- expect(result.content).toContain('/mcp auth <server-name>');
837
- }
838
- });
839
-
840
- it('should show message when no OAuth servers are configured', async () => {
841
- const context = createMockCommandContext({
842
- services: {
843
- config: {
844
- getMcpServers: vi.fn().mockReturnValue({
845
- 'regular-server': {},
846
- }),
847
- },
848
- },
849
- });
850
-
851
- const authCommand = mcpCommand.subCommands?.find(
852
- (cmd) => cmd.name === 'auth',
853
- );
854
- const result = await authCommand!.action!(context, '');
855
-
856
- expect(isMessageAction(result)).toBe(true);
857
- if (isMessageAction(result)) {
858
- expect(result.messageType).toBe('info');
859
- expect(result.content).toBe(
860
- 'No MCP servers configured with OAuth authentication.',
861
- );
862
- }
863
- });
864
-
865
- it('should authenticate with a specific server', async () => {
866
- const mockToolRegistry = {
867
- discoverToolsForServer: vi.fn(),
868
- };
869
- const mockGeminiClient = {
870
- setTools: vi.fn(),
871
- };
872
-
873
- const context = createMockCommandContext({
874
- services: {
875
- config: {
876
- getMcpServers: vi.fn().mockReturnValue({
877
- 'test-server': {
878
- url: 'http://localhost:3000',
879
- oauth: { enabled: true },
880
- },
881
- }),
882
- getToolRegistry: vi.fn().mockReturnValue(mockToolRegistry),
883
- getGeminiClient: vi.fn().mockReturnValue(mockGeminiClient),
884
- getPromptRegistry: vi.fn().mockResolvedValue({
885
- removePromptsByServer: vi.fn(),
886
- }),
887
- },
888
- },
889
- });
890
- // Mock the reloadCommands function
891
- context.ui.reloadCommands = vi.fn();
892
-
893
- const { MCPOAuthProvider } = await import('fss-link-core');
894
-
895
- const authCommand = mcpCommand.subCommands?.find(
896
- (cmd) => cmd.name === 'auth',
897
- );
898
- const result = await authCommand!.action!(context, 'test-server');
899
-
900
- expect(MCPOAuthProvider.authenticate).toHaveBeenCalledWith(
901
- 'test-server',
902
- { enabled: true },
903
- 'http://localhost:3000',
904
- );
905
- expect(mockToolRegistry.discoverToolsForServer).toHaveBeenCalledWith(
906
- 'test-server',
907
- );
908
- expect(mockGeminiClient.setTools).toHaveBeenCalled();
909
- expect(context.ui.reloadCommands).toHaveBeenCalledTimes(1);
910
-
911
- expect(isMessageAction(result)).toBe(true);
912
- if (isMessageAction(result)) {
913
- expect(result.messageType).toBe('info');
914
- expect(result.content).toContain('Successfully authenticated');
915
- }
916
- });
917
-
918
- it('should handle authentication errors', async () => {
919
- const context = createMockCommandContext({
920
- services: {
921
- config: {
922
- getMcpServers: vi.fn().mockReturnValue({
923
- 'test-server': { oauth: { enabled: true } },
924
- }),
925
- },
926
- },
927
- });
928
-
929
- const { MCPOAuthProvider } = await import('fss-link-core');
930
- (
931
- MCPOAuthProvider.authenticate as ReturnType<typeof vi.fn>
932
- ).mockRejectedValue(new Error('Auth failed'));
933
-
934
- const authCommand = mcpCommand.subCommands?.find(
935
- (cmd) => cmd.name === 'auth',
936
- );
937
- const result = await authCommand!.action!(context, 'test-server');
938
-
939
- expect(isMessageAction(result)).toBe(true);
940
- if (isMessageAction(result)) {
941
- expect(result.messageType).toBe('error');
942
- expect(result.content).toContain('Failed to authenticate');
943
- expect(result.content).toContain('Auth failed');
944
- }
945
- });
946
-
947
- it('should handle non-existent server', async () => {
948
- const context = createMockCommandContext({
949
- services: {
950
- config: {
951
- getMcpServers: vi.fn().mockReturnValue({
952
- 'existing-server': {},
953
- }),
954
- },
955
- },
956
- });
957
-
958
- const authCommand = mcpCommand.subCommands?.find(
959
- (cmd) => cmd.name === 'auth',
960
- );
961
- const result = await authCommand!.action!(context, 'non-existent');
962
-
963
- expect(isMessageAction(result)).toBe(true);
964
- if (isMessageAction(result)) {
965
- expect(result.messageType).toBe('error');
966
- expect(result.content).toContain("MCP server 'non-existent' not found");
967
- }
968
- });
969
- });
970
-
971
- describe('refresh subcommand', () => {
972
- it('should refresh the list of tools and display the status', async () => {
973
- const mockToolRegistry = {
974
- discoverMcpTools: vi.fn(),
975
- restartMcpServers: vi.fn(),
976
- getAllTools: vi.fn().mockReturnValue([]),
977
- };
978
- const mockGeminiClient = {
979
- setTools: vi.fn(),
980
- };
981
-
982
- const context = createMockCommandContext({
983
- services: {
984
- config: {
985
- getMcpServers: vi.fn().mockReturnValue({ server1: {} }),
986
- getBlockedMcpServers: vi.fn().mockReturnValue([]),
987
- getToolRegistry: vi.fn().mockReturnValue(mockToolRegistry),
988
- getGeminiClient: vi.fn().mockReturnValue(mockGeminiClient),
989
- getPromptRegistry: vi.fn().mockResolvedValue({
990
- getPromptsByServer: vi.fn().mockReturnValue([]),
991
- }),
992
- },
993
- },
994
- });
995
- // Mock the reloadCommands function, which is new logic.
996
- context.ui.reloadCommands = vi.fn();
997
-
998
- const refreshCommand = mcpCommand.subCommands?.find(
999
- (cmd) => cmd.name === 'refresh',
1000
- );
1001
- expect(refreshCommand).toBeDefined();
1002
-
1003
- const result = await refreshCommand!.action!(context, '');
1004
-
1005
- expect(context.ui.addItem).toHaveBeenCalledWith(
1006
- {
1007
- type: 'info',
1008
- text: 'Restarting MCP servers...',
1009
- },
1010
- expect.any(Number),
1011
- );
1012
- expect(mockToolRegistry.restartMcpServers).toHaveBeenCalled();
1013
- expect(mockGeminiClient.setTools).toHaveBeenCalled();
1014
- expect(context.ui.reloadCommands).toHaveBeenCalledTimes(1);
1015
-
1016
- expect(isMessageAction(result)).toBe(true);
1017
- if (isMessageAction(result)) {
1018
- expect(result.messageType).toBe('info');
1019
- expect(result.content).toContain('Configured MCP servers:');
1020
- }
1021
- });
1022
-
1023
- it('should show an error if config is not available', async () => {
1024
- const contextWithoutConfig = createMockCommandContext({
1025
- services: {
1026
- config: null,
1027
- },
1028
- });
1029
-
1030
- const refreshCommand = mcpCommand.subCommands?.find(
1031
- (cmd) => cmd.name === 'refresh',
1032
- );
1033
- const result = await refreshCommand!.action!(contextWithoutConfig, '');
1034
-
1035
- expect(result).toEqual({
1036
- type: 'message',
1037
- messageType: 'error',
1038
- content: 'Config not loaded.',
1039
- });
1040
- });
1041
-
1042
- it('should show an error if tool registry is not available', async () => {
1043
- mockConfig.getToolRegistry = vi.fn().mockReturnValue(undefined);
1044
-
1045
- const refreshCommand = mcpCommand.subCommands?.find(
1046
- (cmd) => cmd.name === 'refresh',
1047
- );
1048
- const result = await refreshCommand!.action!(mockContext, '');
1049
-
1050
- expect(result).toEqual({
1051
- type: 'message',
1052
- messageType: 'error',
1053
- content: 'Could not retrieve tool registry.',
1054
- });
1055
- });
1056
- });
1057
- });