fss-link 1.0.40 → 1.0.45

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 (428) hide show
  1. package/dist/commands/mcp/add.test.ts +122 -0
  2. package/dist/commands/mcp/add.ts +222 -0
  3. package/dist/commands/mcp/list.test.ts +154 -0
  4. package/dist/commands/mcp/list.ts +139 -0
  5. package/dist/commands/mcp/remove.test.ts +69 -0
  6. package/dist/commands/mcp/remove.ts +60 -0
  7. package/dist/commands/mcp.test.ts +55 -0
  8. package/dist/commands/mcp.ts +27 -0
  9. package/dist/config/apiValidation.test.ts +118 -0
  10. package/dist/config/auth.test.ts +79 -0
  11. package/dist/config/auth.ts +100 -0
  12. package/dist/config/config.integration.test.ts +407 -0
  13. package/dist/config/config.test.ts +1952 -0
  14. package/dist/config/config.ts +690 -0
  15. package/dist/config/database.test.ts +96 -0
  16. package/dist/config/database.ts +752 -0
  17. package/dist/config/extension.test.ts +236 -0
  18. package/dist/config/extension.ts +180 -0
  19. package/dist/config/keyBindings.test.ts +62 -0
  20. package/dist/config/keyBindings.ts +184 -0
  21. package/dist/config/modelManager.ts +275 -0
  22. package/dist/config/providerManager.ts +244 -0
  23. package/dist/config/providerPersistence.test.ts +377 -0
  24. package/dist/config/providerPersistence.ts +105 -0
  25. package/dist/config/sandboxConfig.ts +107 -0
  26. package/dist/config/settings.test.ts +1424 -0
  27. package/dist/config/settings.ts +517 -0
  28. package/dist/config/settingsSchema.test.ts +252 -0
  29. package/dist/config/settingsSchema.ts +728 -0
  30. package/dist/config/trustedFolders.test.ts +208 -0
  31. package/dist/config/trustedFolders.ts +167 -0
  32. package/dist/gemini.test.tsx +252 -0
  33. package/dist/gemini.tsx +357 -0
  34. package/dist/generated/git-commit.ts +10 -0
  35. package/dist/index.ts +21 -0
  36. package/dist/nonInteractiveCli.test.ts +276 -0
  37. package/dist/nonInteractiveCli.ts +143 -0
  38. package/dist/package.json +87 -87
  39. package/dist/patches/is-in-ci.ts +17 -0
  40. package/dist/services/BuiltinCommandLoader.test.ts +127 -0
  41. package/dist/services/BuiltinCommandLoader.ts +95 -0
  42. package/dist/services/CommandService.test.ts +352 -0
  43. package/dist/services/CommandService.ts +103 -0
  44. package/dist/services/FileCommandLoader.test.ts +1002 -0
  45. package/dist/services/FileCommandLoader.ts +289 -0
  46. package/dist/services/McpPromptLoader.ts +231 -0
  47. package/dist/services/SearchEngineConfigProvider.ts +100 -0
  48. package/dist/services/prompt-processors/argumentProcessor.test.ts +41 -0
  49. package/dist/services/prompt-processors/argumentProcessor.ts +23 -0
  50. package/dist/services/prompt-processors/shellProcessor.test.ts +709 -0
  51. package/dist/services/prompt-processors/shellProcessor.ts +248 -0
  52. package/dist/services/prompt-processors/types.ts +44 -0
  53. package/dist/services/types.ts +24 -0
  54. package/dist/src/config/apiValidation.test.d.ts +6 -0
  55. package/dist/src/config/apiValidation.test.js +99 -0
  56. package/dist/src/config/apiValidation.test.js.map +1 -0
  57. package/dist/src/config/database.d.ts +32 -0
  58. package/dist/src/config/database.js +281 -2
  59. package/dist/src/config/database.js.map +1 -1
  60. package/dist/src/config/database.test.d.ts +6 -0
  61. package/dist/src/config/database.test.js +80 -0
  62. package/dist/src/config/database.test.js.map +1 -0
  63. package/dist/src/config/providerManager.d.ts +74 -0
  64. package/dist/src/config/providerManager.js +203 -0
  65. package/dist/src/config/providerManager.js.map +1 -0
  66. package/dist/src/config/providerPersistence.d.ts +75 -0
  67. package/dist/src/config/providerPersistence.js +55 -0
  68. package/dist/src/config/providerPersistence.js.map +1 -0
  69. package/dist/src/config/providerPersistence.test.d.ts +6 -0
  70. package/dist/src/config/providerPersistence.test.js +283 -0
  71. package/dist/src/config/providerPersistence.test.js.map +1 -0
  72. package/dist/src/config/settingsSchema.d.ts +9 -0
  73. package/dist/src/config/settingsSchema.js +9 -0
  74. package/dist/src/config/settingsSchema.js.map +1 -1
  75. package/dist/src/generated/git-commit.d.ts +1 -1
  76. package/dist/src/generated/git-commit.js +1 -1
  77. package/dist/src/services/BuiltinCommandLoader.js +2 -0
  78. package/dist/src/services/BuiltinCommandLoader.js.map +1 -1
  79. package/dist/src/ui/App.js +14 -2
  80. package/dist/src/ui/App.js.map +1 -1
  81. package/dist/src/ui/commands/contextCommand.d.ts +7 -0
  82. package/dist/src/ui/commands/contextCommand.js +115 -0
  83. package/dist/src/ui/commands/contextCommand.js.map +1 -0
  84. package/dist/src/ui/components/ContextUsageDisplay.d.ts +3 -1
  85. package/dist/src/ui/components/ContextUsageDisplay.js +43 -3
  86. package/dist/src/ui/components/ContextUsageDisplay.js.map +1 -1
  87. package/dist/src/ui/components/Footer.d.ts +1 -0
  88. package/dist/src/ui/components/Footer.js +2 -2
  89. package/dist/src/ui/components/Footer.js.map +1 -1
  90. package/dist/src/ui/components/GeminiKeyDialog.d.ts +11 -0
  91. package/dist/src/ui/components/GeminiKeyDialog.js +156 -0
  92. package/dist/src/ui/components/GeminiKeyDialog.js.map +1 -0
  93. package/dist/src/ui/components/OpenAIEndpointDialog.d.ts +19 -0
  94. package/dist/src/ui/components/OpenAIEndpointDialog.js +163 -0
  95. package/dist/src/ui/components/OpenAIEndpointDialog.js.map +1 -0
  96. package/dist/src/ui/components/WelcomeBackDialog.d.ts +36 -0
  97. package/dist/src/ui/components/WelcomeBackDialog.js +109 -0
  98. package/dist/src/ui/components/WelcomeBackDialog.js.map +1 -0
  99. package/dist/src/ui/hooks/useWelcomeBack.d.ts +52 -0
  100. package/dist/src/ui/hooks/useWelcomeBack.js +214 -0
  101. package/dist/src/ui/hooks/useWelcomeBack.js.map +1 -0
  102. package/dist/src/zed-integration/schema.d.ts +1516 -1516
  103. package/dist/test-setup.ts +12 -0
  104. package/dist/test-utils/customMatchers.ts +65 -0
  105. package/dist/test-utils/mockCommandContext.test.ts +62 -0
  106. package/dist/test-utils/mockCommandContext.ts +105 -0
  107. package/dist/test-utils/render.tsx +18 -0
  108. package/dist/tsconfig.tsbuildinfo +1 -1
  109. package/dist/ui/App.test.tsx +2181 -0
  110. package/dist/ui/App.tsx +1344 -0
  111. package/dist/ui/IdeIntegrationNudge.tsx +98 -0
  112. package/dist/ui/__snapshots__/App.test.tsx.snap +124 -0
  113. package/dist/ui/colors.ts +56 -0
  114. package/dist/ui/commands/aboutCommand.test.ts +153 -0
  115. package/dist/ui/commands/aboutCommand.ts +49 -0
  116. package/dist/ui/commands/authCommand.test.ts +36 -0
  117. package/dist/ui/commands/authCommand.ts +17 -0
  118. package/dist/ui/commands/bugCommand.test.ts +114 -0
  119. package/dist/ui/commands/bugCommand.ts +92 -0
  120. package/dist/ui/commands/chatCommand.test.ts +414 -0
  121. package/dist/ui/commands/chatCommand.ts +280 -0
  122. package/dist/ui/commands/clearCommand.test.ts +100 -0
  123. package/dist/ui/commands/clearCommand.ts +29 -0
  124. package/dist/ui/commands/compressCommand.test.ts +129 -0
  125. package/dist/ui/commands/compressCommand.ts +78 -0
  126. package/dist/ui/commands/contextCommand.ts +132 -0
  127. package/dist/ui/commands/copyCommand.test.ts +296 -0
  128. package/dist/ui/commands/copyCommand.ts +67 -0
  129. package/dist/ui/commands/corgiCommand.test.ts +34 -0
  130. package/dist/ui/commands/corgiCommand.ts +16 -0
  131. package/dist/ui/commands/directoryCommand.test.tsx +185 -0
  132. package/dist/ui/commands/directoryCommand.tsx +179 -0
  133. package/dist/ui/commands/docsCommand.test.ts +99 -0
  134. package/dist/ui/commands/docsCommand.ts +42 -0
  135. package/dist/ui/commands/editorCommand.test.ts +30 -0
  136. package/dist/ui/commands/editorCommand.ts +21 -0
  137. package/dist/ui/commands/extensionsCommand.test.ts +67 -0
  138. package/dist/ui/commands/extensionsCommand.ts +46 -0
  139. package/dist/ui/commands/helpCommand.test.ts +52 -0
  140. package/dist/ui/commands/helpCommand.ts +23 -0
  141. package/dist/ui/commands/ideCommand.test.ts +255 -0
  142. package/dist/ui/commands/ideCommand.ts +283 -0
  143. package/dist/ui/commands/initCommand.test.ts +127 -0
  144. package/dist/ui/commands/initCommand.ts +117 -0
  145. package/dist/ui/commands/mcpCommand.test.ts +1057 -0
  146. package/dist/ui/commands/mcpCommand.ts +531 -0
  147. package/dist/ui/commands/memoryCommand.test.ts +344 -0
  148. package/dist/ui/commands/memoryCommand.ts +305 -0
  149. package/dist/ui/commands/privacyCommand.test.ts +38 -0
  150. package/dist/ui/commands/privacyCommand.ts +17 -0
  151. package/dist/ui/commands/quitCommand.test.ts +55 -0
  152. package/dist/ui/commands/quitCommand.ts +36 -0
  153. package/dist/ui/commands/restoreCommand.test.ts +250 -0
  154. package/dist/ui/commands/restoreCommand.ts +157 -0
  155. package/dist/ui/commands/searchEngineSetupCommand.ts +18 -0
  156. package/dist/ui/commands/settingsCommand.test.ts +36 -0
  157. package/dist/ui/commands/settingsCommand.ts +17 -0
  158. package/dist/ui/commands/setupGithubCommand.test.ts +238 -0
  159. package/dist/ui/commands/setupGithubCommand.ts +212 -0
  160. package/dist/ui/commands/speakCommand.ts +175 -0
  161. package/dist/ui/commands/statsCommand.test.ts +78 -0
  162. package/dist/ui/commands/statsCommand.ts +70 -0
  163. package/dist/ui/commands/terminalSetupCommand.test.ts +85 -0
  164. package/dist/ui/commands/terminalSetupCommand.ts +45 -0
  165. package/dist/ui/commands/themeCommand.test.ts +38 -0
  166. package/dist/ui/commands/themeCommand.ts +17 -0
  167. package/dist/ui/commands/toolsCommand.test.ts +105 -0
  168. package/dist/ui/commands/toolsCommand.ts +71 -0
  169. package/dist/ui/commands/ttsCommand.ts +143 -0
  170. package/dist/ui/commands/types.ts +204 -0
  171. package/dist/ui/commands/vimCommand.ts +25 -0
  172. package/dist/ui/commands/voiceCommand.ts +125 -0
  173. package/dist/ui/components/AboutBox.tsx +133 -0
  174. package/dist/ui/components/AsciiArt.ts +54 -0
  175. package/dist/ui/components/AuthDialog.test.tsx +334 -0
  176. package/dist/ui/components/AuthDialog.tsx +289 -0
  177. package/dist/ui/components/AuthInProgress.tsx +62 -0
  178. package/dist/ui/components/AutoAcceptIndicator.tsx +47 -0
  179. package/dist/ui/components/ConsoleSummaryDisplay.tsx +35 -0
  180. package/dist/ui/components/ContextSummaryDisplay.test.tsx +85 -0
  181. package/dist/ui/components/ContextSummaryDisplay.tsx +120 -0
  182. package/dist/ui/components/ContextUsageDisplay.tsx +77 -0
  183. package/dist/ui/components/DebugProfiler.tsx +36 -0
  184. package/dist/ui/components/DetailedMessagesDisplay.tsx +82 -0
  185. package/dist/ui/components/EditorSettingsDialog.tsx +172 -0
  186. package/dist/ui/components/FolderTrustDialog.test.tsx +36 -0
  187. package/dist/ui/components/FolderTrustDialog.tsx +74 -0
  188. package/dist/ui/components/Footer.test.tsx +159 -0
  189. package/dist/ui/components/Footer.tsx +158 -0
  190. package/dist/ui/components/GeminiKeyDialog.tsx +252 -0
  191. package/dist/ui/components/GeminiRespondingSpinner.tsx +34 -0
  192. package/dist/ui/components/Header.test.tsx +44 -0
  193. package/dist/ui/components/Header.tsx +70 -0
  194. package/dist/ui/components/Help.tsx +174 -0
  195. package/dist/ui/components/HistoryItemDisplay.test.tsx +125 -0
  196. package/dist/ui/components/HistoryItemDisplay.tsx +98 -0
  197. package/dist/ui/components/InputPrompt.test.tsx +1467 -0
  198. package/dist/ui/components/InputPrompt.tsx +641 -0
  199. package/dist/ui/components/LMStudioModelPrompt.tsx +215 -0
  200. package/dist/ui/components/LoadingIndicator.test.tsx +296 -0
  201. package/dist/ui/components/LoadingIndicator.tsx +82 -0
  202. package/dist/ui/components/MemoryUsageDisplay.tsx +36 -0
  203. package/dist/ui/components/ModelStatsDisplay.test.tsx +252 -0
  204. package/dist/ui/components/ModelStatsDisplay.tsx +197 -0
  205. package/dist/ui/components/OllamaModelPrompt.tsx +206 -0
  206. package/dist/ui/components/OpenAIEndpointDialog.tsx +261 -0
  207. package/dist/ui/components/OpenAIKeyPrompt.test.tsx +64 -0
  208. package/dist/ui/components/OpenAIKeyPrompt.tsx +197 -0
  209. package/dist/ui/components/PrepareLabel.tsx +48 -0
  210. package/dist/ui/components/SearchEngineConfigDialog.tsx +280 -0
  211. package/dist/ui/components/SessionSummaryDisplay.test.tsx +75 -0
  212. package/dist/ui/components/SessionSummaryDisplay.tsx +18 -0
  213. package/dist/ui/components/SettingsDialog.test.tsx +865 -0
  214. package/dist/ui/components/SettingsDialog.tsx +753 -0
  215. package/dist/ui/components/ShellConfirmationDialog.test.tsx +53 -0
  216. package/dist/ui/components/ShellConfirmationDialog.tsx +103 -0
  217. package/dist/ui/components/ShellModeIndicator.tsx +18 -0
  218. package/dist/ui/components/ShowMoreLines.tsx +40 -0
  219. package/dist/ui/components/StatsDisplay.test.tsx +401 -0
  220. package/dist/ui/components/StatsDisplay.tsx +273 -0
  221. package/dist/ui/components/SuggestionsDisplay.tsx +102 -0
  222. package/dist/ui/components/ThemeDialog.tsx +310 -0
  223. package/dist/ui/components/Tips.tsx +45 -0
  224. package/dist/ui/components/TodoDisplay.test.tsx +97 -0
  225. package/dist/ui/components/TodoDisplay.tsx +72 -0
  226. package/dist/ui/components/ToolStatsDisplay.test.tsx +180 -0
  227. package/dist/ui/components/ToolStatsDisplay.tsx +208 -0
  228. package/dist/ui/components/UpdateNotification.tsx +23 -0
  229. package/dist/ui/components/WelcomeBackDialog.tsx +290 -0
  230. package/dist/ui/components/__snapshots__/IDEContextDetailDisplay.test.tsx.snap +24 -0
  231. package/dist/ui/components/__snapshots__/ModelStatsDisplay.test.tsx.snap +121 -0
  232. package/dist/ui/components/__snapshots__/SessionSummaryDisplay.test.tsx.snap +30 -0
  233. package/dist/ui/components/__snapshots__/ShellConfirmationDialog.test.tsx.snap +21 -0
  234. package/dist/ui/components/__snapshots__/StatsDisplay.test.tsx.snap +264 -0
  235. package/dist/ui/components/__snapshots__/ToolStatsDisplay.test.tsx.snap +91 -0
  236. package/dist/ui/components/messages/CompressionMessage.tsx +49 -0
  237. package/dist/ui/components/messages/DiffRenderer.test.tsx +365 -0
  238. package/dist/ui/components/messages/DiffRenderer.tsx +358 -0
  239. package/dist/ui/components/messages/ErrorMessage.tsx +31 -0
  240. package/dist/ui/components/messages/GeminiMessage.tsx +43 -0
  241. package/dist/ui/components/messages/GeminiMessageContent.tsx +43 -0
  242. package/dist/ui/components/messages/InfoMessage.tsx +32 -0
  243. package/dist/ui/components/messages/ToolConfirmationMessage.test.tsx +58 -0
  244. package/dist/ui/components/messages/ToolConfirmationMessage.tsx +297 -0
  245. package/dist/ui/components/messages/ToolGroupMessage.tsx +126 -0
  246. package/dist/ui/components/messages/ToolMessage.test.tsx +183 -0
  247. package/dist/ui/components/messages/ToolMessage.tsx +296 -0
  248. package/dist/ui/components/messages/UserMessage.tsx +43 -0
  249. package/dist/ui/components/messages/UserShellMessage.tsx +25 -0
  250. package/dist/ui/components/shared/MaxSizedBox.test.tsx +425 -0
  251. package/dist/ui/components/shared/MaxSizedBox.tsx +624 -0
  252. package/dist/ui/components/shared/RadioButtonSelect.test.tsx +181 -0
  253. package/dist/ui/components/shared/RadioButtonSelect.tsx +234 -0
  254. package/dist/ui/components/shared/__snapshots__/RadioButtonSelect.test.tsx.snap +47 -0
  255. package/dist/ui/components/shared/text-buffer.test.ts +1728 -0
  256. package/dist/ui/components/shared/text-buffer.ts +2227 -0
  257. package/dist/ui/components/shared/vim-buffer-actions.test.ts +1119 -0
  258. package/dist/ui/components/shared/vim-buffer-actions.ts +814 -0
  259. package/dist/ui/constants.ts +17 -0
  260. package/dist/ui/contexts/KeypressContext.test.tsx +391 -0
  261. package/dist/ui/contexts/KeypressContext.tsx +440 -0
  262. package/dist/ui/contexts/OverflowContext.tsx +87 -0
  263. package/dist/ui/contexts/SessionContext.test.tsx +132 -0
  264. package/dist/ui/contexts/SessionContext.tsx +143 -0
  265. package/dist/ui/contexts/SettingsContext.tsx +20 -0
  266. package/dist/ui/contexts/StreamingContext.tsx +22 -0
  267. package/dist/ui/contexts/VimModeContext.tsx +79 -0
  268. package/dist/ui/editors/editorSettingsManager.ts +66 -0
  269. package/dist/ui/hooks/atCommandProcessor.test.ts +1102 -0
  270. package/dist/ui/hooks/atCommandProcessor.ts +485 -0
  271. package/dist/ui/hooks/shellCommandProcessor.test.ts +481 -0
  272. package/dist/ui/hooks/shellCommandProcessor.ts +314 -0
  273. package/dist/ui/hooks/slashCommandProcessor.test.ts +1044 -0
  274. package/dist/ui/hooks/slashCommandProcessor.ts +595 -0
  275. package/dist/ui/hooks/useAtCompletion.test.ts +497 -0
  276. package/dist/ui/hooks/useAtCompletion.ts +244 -0
  277. package/dist/ui/hooks/useAuthCommand.ts +129 -0
  278. package/dist/ui/hooks/useAutoAcceptIndicator.test.ts +300 -0
  279. package/dist/ui/hooks/useAutoAcceptIndicator.ts +52 -0
  280. package/dist/ui/hooks/useBracketedPaste.ts +37 -0
  281. package/dist/ui/hooks/useCommandCompletion.test.ts +518 -0
  282. package/dist/ui/hooks/useCommandCompletion.tsx +238 -0
  283. package/dist/ui/hooks/useCompletion.ts +128 -0
  284. package/dist/ui/hooks/useConsoleMessages.test.ts +147 -0
  285. package/dist/ui/hooks/useConsoleMessages.ts +110 -0
  286. package/dist/ui/hooks/useEditorSettings.test.ts +283 -0
  287. package/dist/ui/hooks/useEditorSettings.ts +75 -0
  288. package/dist/ui/hooks/useFocus.test.ts +119 -0
  289. package/dist/ui/hooks/useFocus.ts +48 -0
  290. package/dist/ui/hooks/useFolderTrust.test.ts +159 -0
  291. package/dist/ui/hooks/useFolderTrust.ts +72 -0
  292. package/dist/ui/hooks/useGeminiStream.test.tsx +1998 -0
  293. package/dist/ui/hooks/useGeminiStream.ts +1017 -0
  294. package/dist/ui/hooks/useGitBranchName.test.ts +280 -0
  295. package/dist/ui/hooks/useGitBranchName.ts +79 -0
  296. package/dist/ui/hooks/useHistoryManager.test.ts +202 -0
  297. package/dist/ui/hooks/useHistoryManager.ts +111 -0
  298. package/dist/ui/hooks/useInputHistory.test.ts +261 -0
  299. package/dist/ui/hooks/useInputHistory.ts +111 -0
  300. package/dist/ui/hooks/useKeypress.test.ts +280 -0
  301. package/dist/ui/hooks/useKeypress.ts +39 -0
  302. package/dist/ui/hooks/useKittyKeyboardProtocol.ts +31 -0
  303. package/dist/ui/hooks/useLoadingIndicator.test.ts +139 -0
  304. package/dist/ui/hooks/useLoadingIndicator.ts +57 -0
  305. package/dist/ui/hooks/useLogger.ts +32 -0
  306. package/dist/ui/hooks/useMessageQueue.test.ts +226 -0
  307. package/dist/ui/hooks/useMessageQueue.ts +69 -0
  308. package/dist/ui/hooks/usePhraseCycler.test.ts +145 -0
  309. package/dist/ui/hooks/usePhraseCycler.ts +198 -0
  310. package/dist/ui/hooks/usePrivacySettings.test.ts +242 -0
  311. package/dist/ui/hooks/usePrivacySettings.ts +150 -0
  312. package/dist/ui/hooks/useReactToolScheduler.ts +309 -0
  313. package/dist/ui/hooks/useRefreshMemoryCommand.ts +7 -0
  314. package/dist/ui/hooks/useReverseSearchCompletion.test.tsx +260 -0
  315. package/dist/ui/hooks/useReverseSearchCompletion.tsx +95 -0
  316. package/dist/ui/hooks/useSettingsCommand.ts +25 -0
  317. package/dist/ui/hooks/useShellHistory.test.ts +219 -0
  318. package/dist/ui/hooks/useShellHistory.ts +133 -0
  319. package/dist/ui/hooks/useShowMemoryCommand.ts +75 -0
  320. package/dist/ui/hooks/useSlashCompletion.test.ts +434 -0
  321. package/dist/ui/hooks/useSlashCompletion.ts +187 -0
  322. package/dist/ui/hooks/useStateAndRef.ts +36 -0
  323. package/dist/ui/hooks/useTerminalSize.ts +32 -0
  324. package/dist/ui/hooks/useThemeCommand.ts +110 -0
  325. package/dist/ui/hooks/useTimer.test.ts +120 -0
  326. package/dist/ui/hooks/useTimer.ts +65 -0
  327. package/dist/ui/hooks/useToolScheduler.test.ts +1123 -0
  328. package/dist/ui/hooks/useWelcomeBack.ts +253 -0
  329. package/dist/ui/hooks/vim.test.ts +1691 -0
  330. package/dist/ui/hooks/vim.ts +784 -0
  331. package/dist/ui/keyMatchers.test.ts +337 -0
  332. package/dist/ui/keyMatchers.ts +105 -0
  333. package/dist/ui/privacy/CloudFreePrivacyNotice.tsx +117 -0
  334. package/dist/ui/privacy/CloudPaidPrivacyNotice.tsx +59 -0
  335. package/dist/ui/privacy/GeminiPrivacyNotice.tsx +62 -0
  336. package/dist/ui/privacy/PrivacyNotice.tsx +42 -0
  337. package/dist/ui/semantic-colors.ts +26 -0
  338. package/dist/ui/themes/ansi-light.ts +150 -0
  339. package/dist/ui/themes/ansi.ts +159 -0
  340. package/dist/ui/themes/atom-one-dark.ts +147 -0
  341. package/dist/ui/themes/ayu-light.ts +139 -0
  342. package/dist/ui/themes/ayu.ts +113 -0
  343. package/dist/ui/themes/color-utils.test.ts +221 -0
  344. package/dist/ui/themes/color-utils.ts +231 -0
  345. package/dist/ui/themes/default-light.ts +108 -0
  346. package/dist/ui/themes/default.ts +151 -0
  347. package/dist/ui/themes/dracula.ts +124 -0
  348. package/dist/ui/themes/fss-code-dark.ts +156 -0
  349. package/dist/ui/themes/fss-dark.ts +113 -0
  350. package/dist/ui/themes/fss-light.ts +139 -0
  351. package/dist/ui/themes/github-dark.ts +147 -0
  352. package/dist/ui/themes/github-light.ts +149 -0
  353. package/dist/ui/themes/googlecode.ts +146 -0
  354. package/dist/ui/themes/no-color.ts +125 -0
  355. package/dist/ui/themes/qwen-dark.ts +118 -0
  356. package/dist/ui/themes/qwen-light.ts +144 -0
  357. package/dist/ui/themes/semantic-tokens.ts +127 -0
  358. package/dist/ui/themes/shades-of-purple.ts +352 -0
  359. package/dist/ui/themes/theme-manager.test.ts +99 -0
  360. package/dist/ui/themes/theme-manager.ts +257 -0
  361. package/dist/ui/themes/theme.test.ts +97 -0
  362. package/dist/ui/themes/theme.ts +451 -0
  363. package/dist/ui/themes/xcode.ts +154 -0
  364. package/dist/ui/types.ts +255 -0
  365. package/dist/ui/utils/CodeColorizer.tsx +217 -0
  366. package/dist/ui/utils/ConsolePatcher.ts +71 -0
  367. package/dist/ui/utils/InlineMarkdownRenderer.tsx +173 -0
  368. package/dist/ui/utils/MarkdownDisplay.test.tsx +244 -0
  369. package/dist/ui/utils/MarkdownDisplay.tsx +415 -0
  370. package/dist/ui/utils/TableRenderer.tsx +159 -0
  371. package/dist/ui/utils/__snapshots__/MarkdownDisplay.test.tsx.snap +93 -0
  372. package/dist/ui/utils/clipboardUtils.test.ts +76 -0
  373. package/dist/ui/utils/clipboardUtils.ts +149 -0
  374. package/dist/ui/utils/commandUtils.test.ts +384 -0
  375. package/dist/ui/utils/commandUtils.ts +106 -0
  376. package/dist/ui/utils/computeStats.test.ts +292 -0
  377. package/dist/ui/utils/computeStats.ts +86 -0
  378. package/dist/ui/utils/displayUtils.test.ts +58 -0
  379. package/dist/ui/utils/displayUtils.ts +32 -0
  380. package/dist/ui/utils/formatters.test.ts +72 -0
  381. package/dist/ui/utils/formatters.ts +63 -0
  382. package/dist/ui/utils/isNarrowWidth.ts +9 -0
  383. package/dist/ui/utils/kittyProtocolDetector.ts +105 -0
  384. package/dist/ui/utils/markdownUtilities.test.ts +50 -0
  385. package/dist/ui/utils/markdownUtilities.ts +125 -0
  386. package/dist/ui/utils/platformConstants.ts +52 -0
  387. package/dist/ui/utils/terminalSetup.ts +342 -0
  388. package/dist/ui/utils/textUtils.ts +40 -0
  389. package/dist/ui/utils/updateCheck.test.ts +163 -0
  390. package/dist/ui/utils/updateCheck.ts +100 -0
  391. package/dist/utils/checks.ts +28 -0
  392. package/dist/utils/cleanup.test.ts +68 -0
  393. package/dist/utils/cleanup.ts +36 -0
  394. package/dist/utils/dialogScopeUtils.ts +64 -0
  395. package/dist/utils/events.ts +14 -0
  396. package/dist/utils/gitUtils.test.ts +149 -0
  397. package/dist/utils/gitUtils.ts +116 -0
  398. package/dist/utils/handleAutoUpdate.test.ts +272 -0
  399. package/dist/utils/handleAutoUpdate.ts +145 -0
  400. package/dist/utils/installationInfo.test.ts +315 -0
  401. package/dist/utils/installationInfo.ts +176 -0
  402. package/dist/utils/package.ts +38 -0
  403. package/dist/utils/readStdin.ts +51 -0
  404. package/dist/utils/resolvePath.ts +21 -0
  405. package/dist/utils/sandbox-macos-permissive-closed.sb +32 -0
  406. package/dist/utils/sandbox-macos-permissive-open.sb +25 -0
  407. package/dist/utils/sandbox-macos-permissive-proxied.sb +37 -0
  408. package/dist/utils/sandbox-macos-restrictive-closed.sb +93 -0
  409. package/dist/utils/sandbox-macos-restrictive-open.sb +96 -0
  410. package/dist/utils/sandbox-macos-restrictive-proxied.sb +98 -0
  411. package/dist/utils/sandbox.ts +962 -0
  412. package/dist/utils/settingsUtils.test.ts +797 -0
  413. package/dist/utils/settingsUtils.ts +489 -0
  414. package/dist/utils/spawnWrapper.ts +9 -0
  415. package/dist/utils/startupWarnings.test.ts +83 -0
  416. package/dist/utils/startupWarnings.ts +40 -0
  417. package/dist/utils/updateEventEmitter.ts +13 -0
  418. package/dist/utils/userStartupWarnings.test.ts +87 -0
  419. package/dist/utils/userStartupWarnings.ts +69 -0
  420. package/dist/utils/version.ts +12 -0
  421. package/dist/validateNonInterActiveAuth.test.ts +260 -0
  422. package/dist/validateNonInterActiveAuth.ts +51 -0
  423. package/dist/vitest.config.ts +37 -0
  424. package/dist/zed-integration/acp.ts +366 -0
  425. package/dist/zed-integration/fileSystemService.ts +47 -0
  426. package/dist/zed-integration/schema.ts +466 -0
  427. package/dist/zed-integration/zedIntegration.ts +944 -0
  428. package/package.json +2 -2
