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,377 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
8
+ import { AuthType, ModelConfig } from 'fss-link-core';
9
+ import { ProviderManager } from './providerManager.js';
10
+ import { getFSSLinkDatabase, closeFSSLinkDatabase } from './database.js';
11
+ import * as fs from 'fs';
12
+ import * as path from 'path';
13
+ import * as os from 'os';
14
+
15
+ describe('Provider Persistence System', () => {
16
+ let providerManager: ProviderManager;
17
+ let testDbPath: string;
18
+
19
+ beforeEach(async () => {
20
+ // Create temporary database for testing
21
+ testDbPath = path.join(os.tmpdir(), `test-fss-link-${Date.now()}.db`);
22
+
23
+ // Override the database path for testing
24
+ process.env.FSS_LINK_TEST_DB = testDbPath;
25
+
26
+ providerManager = ProviderManager.getInstance();
27
+
28
+ // Initialize clean database
29
+ const db = await getFSSLinkDatabase();
30
+ await db.initialize();
31
+ });
32
+
33
+ afterEach(async () => {
34
+ // Clean up test database
35
+ closeFSSLinkDatabase();
36
+ if (fs.existsSync(testDbPath)) {
37
+ fs.unlinkSync(testDbPath);
38
+ }
39
+ delete process.env.FSS_LINK_TEST_DB;
40
+ });
41
+
42
+ describe('Provider Configuration Management', () => {
43
+ it('should save and retrieve provider configurations', async () => {
44
+ const testConfig: ModelConfig = {
45
+ authType: AuthType.OPENAI_API_KEY,
46
+ modelName: 'gpt-4',
47
+ endpointUrl: 'https://api.openai.com/v1',
48
+ apiKey: 'test-key-123',
49
+ displayName: 'GPT-4 Test',
50
+ isFavorite: false,
51
+ isActive: true
52
+ };
53
+
54
+ const providerId = await providerManager.saveProvider(testConfig);
55
+ expect(providerId).toBeGreaterThan(0);
56
+
57
+ const allProviders = await providerManager.getAllProviders();
58
+ expect(allProviders).toHaveLength(1);
59
+ expect(allProviders[0].modelName).toBe('gpt-4');
60
+ expect(allProviders[0].displayName).toBe('GPT-4 Test');
61
+ });
62
+
63
+ it('should handle provider updates correctly', async () => {
64
+ const testConfig: ModelConfig = {
65
+ authType: AuthType.OLLAMA,
66
+ modelName: 'llama2',
67
+ endpointUrl: 'http://localhost:11434/v1',
68
+ displayName: 'Llama 2 Local',
69
+ isFavorite: false,
70
+ isActive: true
71
+ };
72
+
73
+ const providerId = await providerManager.saveProvider(testConfig);
74
+
75
+ // Update the provider
76
+ await providerManager.updateProvider(providerId, {
77
+ modelName: 'llama2-updated',
78
+ isFavorite: true
79
+ });
80
+
81
+ const allProviders = await providerManager.getAllProviders();
82
+ const updated = allProviders.find(p => p.id === providerId);
83
+
84
+ expect(updated?.modelName).toBe('llama2-updated');
85
+ expect(updated?.isFavorite).toBe(true);
86
+ expect(updated?.displayName).toBe('Llama 2 Local'); // Should remain unchanged
87
+ });
88
+
89
+ it('should track usage statistics correctly', async () => {
90
+ const testConfig: ModelConfig = {
91
+ authType: AuthType.OPENAI_API_KEY,
92
+ modelName: 'gpt-3.5-turbo',
93
+ displayName: 'GPT-3.5 Test',
94
+ isActive: true
95
+ };
96
+
97
+ const providerId = await providerManager.saveProvider(testConfig);
98
+
99
+ // Record multiple usage sessions
100
+ await providerManager.recordUsage(providerId, 1000, 30000); // 1000 tokens, 30 seconds
101
+ await providerManager.recordUsage(providerId, 500, 15000); // 500 tokens, 15 seconds
102
+ await providerManager.recordUsage(providerId, 2000, 60000); // 2000 tokens, 60 seconds
103
+
104
+ const stats = await providerManager.getUsageStatistics();
105
+ expect(stats[providerId]).toBeDefined();
106
+ // Note: Actual validation depends on database implementation
107
+ });
108
+ });
109
+
110
+ describe('Custom Endpoints Management', () => {
111
+ it('should save and retrieve custom endpoints', async () => {
112
+ const testConfig: ModelConfig = {
113
+ authType: AuthType.OPENAI_API_KEY,
114
+ modelName: 'custom-model',
115
+ displayName: 'Custom Provider',
116
+ isActive: true
117
+ };
118
+
119
+ const providerId = await providerManager.saveProvider(testConfig);
120
+
121
+ // Add custom endpoints
122
+ const endpoint1Id = await providerManager.saveCustomEndpoint(
123
+ providerId,
124
+ 'Production API',
125
+ 'https://api.custom.com/v1',
126
+ 'Production endpoint',
127
+ true
128
+ );
129
+
130
+ const endpoint2Id = await providerManager.saveCustomEndpoint(
131
+ providerId,
132
+ 'Staging API',
133
+ 'https://staging-api.custom.com/v1',
134
+ 'Staging endpoint',
135
+ false
136
+ );
137
+
138
+ expect(endpoint1Id).toBeGreaterThan(0);
139
+ expect(endpoint2Id).toBeGreaterThan(0);
140
+
141
+ const endpoints = await providerManager.getCustomEndpoints(providerId);
142
+ expect(endpoints).toHaveLength(2);
143
+
144
+ const defaultEndpoint = endpoints.find(e => e.is_default);
145
+ expect(defaultEndpoint?.name).toBe('Production API');
146
+ });
147
+
148
+ it('should handle default endpoint switching', async () => {
149
+ const testConfig: ModelConfig = {
150
+ authType: AuthType.OPENAI_API_KEY,
151
+ modelName: 'test-model',
152
+ isActive: true
153
+ };
154
+
155
+ const providerId = await providerManager.saveProvider(testConfig);
156
+
157
+ // Add first endpoint as default
158
+ await providerManager.saveCustomEndpoint(
159
+ providerId,
160
+ 'Endpoint 1',
161
+ 'https://api1.com/v1',
162
+ undefined,
163
+ true
164
+ );
165
+
166
+ // Add second endpoint as default (should unset first)
167
+ await providerManager.saveCustomEndpoint(
168
+ providerId,
169
+ 'Endpoint 2',
170
+ 'https://api2.com/v1',
171
+ undefined,
172
+ true
173
+ );
174
+
175
+ const endpoints = await providerManager.getCustomEndpoints(providerId);
176
+ const defaultEndpoints = endpoints.filter(e => e.is_default);
177
+
178
+ expect(defaultEndpoints).toHaveLength(1);
179
+ expect(defaultEndpoints[0].name).toBe('Endpoint 2');
180
+ });
181
+ });
182
+
183
+ describe('Provider Settings Management', () => {
184
+ it('should save and retrieve provider-specific settings', async () => {
185
+ const testConfig: ModelConfig = {
186
+ authType: AuthType.OLLAMA,
187
+ modelName: 'test-model',
188
+ isActive: true
189
+ };
190
+
191
+ const providerId = await providerManager.saveProvider(testConfig);
192
+
193
+ // Save various settings
194
+ await providerManager.saveProviderSetting(providerId, 'num_ctx', 32768);
195
+ await providerManager.saveProviderSetting(providerId, 'temperature', 0.7);
196
+ await providerManager.saveProviderSetting(providerId, 'custom_prompt', 'You are a helpful assistant');
197
+
198
+ const settings = await providerManager.getProviderSettings(providerId);
199
+
200
+ expect(settings.num_ctx).toBe(32768);
201
+ expect(settings.temperature).toBe(0.7);
202
+ expect(settings.custom_prompt).toBe('You are a helpful assistant');
203
+ });
204
+
205
+ it('should handle complex setting values', async () => {
206
+ const testConfig: ModelConfig = {
207
+ authType: AuthType.OPENAI_API_KEY,
208
+ modelName: 'complex-config',
209
+ isActive: true
210
+ };
211
+
212
+ const providerId = await providerManager.saveProvider(testConfig);
213
+
214
+ const complexSetting = {
215
+ samplingParams: {
216
+ temperature: 0.8,
217
+ max_tokens: 2000,
218
+ top_p: 0.9
219
+ },
220
+ customHeaders: {
221
+ 'X-Custom-Header': 'value'
222
+ }
223
+ };
224
+
225
+ await providerManager.saveProviderSetting(providerId, 'advanced_config', complexSetting);
226
+
227
+ const settings = await providerManager.getProviderSettings(providerId);
228
+ expect(settings.advanced_config).toEqual(complexSetting);
229
+ });
230
+ });
231
+
232
+ describe('Smart Default Selection', () => {
233
+ it('should select favorite provider as default', async () => {
234
+ // Create multiple providers
235
+ const _provider1Id = await providerManager.saveProvider({
236
+ authType: AuthType.OPENAI_API_KEY,
237
+ modelName: 'gpt-4',
238
+ displayName: 'GPT-4',
239
+ isFavorite: false,
240
+ isActive: true
241
+ });
242
+
243
+ const _provider2Id = await providerManager.saveProvider({
244
+ authType: AuthType.OLLAMA,
245
+ modelName: 'llama2',
246
+ displayName: 'Llama 2',
247
+ isFavorite: true, // This should be selected as default
248
+ isActive: true
249
+ });
250
+
251
+ const provider3Id = await providerManager.saveProvider({
252
+ authType: AuthType.LM_STUDIO,
253
+ modelName: 'local-model',
254
+ displayName: 'Local Model',
255
+ isFavorite: false,
256
+ isActive: true
257
+ });
258
+
259
+ // Record usage for provider3 to make it most used
260
+ await providerManager.recordUsage(provider3Id, 5000, 120000);
261
+ await providerManager.recordUsage(provider3Id, 3000, 90000);
262
+
263
+ const defaultProvider = await providerManager.getDefaultProvider();
264
+ expect(defaultProvider?.displayName).toBe('Llama 2'); // Should prefer favorite over usage
265
+ });
266
+
267
+ it('should select most used provider when no favorite is set', async () => {
268
+ const provider1Id = await providerManager.saveProvider({
269
+ authType: AuthType.OPENAI_API_KEY,
270
+ modelName: 'gpt-4',
271
+ displayName: 'GPT-4',
272
+ isActive: true
273
+ });
274
+
275
+ const provider2Id = await providerManager.saveProvider({
276
+ authType: AuthType.OLLAMA,
277
+ modelName: 'llama2',
278
+ displayName: 'Llama 2',
279
+ isActive: true
280
+ });
281
+
282
+ // Make provider2 most used
283
+ await providerManager.recordUsage(provider2Id, 10000, 300000);
284
+ await providerManager.recordUsage(provider2Id, 8000, 200000);
285
+ await providerManager.recordUsage(provider1Id, 2000, 60000);
286
+
287
+ const defaultProvider = await providerManager.getDefaultProvider();
288
+ expect(defaultProvider?.displayName).toBe('Llama 2');
289
+ });
290
+ });
291
+
292
+ describe('Provider Filtering and Querying', () => {
293
+ it('should filter providers by type correctly', async () => {
294
+ // Create providers of different types
295
+ await providerManager.saveProvider({
296
+ authType: AuthType.OPENAI_API_KEY,
297
+ modelName: 'gpt-4',
298
+ isActive: true
299
+ });
300
+
301
+ await providerManager.saveProvider({
302
+ authType: AuthType.OLLAMA,
303
+ modelName: 'llama2',
304
+ isActive: true
305
+ });
306
+
307
+ await providerManager.saveProvider({
308
+ authType: AuthType.OPENAI_API_KEY,
309
+ modelName: 'gpt-3.5-turbo',
310
+ isActive: true
311
+ });
312
+
313
+ const openaiProviders = await providerManager.getProvidersByType(AuthType.OPENAI_API_KEY);
314
+ const ollamaProviders = await providerManager.getProvidersByType(AuthType.OLLAMA);
315
+
316
+ expect(openaiProviders).toHaveLength(2);
317
+ expect(ollamaProviders).toHaveLength(1);
318
+ expect(ollamaProviders[0].modelName).toBe('llama2');
319
+ });
320
+
321
+ it('should get recommended providers based on usage', async () => {
322
+ // Create several providers with different usage patterns
323
+ const providers = [];
324
+ for (let i = 0; i < 8; i++) {
325
+ const id = await providerManager.saveProvider({
326
+ authType: AuthType.OPENAI_API_KEY,
327
+ modelName: `model-${i}`,
328
+ displayName: `Model ${i}`,
329
+ isActive: true,
330
+ isFavorite: i === 2 // Make one a favorite
331
+ });
332
+ providers.push(id);
333
+
334
+ // Record different amounts of usage
335
+ if (i < 5) {
336
+ await providerManager.recordUsage(id, i * 1000, i * 30000);
337
+ }
338
+ }
339
+
340
+ const recommended = await providerManager.getRecommendedProviders(3);
341
+ expect(recommended).toHaveLength(3);
342
+
343
+ // First should be the favorite
344
+ expect(recommended[0].displayName).toBe('Model 2');
345
+ });
346
+ });
347
+
348
+ describe('Auto-Configuration', () => {
349
+ it('should auto-configure provider from successful authentication', async () => {
350
+ const providerId = await providerManager.autoConfigureFromSuccess(
351
+ AuthType.USE_GEMINI,
352
+ 'gemini-2.0-flash',
353
+ undefined,
354
+ 'test-gemini-key',
355
+ {
356
+ temperature: 0.7,
357
+ max_tokens: 2048
358
+ }
359
+ );
360
+
361
+ expect(providerId).toBeGreaterThan(0);
362
+
363
+ const allProviders = await providerManager.getAllProviders();
364
+ const newProvider = allProviders.find(p => p.id === providerId);
365
+
366
+ expect(newProvider?.authType).toBe(AuthType.USE_GEMINI);
367
+ expect(newProvider?.modelName).toBe('gemini-2.0-flash');
368
+ expect(newProvider?.apiKey).toBe('test-gemini-key');
369
+ expect(newProvider?.displayName).toContain('gemini-2.0-flash');
370
+
371
+ // Check additional settings were saved
372
+ const settings = await providerManager.getProviderSettings(providerId);
373
+ expect(settings.temperature).toBe(0.7);
374
+ expect(settings.max_tokens).toBe(2048);
375
+ });
376
+ });
377
+ });
@@ -0,0 +1,105 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import { AuthType } from 'fss-link-core';
8
+
9
+ /**
10
+ * Provider configuration with full persistence support
11
+ */
12
+ export interface ProviderConfig {
13
+ id?: number;
14
+ provider: AuthType;
15
+ displayName: string;
16
+ baseUrl?: string;
17
+ apiKey?: string;
18
+ model?: string;
19
+ isDefault: boolean;
20
+ usageCount: number;
21
+ lastUsed: Date;
22
+ customEndpoints?: CustomEndpoint[];
23
+ additionalSettings?: Record<string, unknown>;
24
+ }
25
+
26
+ /**
27
+ * Custom endpoint configuration
28
+ */
29
+ export interface CustomEndpoint {
30
+ id?: number;
31
+ providerId: number;
32
+ name: string;
33
+ url: string;
34
+ description?: string;
35
+ isDefault: boolean;
36
+ created: Date;
37
+ }
38
+
39
+ /**
40
+ * Provider usage tracking
41
+ */
42
+ export interface ProviderUsage {
43
+ providerId: number;
44
+ sessionCount: number;
45
+ totalTokens: number;
46
+ lastSession: Date;
47
+ averageSessionLength: number;
48
+ }
49
+
50
+ /**
51
+ * Enhanced provider persistence manager
52
+ */
53
+ export class ProviderPersistenceManager {
54
+ private static instance: ProviderPersistenceManager;
55
+
56
+ static getInstance(): ProviderPersistenceManager {
57
+ if (!ProviderPersistenceManager.instance) {
58
+ ProviderPersistenceManager.instance = new ProviderPersistenceManager();
59
+ }
60
+ return ProviderPersistenceManager.instance;
61
+ }
62
+
63
+ /**
64
+ * Save provider configuration with usage tracking
65
+ */
66
+ async saveProviderConfig(_config: ProviderConfig): Promise<number> {
67
+ // Implementation will integrate with existing database
68
+ throw new Error('Implementation pending');
69
+ }
70
+
71
+ /**
72
+ * Get all saved providers with usage statistics
73
+ */
74
+ async getAllProviders(): Promise<ProviderConfig[]> {
75
+ throw new Error('Implementation pending');
76
+ }
77
+
78
+ /**
79
+ * Get smart default provider based on usage and configuration
80
+ */
81
+ async getDefaultProvider(): Promise<ProviderConfig | null> {
82
+ throw new Error('Implementation pending');
83
+ }
84
+
85
+ /**
86
+ * Track provider usage for intelligent defaults
87
+ */
88
+ async recordProviderUsage(_providerId: number, _tokens: number, _sessionDuration: number): Promise<void> {
89
+ throw new Error('Implementation pending');
90
+ }
91
+
92
+ /**
93
+ * Get custom endpoints for a provider
94
+ */
95
+ async getCustomEndpoints(_providerId: number): Promise<CustomEndpoint[]> {
96
+ throw new Error('Implementation pending');
97
+ }
98
+
99
+ /**
100
+ * Save custom endpoint
101
+ */
102
+ async saveCustomEndpoint(_endpoint: CustomEndpoint): Promise<number> {
103
+ throw new Error('Implementation pending');
104
+ }
105
+ }
@@ -0,0 +1,107 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import { SandboxConfig } from 'fss-link-core';
8
+ import commandExists from 'command-exists';
9
+ import * as os from 'node:os';
10
+ import { getPackageJson } from '../utils/package.js';
11
+ import { Settings } from './settings.js';
12
+
13
+ // This is a stripped-down version of the CliArgs interface from config.ts
14
+ // to avoid circular dependencies.
15
+ interface SandboxCliArgs {
16
+ sandbox?: boolean | string;
17
+ sandboxImage?: string;
18
+ }
19
+
20
+ const VALID_SANDBOX_COMMANDS: ReadonlyArray<SandboxConfig['command']> = [
21
+ 'docker',
22
+ 'podman',
23
+ 'sandbox-exec',
24
+ ];
25
+
26
+ function isSandboxCommand(value: string): value is SandboxConfig['command'] {
27
+ return (VALID_SANDBOX_COMMANDS as readonly string[]).includes(value);
28
+ }
29
+
30
+ function getSandboxCommand(
31
+ sandbox?: boolean | string,
32
+ ): SandboxConfig['command'] | '' {
33
+ // If the SANDBOX env var is set, we're already inside the sandbox.
34
+ if (process.env['SANDBOX']) {
35
+ return '';
36
+ }
37
+
38
+ // note environment variable takes precedence over argument (from command line or settings)
39
+ const environmentConfiguredSandbox =
40
+ process.env['GEMINI_SANDBOX']?.toLowerCase().trim() ?? '';
41
+ sandbox =
42
+ environmentConfiguredSandbox?.length > 0
43
+ ? environmentConfiguredSandbox
44
+ : sandbox;
45
+ if (sandbox === '1' || sandbox === 'true') sandbox = true;
46
+ else if (sandbox === '0' || sandbox === 'false' || !sandbox) sandbox = false;
47
+
48
+ if (sandbox === false) {
49
+ return '';
50
+ }
51
+
52
+ if (typeof sandbox === 'string' && sandbox) {
53
+ if (!isSandboxCommand(sandbox)) {
54
+ console.error(
55
+ `ERROR: invalid sandbox command '${sandbox}'. Must be one of ${VALID_SANDBOX_COMMANDS.join(
56
+ ', ',
57
+ )}`,
58
+ );
59
+ process.exit(1);
60
+ }
61
+ // confirm that specified command exists
62
+ if (commandExists.sync(sandbox)) {
63
+ return sandbox;
64
+ }
65
+ console.error(
66
+ `ERROR: missing sandbox command '${sandbox}' (from GEMINI_SANDBOX)`,
67
+ );
68
+ process.exit(1);
69
+ }
70
+
71
+ // look for seatbelt, docker, or podman, in that order
72
+ // for container-based sandboxing, require sandbox to be enabled explicitly
73
+ if (os.platform() === 'darwin' && commandExists.sync('sandbox-exec')) {
74
+ return 'sandbox-exec';
75
+ } else if (commandExists.sync('docker') && sandbox === true) {
76
+ return 'docker';
77
+ } else if (commandExists.sync('podman') && sandbox === true) {
78
+ return 'podman';
79
+ }
80
+
81
+ // throw an error if user requested sandbox but no command was found
82
+ if (sandbox === true) {
83
+ console.error(
84
+ 'ERROR: GEMINI_SANDBOX is true but failed to determine command for sandbox; ' +
85
+ 'install docker or podman or specify command in GEMINI_SANDBOX',
86
+ );
87
+ process.exit(1);
88
+ }
89
+
90
+ return '';
91
+ }
92
+
93
+ export async function loadSandboxConfig(
94
+ settings: Settings,
95
+ argv: SandboxCliArgs,
96
+ ): Promise<SandboxConfig | undefined> {
97
+ const sandboxOption = argv.sandbox ?? settings.sandbox;
98
+ const command = getSandboxCommand(sandboxOption);
99
+
100
+ const packageJson = await getPackageJson();
101
+ const image =
102
+ argv.sandboxImage ??
103
+ process.env['GEMINI_SANDBOX_IMAGE'] ??
104
+ packageJson?.config?.sandboxImageUri;
105
+
106
+ return command && image ? { command, image } : undefined;
107
+ }