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,1044 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
-
7
- const { logSlashCommand } = vi.hoisted(() => ({
8
- logSlashCommand: vi.fn(),
9
- }));
10
-
11
- vi.mock('fss-link-core', async (importOriginal) => {
12
- const original =
13
- await importOriginal<typeof import('fss-link-core')>();
14
- return {
15
- ...original,
16
- logSlashCommand,
17
- getIdeInstaller: vi.fn().mockReturnValue(null),
18
- };
19
- });
20
-
21
- const { mockProcessExit } = vi.hoisted(() => ({
22
- mockProcessExit: vi.fn((_code?: number): never => undefined as never),
23
- }));
24
-
25
- vi.mock('node:process', () => {
26
- const mockProcess: Partial<NodeJS.Process> = {
27
- exit: mockProcessExit,
28
- platform: 'sunos',
29
- } as unknown as NodeJS.Process;
30
- return {
31
- ...mockProcess,
32
- default: mockProcess,
33
- };
34
- });
35
-
36
- const mockBuiltinLoadCommands = vi.fn();
37
- vi.mock('../../services/BuiltinCommandLoader.js', () => ({
38
- BuiltinCommandLoader: vi.fn().mockImplementation(() => ({
39
- loadCommands: mockBuiltinLoadCommands,
40
- })),
41
- }));
42
-
43
- const mockFileLoadCommands = vi.fn();
44
- vi.mock('../../services/FileCommandLoader.js', () => ({
45
- FileCommandLoader: vi.fn().mockImplementation(() => ({
46
- loadCommands: mockFileLoadCommands,
47
- })),
48
- }));
49
-
50
- const mockMcpLoadCommands = vi.fn();
51
- vi.mock('../../services/McpPromptLoader.js', () => ({
52
- McpPromptLoader: vi.fn().mockImplementation(() => ({
53
- loadCommands: mockMcpLoadCommands,
54
- })),
55
- }));
56
-
57
- vi.mock('../contexts/SessionContext.js', () => ({
58
- useSessionStats: vi.fn(() => ({ stats: {} })),
59
- }));
60
-
61
- const { mockRunExitCleanup } = vi.hoisted(() => ({
62
- mockRunExitCleanup: vi.fn(),
63
- }));
64
-
65
- vi.mock('../../utils/cleanup.js', () => ({
66
- runExitCleanup: mockRunExitCleanup,
67
- }));
68
-
69
- import {
70
- SlashCommandStatus,
71
- ToolConfirmationOutcome,
72
- makeFakeConfig,
73
- } from 'fss-link-core';
74
- import { act, renderHook, waitFor } from '@testing-library/react';
75
- import { beforeEach, describe, expect, it, vi, type Mock } from 'vitest';
76
- import { LoadedSettings } from '../../config/settings.js';
77
- import { BuiltinCommandLoader } from '../../services/BuiltinCommandLoader.js';
78
- import { FileCommandLoader } from '../../services/FileCommandLoader.js';
79
- import { McpPromptLoader } from '../../services/McpPromptLoader.js';
80
- import {
81
- CommandContext,
82
- CommandKind,
83
- ConfirmShellCommandsActionReturn,
84
- SlashCommand,
85
- } from '../commands/types.js';
86
- import { MessageType } from '../types.js';
87
- import { useSlashCommandProcessor } from './slashCommandProcessor.js';
88
-
89
- function createTestCommand(
90
- overrides: Partial<SlashCommand>,
91
- kind: CommandKind = CommandKind.BUILT_IN,
92
- ): SlashCommand {
93
- return {
94
- name: 'test',
95
- description: 'a test command',
96
- kind,
97
- ...overrides,
98
- };
99
- }
100
-
101
- describe('useSlashCommandProcessor', () => {
102
- const mockAddItem = vi.fn();
103
- const mockClearItems = vi.fn();
104
- const mockLoadHistory = vi.fn();
105
- const mockOpenThemeDialog = vi.fn();
106
- const mockOpenAuthDialog = vi.fn();
107
- const mockSetQuittingMessages = vi.fn();
108
-
109
- const mockConfig = makeFakeConfig({});
110
-
111
- const mockSettings = {} as LoadedSettings;
112
-
113
- beforeEach(() => {
114
- vi.clearAllMocks();
115
- (vi.mocked(BuiltinCommandLoader) as Mock).mockClear();
116
- mockBuiltinLoadCommands.mockResolvedValue([]);
117
- mockFileLoadCommands.mockResolvedValue([]);
118
- mockMcpLoadCommands.mockResolvedValue([]);
119
- });
120
-
121
- const setupProcessorHook = (
122
- builtinCommands: SlashCommand[] = [],
123
- fileCommands: SlashCommand[] = [],
124
- mcpCommands: SlashCommand[] = [],
125
- setIsProcessing = vi.fn(),
126
- ) => {
127
- mockBuiltinLoadCommands.mockResolvedValue(Object.freeze(builtinCommands));
128
- mockFileLoadCommands.mockResolvedValue(Object.freeze(fileCommands));
129
- mockMcpLoadCommands.mockResolvedValue(Object.freeze(mcpCommands));
130
-
131
- const { result } = renderHook(() =>
132
- useSlashCommandProcessor(
133
- mockConfig,
134
- mockSettings,
135
- mockAddItem,
136
- mockClearItems,
137
- mockLoadHistory,
138
- vi.fn(), // refreshStatic
139
- vi.fn(), // onDebugMessage
140
- mockOpenThemeDialog, // openThemeDialog
141
- mockOpenAuthDialog,
142
- vi.fn(), // openEditorDialog
143
- vi.fn(), // openSearchDialog
144
- vi.fn(), // toggleCorgiMode
145
- mockSetQuittingMessages,
146
- vi.fn(), // openPrivacyNotice
147
- vi.fn(), // openSettingsDialog
148
- vi.fn(), // toggleVimEnabled
149
- setIsProcessing,
150
- ),
151
- );
152
-
153
- return result;
154
- };
155
-
156
- describe('Initialization and Command Loading', () => {
157
- it('should initialize CommandService with all required loaders', () => {
158
- setupProcessorHook();
159
- expect(BuiltinCommandLoader).toHaveBeenCalledWith(mockConfig);
160
- expect(FileCommandLoader).toHaveBeenCalledWith(mockConfig);
161
- expect(McpPromptLoader).toHaveBeenCalledWith(mockConfig);
162
- });
163
-
164
- it('should call loadCommands and populate state after mounting', async () => {
165
- const testCommand = createTestCommand({ name: 'test' });
166
- const result = setupProcessorHook([testCommand]);
167
-
168
- await waitFor(() => {
169
- expect(result.current.slashCommands).toHaveLength(1);
170
- });
171
-
172
- expect(result.current.slashCommands[0]?.name).toBe('test');
173
- expect(mockBuiltinLoadCommands).toHaveBeenCalledTimes(1);
174
- expect(mockFileLoadCommands).toHaveBeenCalledTimes(1);
175
- expect(mockMcpLoadCommands).toHaveBeenCalledTimes(1);
176
- });
177
-
178
- it('should provide an immutable array of commands to consumers', async () => {
179
- const testCommand = createTestCommand({ name: 'test' });
180
- const result = setupProcessorHook([testCommand]);
181
-
182
- await waitFor(() => {
183
- expect(result.current.slashCommands).toHaveLength(1);
184
- });
185
-
186
- const commands = result.current.slashCommands;
187
-
188
- expect(() => {
189
- // @ts-expect-error - We are intentionally testing a violation of the readonly type.
190
- commands.push(createTestCommand({ name: 'rogue' }));
191
- }).toThrow(TypeError);
192
- });
193
-
194
- it('should override built-in commands with file-based commands of the same name', async () => {
195
- const builtinAction = vi.fn();
196
- const fileAction = vi.fn();
197
-
198
- const builtinCommand = createTestCommand({
199
- name: 'override',
200
- description: 'builtin',
201
- action: builtinAction,
202
- });
203
- const fileCommand = createTestCommand(
204
- { name: 'override', description: 'file', action: fileAction },
205
- CommandKind.FILE,
206
- );
207
-
208
- const result = setupProcessorHook([builtinCommand], [fileCommand]);
209
-
210
- await waitFor(() => {
211
- // The service should only return one command with the name 'override'
212
- expect(result.current.slashCommands).toHaveLength(1);
213
- });
214
-
215
- await act(async () => {
216
- await result.current.handleSlashCommand('/override');
217
- });
218
-
219
- // Only the file-based command's action should be called.
220
- expect(fileAction).toHaveBeenCalledTimes(1);
221
- expect(builtinAction).not.toHaveBeenCalled();
222
- });
223
- });
224
-
225
- describe('Command Execution Logic', () => {
226
- it('should pass unknown commands to AI instead of showing error', async () => {
227
- const result = setupProcessorHook();
228
- await waitFor(() => expect(result.current.slashCommands).toBeDefined());
229
-
230
- const commandResult = await act(async () => await result.current.handleSlashCommand('/nonexistent'));
231
-
232
- // Should return false to let AI process the command
233
- expect(commandResult).toBe(false);
234
- // Should only add user input, no error message
235
- expect(mockAddItem).toHaveBeenCalledTimes(1);
236
- expect(mockAddItem).toHaveBeenCalledWith(
237
- {
238
- type: MessageType.USER,
239
- text: '/nonexistent',
240
- },
241
- expect.any(Number),
242
- );
243
- });
244
-
245
- it('should display help for a parent command invoked without a subcommand', async () => {
246
- const parentCommand: SlashCommand = {
247
- name: 'parent',
248
- description: 'a parent command',
249
- kind: CommandKind.BUILT_IN,
250
- subCommands: [
251
- {
252
- name: 'child1',
253
- description: 'First child.',
254
- kind: CommandKind.BUILT_IN,
255
- },
256
- ],
257
- };
258
- const result = setupProcessorHook([parentCommand]);
259
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
260
-
261
- await act(async () => {
262
- await result.current.handleSlashCommand('/parent');
263
- });
264
-
265
- expect(mockAddItem).toHaveBeenCalledTimes(2);
266
- expect(mockAddItem).toHaveBeenLastCalledWith(
267
- {
268
- type: MessageType.INFO,
269
- text: expect.stringContaining(
270
- "Command '/parent' requires a subcommand.",
271
- ),
272
- },
273
- expect.any(Number),
274
- );
275
- });
276
-
277
- it('should correctly find and execute a nested subcommand', async () => {
278
- const childAction = vi.fn();
279
- const parentCommand: SlashCommand = {
280
- name: 'parent',
281
- description: 'a parent command',
282
- kind: CommandKind.BUILT_IN,
283
- subCommands: [
284
- {
285
- name: 'child',
286
- description: 'a child command',
287
- kind: CommandKind.BUILT_IN,
288
- action: childAction,
289
- },
290
- ],
291
- };
292
- const result = setupProcessorHook([parentCommand]);
293
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
294
-
295
- await act(async () => {
296
- await result.current.handleSlashCommand('/parent child with args');
297
- });
298
-
299
- expect(childAction).toHaveBeenCalledTimes(1);
300
-
301
- expect(childAction).toHaveBeenCalledWith(
302
- expect.objectContaining({
303
- services: expect.objectContaining({
304
- config: mockConfig,
305
- }),
306
- ui: expect.objectContaining({
307
- addItem: mockAddItem,
308
- }),
309
- }),
310
- 'with args',
311
- );
312
- });
313
-
314
- it('sets isProcessing to false if the the input is not a command', async () => {
315
- const setMockIsProcessing = vi.fn();
316
- const result = setupProcessorHook([], [], [], setMockIsProcessing);
317
-
318
- await act(async () => {
319
- await result.current.handleSlashCommand('imnotacommand');
320
- });
321
-
322
- expect(setMockIsProcessing).not.toHaveBeenCalled();
323
- });
324
-
325
- it('sets isProcessing to false if the command has an error', async () => {
326
- const setMockIsProcessing = vi.fn();
327
- const failCommand = createTestCommand({
328
- name: 'fail',
329
- action: vi.fn().mockRejectedValue(new Error('oh no!')),
330
- });
331
-
332
- const result = setupProcessorHook(
333
- [failCommand],
334
- [],
335
- [],
336
- setMockIsProcessing,
337
- );
338
-
339
- await act(async () => {
340
- await result.current.handleSlashCommand('/fail');
341
- });
342
-
343
- expect(setMockIsProcessing).toHaveBeenNthCalledWith(1, true);
344
- expect(setMockIsProcessing).toHaveBeenNthCalledWith(2, false);
345
- });
346
-
347
- it('should set isProcessing to true during execution and false afterwards', async () => {
348
- const mockSetIsProcessing = vi.fn();
349
- const command = createTestCommand({
350
- name: 'long-running',
351
- action: () => new Promise((resolve) => setTimeout(resolve, 50)),
352
- });
353
-
354
- const result = setupProcessorHook([command], [], [], mockSetIsProcessing);
355
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
356
-
357
- const executionPromise = act(async () => {
358
- await result.current.handleSlashCommand('/long-running');
359
- });
360
-
361
- // It should be true immediately after starting
362
- expect(mockSetIsProcessing).toHaveBeenNthCalledWith(1, true);
363
- // It should not have been called with false yet
364
- expect(mockSetIsProcessing).not.toHaveBeenCalledWith(false);
365
-
366
- await executionPromise;
367
-
368
- // After the promise resolves, it should be called with false
369
- expect(mockSetIsProcessing).toHaveBeenNthCalledWith(2, false);
370
- expect(mockSetIsProcessing).toHaveBeenCalledTimes(2);
371
- });
372
- });
373
-
374
- describe('Action Result Handling', () => {
375
- it('should handle "dialog: theme" action', async () => {
376
- const command = createTestCommand({
377
- name: 'themecmd',
378
- action: vi.fn().mockResolvedValue({ type: 'dialog', dialog: 'theme' }),
379
- });
380
- const result = setupProcessorHook([command]);
381
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
382
-
383
- await act(async () => {
384
- await result.current.handleSlashCommand('/themecmd');
385
- });
386
-
387
- expect(mockOpenThemeDialog).toHaveBeenCalled();
388
- });
389
-
390
- it('should handle "load_history" action', async () => {
391
- const command = createTestCommand({
392
- name: 'load',
393
- action: vi.fn().mockResolvedValue({
394
- type: 'load_history',
395
- history: [{ type: MessageType.USER, text: 'old prompt' }],
396
- clientHistory: [{ role: 'user', parts: [{ text: 'old prompt' }] }],
397
- }),
398
- });
399
- const result = setupProcessorHook([command]);
400
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
401
-
402
- await act(async () => {
403
- await result.current.handleSlashCommand('/load');
404
- });
405
-
406
- expect(mockClearItems).toHaveBeenCalledTimes(1);
407
- expect(mockAddItem).toHaveBeenCalledWith(
408
- { type: 'user', text: 'old prompt' },
409
- expect.any(Number),
410
- );
411
- });
412
-
413
- describe('with fake timers', () => {
414
- // This test needs to let the async `waitFor` complete with REAL timers
415
- // before switching to FAKE timers to test setTimeout.
416
- it('should handle a "quit" action', async () => {
417
- const quitAction = vi
418
- .fn()
419
- .mockResolvedValue({ type: 'quit', messages: [] });
420
- const command = createTestCommand({
421
- name: 'exit',
422
- action: quitAction,
423
- });
424
- const result = setupProcessorHook([command]);
425
-
426
- await waitFor(() =>
427
- expect(result.current.slashCommands).toHaveLength(1),
428
- );
429
-
430
- vi.useFakeTimers();
431
-
432
- try {
433
- await act(async () => {
434
- await result.current.handleSlashCommand('/exit');
435
- });
436
-
437
- await act(async () => {
438
- await vi.advanceTimersByTimeAsync(200);
439
- });
440
-
441
- expect(mockSetQuittingMessages).toHaveBeenCalledWith([]);
442
- expect(mockProcessExit).toHaveBeenCalledWith(0);
443
- } finally {
444
- vi.useRealTimers();
445
- }
446
- });
447
-
448
- it('should call runExitCleanup when handling a "quit" action', async () => {
449
- const quitAction = vi
450
- .fn()
451
- .mockResolvedValue({ type: 'quit', messages: [] });
452
- const command = createTestCommand({
453
- name: 'exit',
454
- action: quitAction,
455
- });
456
- const result = setupProcessorHook([command]);
457
-
458
- await waitFor(() =>
459
- expect(result.current.slashCommands).toHaveLength(1),
460
- );
461
-
462
- vi.useFakeTimers();
463
-
464
- try {
465
- await act(async () => {
466
- await result.current.handleSlashCommand('/exit');
467
- });
468
-
469
- await act(async () => {
470
- await vi.advanceTimersByTimeAsync(200);
471
- });
472
-
473
- expect(mockRunExitCleanup).toHaveBeenCalledTimes(1);
474
- } finally {
475
- vi.useRealTimers();
476
- }
477
- });
478
- });
479
-
480
- it('should handle "submit_prompt" action returned from a file-based command', async () => {
481
- const fileCommand = createTestCommand(
482
- {
483
- name: 'filecmd',
484
- description: 'A command from a file',
485
- action: async () => ({
486
- type: 'submit_prompt',
487
- content: 'The actual prompt from the TOML file.',
488
- }),
489
- },
490
- CommandKind.FILE,
491
- );
492
-
493
- const result = setupProcessorHook([], [fileCommand]);
494
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
495
-
496
- let actionResult;
497
- await act(async () => {
498
- actionResult = await result.current.handleSlashCommand('/filecmd');
499
- });
500
-
501
- expect(actionResult).toEqual({
502
- type: 'submit_prompt',
503
- content: 'The actual prompt from the TOML file.',
504
- });
505
-
506
- expect(mockAddItem).toHaveBeenCalledWith(
507
- { type: MessageType.USER, text: '/filecmd' },
508
- expect.any(Number),
509
- );
510
- });
511
-
512
- it('should handle "submit_prompt" action returned from a mcp-based command', async () => {
513
- const mcpCommand = createTestCommand(
514
- {
515
- name: 'mcpcmd',
516
- description: 'A command from mcp',
517
- action: async () => ({
518
- type: 'submit_prompt',
519
- content: 'The actual prompt from the mcp command.',
520
- }),
521
- },
522
- CommandKind.MCP_PROMPT,
523
- );
524
-
525
- const result = setupProcessorHook([], [], [mcpCommand]);
526
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
527
-
528
- let actionResult;
529
- await act(async () => {
530
- actionResult = await result.current.handleSlashCommand('/mcpcmd');
531
- });
532
-
533
- expect(actionResult).toEqual({
534
- type: 'submit_prompt',
535
- content: 'The actual prompt from the mcp command.',
536
- });
537
-
538
- expect(mockAddItem).toHaveBeenCalledWith(
539
- { type: MessageType.USER, text: '/mcpcmd' },
540
- expect.any(Number),
541
- );
542
- });
543
- });
544
-
545
- describe('Shell Command Confirmation Flow', () => {
546
- // Use a generic vi.fn() for the action. We will change its behavior in each test.
547
- const mockCommandAction = vi.fn();
548
-
549
- const shellCommand = createTestCommand({
550
- name: 'shellcmd',
551
- action: mockCommandAction,
552
- });
553
-
554
- beforeEach(() => {
555
- // Reset the mock before each test
556
- mockCommandAction.mockClear();
557
-
558
- // Default behavior: request confirmation
559
- mockCommandAction.mockResolvedValue({
560
- type: 'confirm_shell_commands',
561
- commandsToConfirm: ['rm -rf /'],
562
- originalInvocation: { raw: '/shellcmd' },
563
- } as ConfirmShellCommandsActionReturn);
564
- });
565
-
566
- it('should set confirmation request when action returns confirm_shell_commands', async () => {
567
- const result = setupProcessorHook([shellCommand]);
568
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
569
-
570
- // This is intentionally not awaited, because the promise it returns
571
- // will not resolve until the user responds to the confirmation.
572
- act(() => {
573
- result.current.handleSlashCommand('/shellcmd');
574
- });
575
-
576
- // We now wait for the state to be updated with the request.
577
- await waitFor(() => {
578
- expect(result.current.shellConfirmationRequest).not.toBeNull();
579
- });
580
-
581
- expect(result.current.shellConfirmationRequest?.commands).toEqual([
582
- 'rm -rf /',
583
- ]);
584
- });
585
-
586
- it('should do nothing if user cancels confirmation', async () => {
587
- const result = setupProcessorHook([shellCommand]);
588
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
589
-
590
- act(() => {
591
- result.current.handleSlashCommand('/shellcmd');
592
- });
593
-
594
- // Wait for the confirmation dialog to be set
595
- await waitFor(() => {
596
- expect(result.current.shellConfirmationRequest).not.toBeNull();
597
- });
598
-
599
- const onConfirm = result.current.shellConfirmationRequest?.onConfirm;
600
- expect(onConfirm).toBeDefined();
601
-
602
- // Change the mock action's behavior for a potential second run.
603
- // If the test is flawed, this will be called, and we can detect it.
604
- mockCommandAction.mockResolvedValue({
605
- type: 'message',
606
- messageType: 'info',
607
- content: 'This should not be called',
608
- });
609
-
610
- await act(async () => {
611
- onConfirm!(ToolConfirmationOutcome.Cancel, []); // Pass empty array for safety
612
- });
613
-
614
- expect(result.current.shellConfirmationRequest).toBeNull();
615
- // Verify the action was only called the initial time.
616
- expect(mockCommandAction).toHaveBeenCalledTimes(1);
617
- });
618
-
619
- it('should re-run command with one-time allowlist on "Proceed Once"', async () => {
620
- const result = setupProcessorHook([shellCommand]);
621
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
622
-
623
- act(() => {
624
- result.current.handleSlashCommand('/shellcmd');
625
- });
626
- await waitFor(() => {
627
- expect(result.current.shellConfirmationRequest).not.toBeNull();
628
- });
629
-
630
- const onConfirm = result.current.shellConfirmationRequest?.onConfirm;
631
-
632
- // **Change the mock's behavior for the SECOND run.**
633
- // This is the key to testing the outcome.
634
- mockCommandAction.mockResolvedValue({
635
- type: 'message',
636
- messageType: 'info',
637
- content: 'Success!',
638
- });
639
-
640
- await act(async () => {
641
- onConfirm!(ToolConfirmationOutcome.ProceedOnce, ['rm -rf /']);
642
- });
643
-
644
- expect(result.current.shellConfirmationRequest).toBeNull();
645
-
646
- // The action should have been called twice (initial + re-run).
647
- await waitFor(() => {
648
- expect(mockCommandAction).toHaveBeenCalledTimes(2);
649
- });
650
-
651
- // We can inspect the context of the second call to ensure the one-time list was used.
652
- const secondCallContext = mockCommandAction.mock
653
- .calls[1][0] as CommandContext;
654
- expect(
655
- secondCallContext.session.sessionShellAllowlist.has('rm -rf /'),
656
- ).toBe(true);
657
-
658
- // Verify the final success message was added.
659
- expect(mockAddItem).toHaveBeenCalledWith(
660
- { type: MessageType.INFO, text: 'Success!' },
661
- expect.any(Number),
662
- );
663
-
664
- // Verify the session-wide allowlist was NOT permanently updated.
665
- // Re-render the hook by calling a no-op command to get the latest context.
666
- await act(async () => {
667
- result.current.handleSlashCommand('/no-op');
668
- });
669
- const finalContext = result.current.commandContext;
670
- expect(finalContext.session.sessionShellAllowlist.size).toBe(0);
671
- });
672
-
673
- it('should re-run command and update session allowlist on "Proceed Always"', async () => {
674
- const result = setupProcessorHook([shellCommand]);
675
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
676
-
677
- act(() => {
678
- result.current.handleSlashCommand('/shellcmd');
679
- });
680
- await waitFor(() => {
681
- expect(result.current.shellConfirmationRequest).not.toBeNull();
682
- });
683
-
684
- const onConfirm = result.current.shellConfirmationRequest?.onConfirm;
685
- mockCommandAction.mockResolvedValue({
686
- type: 'message',
687
- messageType: 'info',
688
- content: 'Success!',
689
- });
690
-
691
- await act(async () => {
692
- onConfirm!(ToolConfirmationOutcome.ProceedAlways, ['rm -rf /']);
693
- });
694
-
695
- expect(result.current.shellConfirmationRequest).toBeNull();
696
- await waitFor(() => {
697
- expect(mockCommandAction).toHaveBeenCalledTimes(2);
698
- });
699
-
700
- expect(mockAddItem).toHaveBeenCalledWith(
701
- { type: MessageType.INFO, text: 'Success!' },
702
- expect.any(Number),
703
- );
704
-
705
- // Check that the session-wide allowlist WAS updated.
706
- await waitFor(() => {
707
- const finalContext = result.current.commandContext;
708
- expect(finalContext.session.sessionShellAllowlist.has('rm -rf /')).toBe(
709
- true,
710
- );
711
- });
712
- });
713
- });
714
-
715
- describe('Command Parsing and Matching', () => {
716
- it('should be case-sensitive and pass wrong case to AI', async () => {
717
- const command = createTestCommand({ name: 'test' });
718
- const result = setupProcessorHook([command]);
719
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
720
-
721
- const commandResult = await act(async () =>
722
- // Use uppercase when command is lowercase
723
- await result.current.handleSlashCommand('/Test')
724
- );
725
-
726
- // Should return false to let AI process the command
727
- expect(commandResult).toBe(false);
728
- // Should only add user input, no error message
729
- expect(mockAddItem).toHaveBeenCalledTimes(1);
730
- expect(mockAddItem).toHaveBeenCalledWith(
731
- {
732
- type: MessageType.USER,
733
- text: '/Test',
734
- },
735
- expect.any(Number),
736
- );
737
- });
738
-
739
- it('should correctly match an altName', async () => {
740
- const action = vi.fn();
741
- const command = createTestCommand({
742
- name: 'main',
743
- altNames: ['alias'],
744
- description: 'a command with an alias',
745
- action,
746
- });
747
- const result = setupProcessorHook([command]);
748
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
749
-
750
- await act(async () => {
751
- await result.current.handleSlashCommand('/alias');
752
- });
753
-
754
- expect(action).toHaveBeenCalledTimes(1);
755
- expect(mockAddItem).not.toHaveBeenCalledWith(
756
- expect.objectContaining({ type: MessageType.ERROR }),
757
- );
758
- });
759
-
760
- it('should handle extra whitespace around the command', async () => {
761
- const action = vi.fn();
762
- const command = createTestCommand({ name: 'test', action });
763
- const result = setupProcessorHook([command]);
764
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
765
-
766
- await act(async () => {
767
- await result.current.handleSlashCommand(' /test with-args ');
768
- });
769
-
770
- expect(action).toHaveBeenCalledWith(expect.anything(), 'with-args');
771
- });
772
-
773
- it('should handle `?` as a command prefix', async () => {
774
- const action = vi.fn();
775
- const command = createTestCommand({ name: 'help', action });
776
- const result = setupProcessorHook([command]);
777
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(1));
778
-
779
- await act(async () => {
780
- await result.current.handleSlashCommand('?help');
781
- });
782
-
783
- expect(action).toHaveBeenCalledTimes(1);
784
- });
785
- });
786
-
787
- describe('Command Precedence', () => {
788
- it('should override mcp-based commands with file-based commands of the same name', async () => {
789
- const mcpAction = vi.fn();
790
- const fileAction = vi.fn();
791
-
792
- const mcpCommand = createTestCommand(
793
- {
794
- name: 'override',
795
- description: 'mcp',
796
- action: mcpAction,
797
- },
798
- CommandKind.MCP_PROMPT,
799
- );
800
- const fileCommand = createTestCommand(
801
- { name: 'override', description: 'file', action: fileAction },
802
- CommandKind.FILE,
803
- );
804
-
805
- const result = setupProcessorHook([], [fileCommand], [mcpCommand]);
806
-
807
- await waitFor(() => {
808
- // The service should only return one command with the name 'override'
809
- expect(result.current.slashCommands).toHaveLength(1);
810
- });
811
-
812
- await act(async () => {
813
- await result.current.handleSlashCommand('/override');
814
- });
815
-
816
- // Only the file-based command's action should be called.
817
- expect(fileAction).toHaveBeenCalledTimes(1);
818
- expect(mcpAction).not.toHaveBeenCalled();
819
- });
820
-
821
- it('should prioritize a command with a primary name over a command with a matching alias', async () => {
822
- const quitAction = vi.fn();
823
- const exitAction = vi.fn();
824
-
825
- const quitCommand = createTestCommand({
826
- name: 'quit',
827
- altNames: ['exit'],
828
- action: quitAction,
829
- });
830
-
831
- const exitCommand = createTestCommand(
832
- {
833
- name: 'exit',
834
- action: exitAction,
835
- },
836
- CommandKind.FILE,
837
- );
838
-
839
- // The order of commands in the final loaded array is not guaranteed,
840
- // so the test must work regardless of which comes first.
841
- const result = setupProcessorHook([quitCommand], [exitCommand]);
842
-
843
- await waitFor(() => {
844
- expect(result.current.slashCommands).toHaveLength(2);
845
- });
846
-
847
- await act(async () => {
848
- await result.current.handleSlashCommand('/exit');
849
- });
850
-
851
- // The action for the command whose primary name is 'exit' should be called.
852
- expect(exitAction).toHaveBeenCalledTimes(1);
853
- // The action for the command that has 'exit' as an alias should NOT be called.
854
- expect(quitAction).not.toHaveBeenCalled();
855
- });
856
-
857
- it('should add an overridden command to the history', async () => {
858
- const quitCommand = createTestCommand({
859
- name: 'quit',
860
- altNames: ['exit'],
861
- action: vi.fn(),
862
- });
863
- const exitCommand = createTestCommand(
864
- { name: 'exit', action: vi.fn() },
865
- CommandKind.FILE,
866
- );
867
-
868
- const result = setupProcessorHook([quitCommand], [exitCommand]);
869
- await waitFor(() => expect(result.current.slashCommands).toHaveLength(2));
870
-
871
- await act(async () => {
872
- await result.current.handleSlashCommand('/exit');
873
- });
874
-
875
- // It should be added to the history.
876
- expect(mockAddItem).toHaveBeenCalledWith(
877
- { type: MessageType.USER, text: '/exit' },
878
- expect.any(Number),
879
- );
880
- });
881
- });
882
-
883
- describe('Lifecycle', () => {
884
- it('should abort command loading when the hook unmounts', () => {
885
- const abortSpy = vi.spyOn(AbortController.prototype, 'abort');
886
- const { unmount } = renderHook(() =>
887
- useSlashCommandProcessor(
888
- mockConfig,
889
- mockSettings,
890
- mockAddItem,
891
- mockClearItems,
892
- mockLoadHistory,
893
- vi.fn(), // refreshStatic
894
- vi.fn(), // onDebugMessage
895
- vi.fn(), // openThemeDialog
896
- mockOpenAuthDialog,
897
- vi.fn(), // openEditorDialog
898
- vi.fn(), // toggleCorgiMode
899
- mockSetQuittingMessages,
900
- vi.fn(), // openPrivacyNotice
901
-
902
- vi.fn(), // openSettingsDialog
903
- vi.fn(), // toggleVimEnabled
904
- vi.fn().mockResolvedValue(false), // toggleVimEnabled
905
- vi.fn(), // setIsProcessing
906
- ),
907
- );
908
-
909
- unmount();
910
-
911
- expect(abortSpy).toHaveBeenCalledTimes(1);
912
- });
913
- });
914
-
915
- describe('Slash Command Logging', () => {
916
- const mockCommandAction = vi.fn().mockResolvedValue({ type: 'handled' });
917
- const loggingTestCommands: SlashCommand[] = [
918
- createTestCommand({
919
- name: 'logtest',
920
- action: vi
921
- .fn()
922
- .mockResolvedValue({ type: 'message', content: 'hello world' }),
923
- }),
924
- createTestCommand({
925
- name: 'logwithsub',
926
- subCommands: [
927
- createTestCommand({
928
- name: 'sub',
929
- action: mockCommandAction,
930
- }),
931
- ],
932
- }),
933
- createTestCommand({
934
- name: 'fail',
935
- action: vi.fn().mockRejectedValue(new Error('oh no!')),
936
- }),
937
- createTestCommand({
938
- name: 'logalias',
939
- altNames: ['la'],
940
- action: mockCommandAction,
941
- }),
942
- ];
943
-
944
- beforeEach(() => {
945
- mockCommandAction.mockClear();
946
- vi.mocked(logSlashCommand).mockClear();
947
- });
948
-
949
- it('should log a simple slash command', async () => {
950
- const result = setupProcessorHook(loggingTestCommands);
951
- await waitFor(() =>
952
- expect(result.current.slashCommands.length).toBeGreaterThan(0),
953
- );
954
- await act(async () => {
955
- await result.current.handleSlashCommand('/logtest');
956
- });
957
-
958
- expect(logSlashCommand).toHaveBeenCalledWith(
959
- mockConfig,
960
- expect.objectContaining({
961
- command: 'logtest',
962
- subcommand: undefined,
963
- status: SlashCommandStatus.SUCCESS,
964
- }),
965
- );
966
- });
967
-
968
- it('logs nothing for a bogus command', async () => {
969
- const result = setupProcessorHook(loggingTestCommands);
970
- await waitFor(() =>
971
- expect(result.current.slashCommands.length).toBeGreaterThan(0),
972
- );
973
- await act(async () => {
974
- await result.current.handleSlashCommand('/bogusbogusbogus');
975
- });
976
-
977
- expect(logSlashCommand).not.toHaveBeenCalled();
978
- });
979
-
980
- it('logs a failure event for a failed command', async () => {
981
- const result = setupProcessorHook(loggingTestCommands);
982
- await waitFor(() =>
983
- expect(result.current.slashCommands.length).toBeGreaterThan(0),
984
- );
985
- await act(async () => {
986
- await result.current.handleSlashCommand('/fail');
987
- });
988
-
989
- expect(logSlashCommand).toHaveBeenCalledWith(
990
- mockConfig,
991
- expect.objectContaining({
992
- command: 'fail',
993
- status: 'error',
994
- subcommand: undefined,
995
- }),
996
- );
997
- });
998
-
999
- it('should log a slash command with a subcommand', async () => {
1000
- const result = setupProcessorHook(loggingTestCommands);
1001
- await waitFor(() =>
1002
- expect(result.current.slashCommands.length).toBeGreaterThan(0),
1003
- );
1004
- await act(async () => {
1005
- await result.current.handleSlashCommand('/logwithsub sub');
1006
- });
1007
-
1008
- expect(logSlashCommand).toHaveBeenCalledWith(
1009
- mockConfig,
1010
- expect.objectContaining({
1011
- command: 'logwithsub',
1012
- subcommand: 'sub',
1013
- }),
1014
- );
1015
- });
1016
-
1017
- it('should log the command path when an alias is used', async () => {
1018
- const result = setupProcessorHook(loggingTestCommands);
1019
- await waitFor(() =>
1020
- expect(result.current.slashCommands.length).toBeGreaterThan(0),
1021
- );
1022
- await act(async () => {
1023
- await result.current.handleSlashCommand('/la');
1024
- });
1025
- expect(logSlashCommand).toHaveBeenCalledWith(
1026
- mockConfig,
1027
- expect.objectContaining({
1028
- command: 'logalias',
1029
- }),
1030
- );
1031
- });
1032
-
1033
- it('should not log for unknown commands', async () => {
1034
- const result = setupProcessorHook(loggingTestCommands);
1035
- await waitFor(() =>
1036
- expect(result.current.slashCommands.length).toBeGreaterThan(0),
1037
- );
1038
- await act(async () => {
1039
- await result.current.handleSlashCommand('/unknown');
1040
- });
1041
- expect(logSlashCommand).not.toHaveBeenCalled();
1042
- });
1043
- });
1044
- });