@@ -0,0 +1,289 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import { promises as fs } from 'fs';
8
+ import path from 'path';
9
+ import toml from '@iarna/toml';
10
+ import { glob } from 'glob';
11
+ import { z } from 'zod';
12
+ import {
13
+ Config,
14
+ getProjectCommandsDir,
15
+ getUserCommandsDir,
16
+ } from 'fss-link-core';
17
+ import { ICommandLoader } from './types.js';
18
+ import {
19
+ CommandContext,
20
+ CommandKind,
21
+ SlashCommand,
22
+ SlashCommandActionReturn,
23
+ } from '../ui/commands/types.js';
24
+ import { DefaultArgumentProcessor } from './prompt-processors/argumentProcessor.js';
25
+ import {
26
+ IPromptProcessor,
27
+ SHORTHAND_ARGS_PLACEHOLDER,
28
+ SHELL_INJECTION_TRIGGER,
29
+ } from './prompt-processors/types.js';
30
+ import {
31
+ ConfirmationRequiredError,
32
+ ShellProcessor,
33
+ } from './prompt-processors/shellProcessor.js';
34
+
35
+ interface CommandDirectory {
36
+ path: string;
37
+ extensionName?: string;
38
+ }
39
+
40
+ /**
41
+ * Defines the Zod schema for a command definition file. This serves as the
42
+ * single source of truth for both validation and type inference.
43
+ */
44
+ const TomlCommandDefSchema = z.object({
45
+ prompt: z.string({
46
+ required_error: "The 'prompt' field is required.",
47
+ invalid_type_error: "The 'prompt' field must be a string.",
48
+ }),
49
+ description: z.string().optional(),
50
+ });
51
+
52
+ /**
53
+ * Discovers and loads custom slash commands from .toml files in both the
54
+ * user's global config directory and the current project's directory.
55
+ *
56
+ * This loader is responsible for:
57
+ * - Recursively scanning command directories.
58
+ * - Parsing and validating TOML files.
59
+ * - Adapting valid definitions into executable SlashCommand objects.
60
+ * - Handling file system errors and malformed files gracefully.
61
+ */
62
+ export class FileCommandLoader implements ICommandLoader {
63
+ private readonly projectRoot: string;
64
+
65
+ constructor(private readonly config: Config | null) {
66
+ this.projectRoot = config?.getProjectRoot() || process.cwd();
67
+ }
68
+
69
+ /**
70
+ * Loads all commands from user, project, and extension directories.
71
+ * Returns commands in order: user → project → extensions (alphabetically).
72
+ *
73
+ * Order is important for conflict resolution in CommandService:
74
+ * - User/project commands (without extensionName) use "last wins" strategy
75
+ * - Extension commands (with extensionName) get renamed if conflicts exist
76
+ *
77
+ * @param signal An AbortSignal to cancel the loading process.
78
+ * @returns A promise that resolves to an array of all loaded SlashCommands.
79
+ */
80
+ async loadCommands(signal: AbortSignal): Promise<SlashCommand[]> {
81
+ const allCommands: SlashCommand[] = [];
82
+ const globOptions = {
83
+ nodir: true,
84
+ dot: true,
85
+ signal,
86
+ follow: true,
87
+ };
88
+
89
+ // Load commands from each directory
90
+ const commandDirs = this.getCommandDirectories();
91
+ for (const dirInfo of commandDirs) {
92
+ try {
93
+ const files = await glob('**/*.toml', {
94
+ ...globOptions,
95
+ cwd: dirInfo.path,
96
+ });
97
+
98
+ const commandPromises = files.map((file) =>
99
+ this.parseAndAdaptFile(
100
+ path.join(dirInfo.path, file),
101
+ dirInfo.path,
102
+ dirInfo.extensionName,
103
+ ),
104
+ );
105
+
106
+ const commands = (await Promise.all(commandPromises)).filter(
107
+ (cmd): cmd is SlashCommand => cmd !== null,
108
+ );
109
+
110
+ // Add all commands without deduplication
111
+ allCommands.push(...commands);
112
+ } catch (error) {
113
+ if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {
114
+ console.error(
115
+ `[FileCommandLoader] Error loading commands from ${dirInfo.path}:`,
116
+ error,
117
+ );
118
+ }
119
+ }
120
+ }
121
+
122
+ return allCommands;
123
+ }
124
+
125
+ /**
126
+ * Get all command directories in order for loading.
127
+ * User commands → Project commands → Extension commands
128
+ * This order ensures extension commands can detect all conflicts.
129
+ */
130
+ private getCommandDirectories(): CommandDirectory[] {
131
+ const dirs: CommandDirectory[] = [];
132
+
133
+ // 1. User commands
134
+ dirs.push({ path: getUserCommandsDir() });
135
+
136
+ // 2. Project commands (override user commands)
137
+ dirs.push({ path: getProjectCommandsDir(this.projectRoot) });
138
+
139
+ // 3. Extension commands (processed last to detect all conflicts)
140
+ if (this.config) {
141
+ const activeExtensions = this.config
142
+ .getExtensions()
143
+ .filter((ext) => ext.isActive)
144
+ .sort((a, b) => a.name.localeCompare(b.name)); // Sort alphabetically for deterministic loading
145
+
146
+ const extensionCommandDirs = activeExtensions.map((ext) => ({
147
+ path: path.join(ext.path, 'commands'),
148
+ extensionName: ext.name,
149
+ }));
150
+
151
+ dirs.push(...extensionCommandDirs);
152
+ }
153
+
154
+ return dirs;
155
+ }
156
+
157
+ /**
158
+ * Parses a single .toml file and transforms it into a SlashCommand object.
159
+ * @param filePath The absolute path to the .toml file.
160
+ * @param baseDir The root command directory for name calculation.
161
+ * @param extensionName Optional extension name to prefix commands with.
162
+ * @returns A promise resolving to a SlashCommand, or null if the file is invalid.
163
+ */
164
+ private async parseAndAdaptFile(
165
+ filePath: string,
166
+ baseDir: string,
167
+ extensionName?: string,
168
+ ): Promise<SlashCommand | null> {
169
+ let fileContent: string;
170
+ try {
171
+ fileContent = await fs.readFile(filePath, 'utf-8');
172
+ } catch (error: unknown) {
173
+ console.error(
174
+ `[FileCommandLoader] Failed to read file ${filePath}:`,
175
+ error instanceof Error ? error.message : String(error),
176
+ );
177
+ return null;
178
+ }
179
+
180
+ let parsed: unknown;
181
+ try {
182
+ parsed = toml.parse(fileContent);
183
+ } catch (error: unknown) {
184
+ console.error(
185
+ `[FileCommandLoader] Failed to parse TOML file ${filePath}:`,
186
+ error instanceof Error ? error.message : String(error),
187
+ );
188
+ return null;
189
+ }
190
+
191
+ const validationResult = TomlCommandDefSchema.safeParse(parsed);
192
+
193
+ if (!validationResult.success) {
194
+ console.error(
195
+ `[FileCommandLoader] Skipping invalid command file: ${filePath}. Validation errors:`,
196
+ validationResult.error.flatten(),
197
+ );
198
+ return null;
199
+ }
200
+
201
+ const validDef = validationResult.data;
202
+
203
+ const relativePathWithExt = path.relative(baseDir, filePath);
204
+ const relativePath = relativePathWithExt.substring(
205
+ 0,
206
+ relativePathWithExt.length - 5, // length of '.toml'
207
+ );
208
+ const baseCommandName = relativePath
209
+ .split(path.sep)
210
+ // Sanitize each path segment to prevent ambiguity. Since ':' is our
211
+ // namespace separator, we replace any literal colons in filenames
212
+ // with underscores to avoid naming conflicts.
213
+ .map((segment) => segment.replaceAll(':', '_'))
214
+ .join(':');
215
+
216
+ // Add extension name tag for extension commands
217
+ const defaultDescription = `Custom command from ${path.basename(filePath)}`;
218
+ let description = validDef.description || defaultDescription;
219
+ if (extensionName) {
220
+ description = `[${extensionName}] ${description}`;
221
+ }
222
+
223
+ const processors: IPromptProcessor[] = [];
224
+ const usesArgs = validDef.prompt.includes(SHORTHAND_ARGS_PLACEHOLDER);
225
+ const usesShellInjection = validDef.prompt.includes(
226
+ SHELL_INJECTION_TRIGGER,
227
+ );
228
+
229
+ // Interpolation (Shell Execution and Argument Injection)
230
+ // If the prompt uses either shell injection OR argument placeholders,
231
+ // we must use the ShellProcessor.
232
+ if (usesShellInjection || usesArgs) {
233
+ processors.push(new ShellProcessor(baseCommandName));
234
+ }
235
+
236
+ // Default Argument Handling
237
+ // If NO explicit argument injection ({{args}}) was used, we append the raw invocation.
238
+ if (!usesArgs) {
239
+ processors.push(new DefaultArgumentProcessor());
240
+ }
241
+
242
+ return {
243
+ name: baseCommandName,
244
+ description,
245
+ kind: CommandKind.FILE,
246
+ extensionName,
247
+ action: async (
248
+ context: CommandContext,
249
+ _args: string,
250
+ ): Promise<SlashCommandActionReturn> => {
251
+ if (!context.invocation) {
252
+ console.error(
253
+ `[FileCommandLoader] Critical error: Command '${baseCommandName}' was executed without invocation context.`,
254
+ );
255
+ return {
256
+ type: 'submit_prompt',
257
+ content: validDef.prompt, // Fallback to unprocessed prompt
258
+ };
259
+ }
260
+
261
+ try {
262
+ let processedPrompt = validDef.prompt;
263
+ for (const processor of processors) {
264
+ processedPrompt = await processor.process(processedPrompt, context);
265
+ }
266
+
267
+ return {
268
+ type: 'submit_prompt',
269
+ content: processedPrompt,
270
+ };
271
+ } catch (e) {
272
+ // Check if it's our specific error type
273
+ if (e instanceof ConfirmationRequiredError) {
274
+ // Halt and request confirmation from the UI layer.
275
+ return {
276
+ type: 'confirm_shell_commands',
277
+ commandsToConfirm: e.commandsToConfirm,
278
+ originalInvocation: {
279
+ raw: context.invocation.raw,
280
+ },
281
+ };
282
+ }
283
+ // Re-throw other errors to be handled by the global error handler.
284
+ throw e;
285
+ }
286
+ },
287
+ };
288
+ }
289
+ }
@@ -0,0 +1,231 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import {
8
+ Config,
9
+ getErrorMessage,
10
+ getMCPServerPrompts,
11
+ } from 'fss-link-core';
12
+ import {
13
+ CommandContext,
14
+ CommandKind,
15
+ SlashCommand,
16
+ SlashCommandActionReturn,
17
+ } from '../ui/commands/types.js';
18
+ import { ICommandLoader } from './types.js';
19
+ import { PromptArgument } from '@modelcontextprotocol/sdk/types.js';
20
+
21
+ /**
22
+ * Discovers and loads executable slash commands from prompts exposed by
23
+ * Model-Context-Protocol (MCP) servers.
24
+ */
25
+ export class McpPromptLoader implements ICommandLoader {
26
+ constructor(private readonly config: Config | null) {}
27
+
28
+ /**
29
+ * Loads all available prompts from all configured MCP servers and adapts
30
+ * them into executable SlashCommand objects.
31
+ *
32
+ * @param _signal An AbortSignal (unused for this synchronous loader).
33
+ * @returns A promise that resolves to an array of loaded SlashCommands.
34
+ */
35
+ loadCommands(_signal: AbortSignal): Promise<SlashCommand[]> {
36
+ const promptCommands: SlashCommand[] = [];
37
+ if (!this.config) {
38
+ return Promise.resolve([]);
39
+ }
40
+ const mcpServers = this.config.getMcpServers() || {};
41
+ for (const serverName in mcpServers) {
42
+ const prompts = getMCPServerPrompts(this.config, serverName) || [];
43
+ for (const prompt of prompts) {
44
+ const commandName = `${prompt.name}`;
45
+ const newPromptCommand: SlashCommand = {
46
+ name: commandName,
47
+ description: prompt.description || `Invoke prompt ${prompt.name}`,
48
+ kind: CommandKind.MCP_PROMPT,
49
+ subCommands: [
50
+ {
51
+ name: 'help',
52
+ description: 'Show help for this prompt',
53
+ kind: CommandKind.MCP_PROMPT,
54
+ action: async (): Promise<SlashCommandActionReturn> => {
55
+ if (!prompt.arguments || prompt.arguments.length === 0) {
56
+ return {
57
+ type: 'message',
58
+ messageType: 'info',
59
+ content: `Prompt "${prompt.name}" has no arguments.`,
60
+ };
61
+ }
62
+
63
+ let helpMessage = `Arguments for "${prompt.name}":\n\n`;
64
+ if (prompt.arguments && prompt.arguments.length > 0) {
65
+ helpMessage += `You can provide arguments by name (e.g., --argName="value") or by position.\n\n`;
66
+ helpMessage += `e.g., ${prompt.name} ${prompt.arguments?.map((_) => `"foo"`)} is equivalent to ${prompt.name} ${prompt.arguments?.map((arg) => `--${arg.name}="foo"`)}\n\n`;
67
+ }
68
+ for (const arg of prompt.arguments) {
69
+ helpMessage += ` --${arg.name}\n`;
70
+ if (arg.description) {
71
+ helpMessage += ` ${arg.description}\n`;
72
+ }
73
+ helpMessage += ` (required: ${
74
+ arg.required ? 'yes' : 'no'
75
+ })\n\n`;
76
+ }
77
+ return {
78
+ type: 'message',
79
+ messageType: 'info',
80
+ content: helpMessage,
81
+ };
82
+ },
83
+ },
84
+ ],
85
+ action: async (
86
+ context: CommandContext,
87
+ args: string,
88
+ ): Promise<SlashCommandActionReturn> => {
89
+ if (!this.config) {
90
+ return {
91
+ type: 'message',
92
+ messageType: 'error',
93
+ content: 'Config not loaded.',
94
+ };
95
+ }
96
+
97
+ const promptInputs = this.parseArgs(args, prompt.arguments);
98
+ if (promptInputs instanceof Error) {
99
+ return {
100
+ type: 'message',
101
+ messageType: 'error',
102
+ content: promptInputs.message,
103
+ };
104
+ }
105
+
106
+ try {
107
+ const mcpServers = this.config.getMcpServers() || {};
108
+ const mcpServerConfig = mcpServers[serverName];
109
+ if (!mcpServerConfig) {
110
+ return {
111
+ type: 'message',
112
+ messageType: 'error',
113
+ content: `MCP server config not found for '${serverName}'.`,
114
+ };
115
+ }
116
+ const result = await prompt.invoke(promptInputs);
117
+
118
+ if (result['error']) {
119
+ return {
120
+ type: 'message',
121
+ messageType: 'error',
122
+ content: `Error invoking prompt: ${result['error']}`,
123
+ };
124
+ }
125
+
126
+ if (!result.messages?.[0]?.content?.['text']) {
127
+ return {
128
+ type: 'message',
129
+ messageType: 'error',
130
+ content:
131
+ 'Received an empty or invalid prompt response from the server.',
132
+ };
133
+ }
134
+
135
+ return {
136
+ type: 'submit_prompt',
137
+ content: JSON.stringify(result.messages[0].content.text),
138
+ };
139
+ } catch (error) {
140
+ return {
141
+ type: 'message',
142
+ messageType: 'error',
143
+ content: `Error: ${getErrorMessage(error)}`,
144
+ };
145
+ }
146
+ },
147
+ completion: async (_: CommandContext, partialArg: string) => {
148
+ if (!prompt || !prompt.arguments) {
149
+ return [];
150
+ }
151
+
152
+ const suggestions: string[] = [];
153
+ const usedArgNames = new Set(
154
+ (partialArg.match(/--([^=]+)/g) || []).map((s) => s.substring(2)),
155
+ );
156
+
157
+ for (const arg of prompt.arguments) {
158
+ if (!usedArgNames.has(arg.name)) {
159
+ suggestions.push(`--${arg.name}=""`);
160
+ }
161
+ }
162
+
163
+ return suggestions;
164
+ },
165
+ };
166
+ promptCommands.push(newPromptCommand);
167
+ }
168
+ }
169
+ return Promise.resolve(promptCommands);
170
+ }
171
+
172
+ private parseArgs(
173
+ userArgs: string,
174
+ promptArgs: PromptArgument[] | undefined,
175
+ ): Record<string, unknown> | Error {
176
+ const argValues: { [key: string]: string } = {};
177
+ const promptInputs: Record<string, unknown> = {};
178
+
179
+ // arg parsing: --key="value" or --key=value
180
+ const namedArgRegex = /--([^=]+)=(?:"((?:\\.|[^"\\])*)"|([^ ]*))/g;
181
+ let match;
182
+ const remainingArgs: string[] = [];
183
+ let lastIndex = 0;
184
+
185
+ while ((match = namedArgRegex.exec(userArgs)) !== null) {
186
+ const key = match[1];
187
+ const value = match[2] ?? match[3]; // Quoted or unquoted value
188
+ argValues[key] = value;
189
+ // Capture text between matches as potential positional args
190
+ if (match.index > lastIndex) {
191
+ remainingArgs.push(userArgs.substring(lastIndex, match.index).trim());
192
+ }
193
+ lastIndex = namedArgRegex.lastIndex;
194
+ }
195
+
196
+ // Capture any remaining text after the last named arg
197
+ if (lastIndex < userArgs.length) {
198
+ remainingArgs.push(userArgs.substring(lastIndex).trim());
199
+ }
200
+
201
+ const positionalArgs = remainingArgs.join(' ').split(/ +/);
202
+
203
+ if (!promptArgs) {
204
+ return promptInputs;
205
+ }
206
+ for (const arg of promptArgs) {
207
+ if (argValues[arg.name]) {
208
+ promptInputs[arg.name] = argValues[arg.name];
209
+ }
210
+ }
211
+
212
+ const unfilledArgs = promptArgs.filter(
213
+ (arg) => arg.required && !promptInputs[arg.name],
214
+ );
215
+
216
+ const missingArgs: string[] = [];
217
+ for (let i = 0; i < unfilledArgs.length; i++) {
218
+ if (positionalArgs.length > i && positionalArgs[i]) {
219
+ promptInputs[unfilledArgs[i].name] = positionalArgs[i];
220
+ } else {
221
+ missingArgs.push(unfilledArgs[i].name);
222
+ }
223
+ }
224
+
225
+ if (missingArgs.length > 0) {
226
+ const missingArgNames = missingArgs.map((name) => `--${name}`).join(', ');
227
+ return new Error(`Missing required argument(s): ${missingArgNames}`);
228
+ }
229
+ return promptInputs;
230
+ }
231
+ }
@@ -0,0 +1,100 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import { getFSSLinkDatabase } from '../config/database.js';
8
+
9
+ export interface SearchEngineConfig {
10
+ braveApiKey?: string;
11
+ tavilyApiKey?: string;
12
+ }
13
+
14
+ /**
15
+ * Configuration provider for search engine API keys
16
+ * Loads from database and sets environment variables
17
+ */
18
+ export class SearchEngineConfigProvider {
19
+ private static instance: SearchEngineConfigProvider | null = null;
20
+
21
+ static getInstance(): SearchEngineConfigProvider {
22
+ if (!SearchEngineConfigProvider.instance) {
23
+ SearchEngineConfigProvider.instance = new SearchEngineConfigProvider();
24
+ }
25
+ return SearchEngineConfigProvider.instance;
26
+ }
27
+
28
+ /**
29
+ * Load search engine configuration from database and set environment variables
30
+ */
31
+ async loadConfiguration(): Promise<SearchEngineConfig> {
32
+ const db = await getFSSLinkDatabase();
33
+
34
+ const braveApiKey = await db.getUserPreference('webscraper.brave_api_key');
35
+ const tavilyApiKey = await db.getUserPreference('webscraper.tavily_api_key');
36
+
37
+ // Set environment variables for immediate use
38
+ if (braveApiKey) {
39
+ process.env['BRAVE_SEARCH_API_KEY'] = braveApiKey;
40
+ }
41
+ if (tavilyApiKey) {
42
+ process.env['TAVILY_API_KEY'] = tavilyApiKey;
43
+ }
44
+
45
+ return {
46
+ braveApiKey: braveApiKey || undefined,
47
+ tavilyApiKey: tavilyApiKey || undefined,
48
+ };
49
+ }
50
+
51
+ /**
52
+ * Save search engine configuration to database
53
+ */
54
+ async saveConfiguration(config: SearchEngineConfig): Promise<void> {
55
+ const db = await getFSSLinkDatabase();
56
+
57
+ if (config.braveApiKey) {
58
+ await db.setUserPreference('webscraper.brave_api_key', config.braveApiKey);
59
+ await db.setUserPreference('webscraper.brave_enabled', 'true');
60
+ process.env['BRAVE_SEARCH_API_KEY'] = config.braveApiKey;
61
+ }
62
+
63
+ if (config.tavilyApiKey) {
64
+ await db.setUserPreference('webscraper.tavily_api_key', config.tavilyApiKey);
65
+ await db.setUserPreference('webscraper.tavily_enabled', 'true');
66
+ process.env['TAVILY_API_KEY'] = config.tavilyApiKey;
67
+ }
68
+ }
69
+
70
+ /**
71
+ * Check if search engines are configured
72
+ */
73
+ async hasConfiguration(): Promise<boolean> {
74
+ const config = await this.loadConfiguration();
75
+ return !!(config.braveApiKey || config.tavilyApiKey);
76
+ }
77
+
78
+ /**
79
+ * Get configuration status for display
80
+ */
81
+ async getConfigurationStatus(): Promise<string> {
82
+ const config = await this.loadConfiguration();
83
+
84
+ let status = '🔍 Search Engine Configuration:\n';
85
+
86
+ if (config.braveApiKey) {
87
+ status += `✅ Brave Search: Configured\n`;
88
+ } else {
89
+ status += `❌ Brave Search: Not configured\n`;
90
+ }
91
+
92
+ if (config.tavilyApiKey) {
93
+ status += `✅ Tavily Search: Configured\n`;
94
+ } else {
95
+ status += `❌ Tavily Search: Not configured\n`;
96
+ }
97
+
98
+ return status;
99
+ }
100
+ }
@@ -0,0 +1,41 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import { DefaultArgumentProcessor } from './argumentProcessor.js';
8
+ import { createMockCommandContext } from '../../test-utils/mockCommandContext.js';
9
+ import { describe, it, expect } from 'vitest';
10
+
11
+ describe('Argument Processors', () => {
12
+ describe('DefaultArgumentProcessor', () => {
13
+ const processor = new DefaultArgumentProcessor();
14
+
15
+ it('should append the full command if args are provided', async () => {
16
+ const prompt = 'Parse the command.';
17
+ const context = createMockCommandContext({
18
+ invocation: {
19
+ raw: '/mycommand arg1 "arg two"',
20
+ name: 'mycommand',
21
+ args: 'arg1 "arg two"',
22
+ },
23
+ });
24
+ const result = await processor.process(prompt, context);
25
+ expect(result).toBe('Parse the command.\n\n/mycommand arg1 "arg two"');
26
+ });
27
+
28
+ it('should NOT append the full command if no args are provided', async () => {
29
+ const prompt = 'Parse the command.';
30
+ const context = createMockCommandContext({
31
+ invocation: {
32
+ raw: '/mycommand',
33
+ name: 'mycommand',
34
+ args: '',
35
+ },
36
+ });
37
+ const result = await processor.process(prompt, context);
38
+ expect(result).toBe('Parse the command.');
39
+ });
40
+ });
41
+ });
@@ -0,0 +1,23 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import { IPromptProcessor } from './types.js';
8
+ import { CommandContext } from '../../ui/commands/types.js';
9
+
10
+ /**
11
+ * Appends the user's full command invocation to the prompt if arguments are
12
+ * provided, allowing the model to perform its own argument parsing.
13
+ *
14
+ * This processor is only used if the prompt does NOT contain {{args}}.
15
+ */
16
+ export class DefaultArgumentProcessor implements IPromptProcessor {
17
+ async process(prompt: string, context: CommandContext): Promise<string> {
18
+ if (context.invocation!.args) {
19
+ return `${prompt}\n\n${context.invocation!.raw}`;
20
+ }
21
+ return prompt;
22
+ }
23
+ }