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,517 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+
7
+ import * as fs from 'fs';
8
+ import * as path from 'path';
9
+ import { homedir, platform } from 'os';
10
+ import * as dotenv from 'dotenv';
11
+ import {
12
+ FSS_LINK_CONFIG_DIR as GEMINI_DIR,
13
+ getErrorMessage,
14
+ } from 'fss-link-core';
15
+ import stripJsonComments from 'strip-json-comments';
16
+ import { DefaultLight } from '../ui/themes/default-light.js';
17
+ import { DefaultDark } from '../ui/themes/default.js';
18
+ import { Settings, MemoryImportFormat } from './settingsSchema.js';
19
+ import { getModelManager } from './modelManager.js';
20
+
21
+ export type { Settings, MemoryImportFormat };
22
+
23
+ export const SETTINGS_DIRECTORY_NAME = '.fss-link';
24
+ export const USER_SETTINGS_DIR = path.join(homedir(), SETTINGS_DIRECTORY_NAME);
25
+ export const USER_SETTINGS_PATH = path.join(USER_SETTINGS_DIR, 'settings.json');
26
+ export const DEFAULT_EXCLUDED_ENV_VARS = ['DEBUG', 'DEBUG_MODE'];
27
+
28
+ export function getSystemSettingsPath(): string {
29
+ if (process.env['FSS_LINK_CLI_SYSTEM_SETTINGS_PATH']) {
30
+ return process.env['FSS_LINK_CLI_SYSTEM_SETTINGS_PATH'];
31
+ }
32
+ if (platform() === 'darwin') {
33
+ return '/Library/Application Support/FSS-Link/settings.json';
34
+ } else if (platform() === 'win32') {
35
+ return 'C:\\ProgramData\\fss-link\\settings.json';
36
+ } else {
37
+ return '/etc/fss-link/settings.json';
38
+ }
39
+ }
40
+
41
+ export function getWorkspaceSettingsPath(workspaceDir: string): string {
42
+ return path.join(workspaceDir, SETTINGS_DIRECTORY_NAME, 'settings.json');
43
+ }
44
+
45
+ export type { DnsResolutionOrder } from './settingsSchema.js';
46
+
47
+ export enum SettingScope {
48
+ User = 'User',
49
+ Workspace = 'Workspace',
50
+ System = 'System',
51
+ }
52
+
53
+ export interface CheckpointingSettings {
54
+ enabled?: boolean;
55
+ }
56
+
57
+ export interface SummarizeToolOutputSettings {
58
+ tokenBudget?: number;
59
+ }
60
+
61
+ export interface AccessibilitySettings {
62
+ disableLoadingPhrases?: boolean;
63
+ }
64
+
65
+ export interface SettingsError {
66
+ message: string;
67
+ path: string;
68
+ }
69
+
70
+ export interface SettingsFile {
71
+ settings: Settings;
72
+ path: string;
73
+ }
74
+
75
+ function mergeSettings(
76
+ system: Settings,
77
+ user: Settings,
78
+ workspace: Settings,
79
+ ): Settings {
80
+ // folderTrust is not supported at workspace level.
81
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
82
+ const { folderTrust, ...workspaceWithoutFolderTrust } = workspace;
83
+
84
+ return {
85
+ ...user,
86
+ ...workspaceWithoutFolderTrust,
87
+ ...system,
88
+ customThemes: {
89
+ ...(user.customThemes || {}),
90
+ ...(workspace.customThemes || {}),
91
+ ...(system.customThemes || {}),
92
+ },
93
+ mcpServers: {
94
+ ...(user.mcpServers || {}),
95
+ ...(workspace.mcpServers || {}),
96
+ ...(system.mcpServers || {}),
97
+ },
98
+ includeDirectories: [
99
+ ...(system.includeDirectories || []),
100
+ ...(user.includeDirectories || []),
101
+ ...(workspace.includeDirectories || []),
102
+ ],
103
+ chatCompression: {
104
+ ...(system.chatCompression || {}),
105
+ ...(user.chatCompression || {}),
106
+ ...(workspace.chatCompression || {}),
107
+ },
108
+ };
109
+ }
110
+
111
+ export class LoadedSettings {
112
+ constructor(
113
+ system: SettingsFile,
114
+ user: SettingsFile,
115
+ workspace: SettingsFile,
116
+ errors: SettingsError[],
117
+ ) {
118
+ this.system = system;
119
+ this.user = user;
120
+ this.workspace = workspace;
121
+ this.errors = errors;
122
+ this._merged = this.computeMergedSettings();
123
+ }
124
+
125
+ readonly system: SettingsFile;
126
+ readonly user: SettingsFile;
127
+ readonly workspace: SettingsFile;
128
+ readonly errors: SettingsError[];
129
+
130
+ private _merged: Settings;
131
+
132
+ get merged(): Settings {
133
+ return this._merged;
134
+ }
135
+
136
+ private computeMergedSettings(): Settings {
137
+ return mergeSettings(
138
+ this.system.settings,
139
+ this.user.settings,
140
+ this.workspace.settings,
141
+ );
142
+ }
143
+
144
+ forScope(scope: SettingScope): SettingsFile {
145
+ switch (scope) {
146
+ case SettingScope.User:
147
+ return this.user;
148
+ case SettingScope.Workspace:
149
+ return this.workspace;
150
+ case SettingScope.System:
151
+ return this.system;
152
+ default:
153
+ throw new Error(`Invalid scope: ${scope}`);
154
+ }
155
+ }
156
+
157
+ setValue<K extends keyof Settings>(
158
+ scope: SettingScope,
159
+ key: K,
160
+ value: Settings[K],
161
+ ): void {
162
+ const settingsFile = this.forScope(scope);
163
+ settingsFile.settings[key] = value;
164
+ this._merged = this.computeMergedSettings();
165
+ saveSettings(settingsFile);
166
+ }
167
+ }
168
+
169
+ function resolveEnvVarsInString(value: string): string {
170
+ const envVarRegex = /\$(?:(\w+)|{([^}]+)})/g; // Find $VAR_NAME or ${VAR_NAME}
171
+ return value.replace(envVarRegex, (match, varName1, varName2) => {
172
+ const varName = varName1 || varName2;
173
+ if (process && process.env && typeof process.env[varName] === 'string') {
174
+ return process.env[varName]!;
175
+ }
176
+ return match;
177
+ });
178
+ }
179
+
180
+ function resolveEnvVarsInObject<T>(obj: T): T {
181
+ if (
182
+ obj === null ||
183
+ obj === undefined ||
184
+ typeof obj === 'boolean' ||
185
+ typeof obj === 'number'
186
+ ) {
187
+ return obj;
188
+ }
189
+
190
+ if (typeof obj === 'string') {
191
+ return resolveEnvVarsInString(obj) as unknown as T;
192
+ }
193
+
194
+ if (Array.isArray(obj)) {
195
+ return obj.map((item) => resolveEnvVarsInObject(item)) as unknown as T;
196
+ }
197
+
198
+ if (typeof obj === 'object') {
199
+ const newObj = { ...obj } as T;
200
+ for (const key in newObj) {
201
+ if (Object.prototype.hasOwnProperty.call(newObj, key)) {
202
+ newObj[key] = resolveEnvVarsInObject(newObj[key]);
203
+ }
204
+ }
205
+ return newObj;
206
+ }
207
+
208
+ return obj;
209
+ }
210
+
211
+ function findEnvFile(startDir: string): string | null {
212
+ let currentDir = path.resolve(startDir);
213
+ while (true) {
214
+ // prefer gemini-specific .env under GEMINI_DIR
215
+ const geminiEnvPath = path.join(currentDir, GEMINI_DIR, '.env');
216
+ if (fs.existsSync(geminiEnvPath)) {
217
+ return geminiEnvPath;
218
+ }
219
+ const envPath = path.join(currentDir, '.env');
220
+ if (fs.existsSync(envPath)) {
221
+ return envPath;
222
+ }
223
+ const parentDir = path.dirname(currentDir);
224
+ if (parentDir === currentDir || !parentDir) {
225
+ // check .env under home as fallback, again preferring gemini-specific .env
226
+ const homeGeminiEnvPath = path.join(homedir(), GEMINI_DIR, '.env');
227
+ if (fs.existsSync(homeGeminiEnvPath)) {
228
+ return homeGeminiEnvPath;
229
+ }
230
+ const homeEnvPath = path.join(homedir(), '.env');
231
+ if (fs.existsSync(homeEnvPath)) {
232
+ return homeEnvPath;
233
+ }
234
+ return null;
235
+ }
236
+ currentDir = parentDir;
237
+ }
238
+ }
239
+
240
+ export function setUpCloudShellEnvironment(envFilePath: string | null): void {
241
+ // Special handling for GOOGLE_CLOUD_PROJECT in Cloud Shell:
242
+ // Because GOOGLE_CLOUD_PROJECT in Cloud Shell tracks the project
243
+ // set by the user using "gcloud config set project" we do not want to
244
+ // use its value. So, unless the user overrides GOOGLE_CLOUD_PROJECT in
245
+ // one of the .env files, we set the Cloud Shell-specific default here.
246
+ if (envFilePath && fs.existsSync(envFilePath)) {
247
+ const envFileContent = fs.readFileSync(envFilePath);
248
+ const parsedEnv = dotenv.parse(envFileContent);
249
+ if (parsedEnv['GOOGLE_CLOUD_PROJECT']) {
250
+ // .env file takes precedence in Cloud Shell
251
+ process.env['GOOGLE_CLOUD_PROJECT'] = parsedEnv['GOOGLE_CLOUD_PROJECT'];
252
+ } else {
253
+ // If not in .env, set to default and override global
254
+ process.env['GOOGLE_CLOUD_PROJECT'] = 'cloudshell-gca';
255
+ }
256
+ } else {
257
+ // If no .env file, set to default and override global
258
+ process.env['GOOGLE_CLOUD_PROJECT'] = 'cloudshell-gca';
259
+ }
260
+ }
261
+
262
+ export function loadEnvironment(settings?: Settings): void {
263
+ const envFilePath = findEnvFile(process.cwd());
264
+
265
+ // Cloud Shell environment variable handling
266
+ if (process.env['CLOUD_SHELL'] === 'true') {
267
+ setUpCloudShellEnvironment(envFilePath);
268
+ }
269
+
270
+ // If no settings provided, try to load workspace settings for exclusions
271
+ let resolvedSettings = settings;
272
+ if (!resolvedSettings) {
273
+ const workspaceSettingsPath = getWorkspaceSettingsPath(process.cwd());
274
+ try {
275
+ if (fs.existsSync(workspaceSettingsPath)) {
276
+ const workspaceContent = fs.readFileSync(
277
+ workspaceSettingsPath,
278
+ 'utf-8',
279
+ );
280
+ const parsedWorkspaceSettings = JSON.parse(
281
+ stripJsonComments(workspaceContent),
282
+ ) as Settings;
283
+ resolvedSettings = resolveEnvVarsInObject(parsedWorkspaceSettings);
284
+ }
285
+ } catch (_e) {
286
+ // Ignore errors loading workspace settings
287
+ }
288
+ }
289
+
290
+ if (envFilePath) {
291
+ // Manually parse and load environment variables to handle exclusions correctly.
292
+ // This avoids modifying environment variables that were already set from the shell.
293
+ try {
294
+ const envFileContent = fs.readFileSync(envFilePath, 'utf-8');
295
+ const parsedEnv = dotenv.parse(envFileContent);
296
+
297
+ const excludedVars =
298
+ resolvedSettings?.excludedProjectEnvVars || DEFAULT_EXCLUDED_ENV_VARS;
299
+ const isProjectEnvFile = !envFilePath.includes(GEMINI_DIR);
300
+
301
+ for (const key in parsedEnv) {
302
+ if (Object.hasOwn(parsedEnv, key)) {
303
+ // If it's a project .env file, skip loading excluded variables.
304
+ if (isProjectEnvFile && excludedVars.includes(key)) {
305
+ continue;
306
+ }
307
+
308
+ // Load variable only if it's not already set in the environment.
309
+ if (!Object.hasOwn(process.env, key)) {
310
+ process.env[key] = parsedEnv[key];
311
+ }
312
+ }
313
+ }
314
+ } catch (_e) {
315
+ // Errors are ignored to match the behavior of `dotenv.config({ quiet: true })`.
316
+ }
317
+ }
318
+ }
319
+
320
+ /**
321
+ * Loads settings from user and workspace directories.
322
+ * Project settings override user settings.
323
+ */
324
+ export function loadSettings(workspaceDir: string): LoadedSettings {
325
+ let systemSettings: Settings = {};
326
+ let userSettings: Settings = {};
327
+ let workspaceSettings: Settings = {};
328
+ const settingsErrors: SettingsError[] = [];
329
+ const systemSettingsPath = getSystemSettingsPath();
330
+
331
+ // Resolve paths to their canonical representation to handle symlinks
332
+ const resolvedWorkspaceDir = path.resolve(workspaceDir);
333
+ const resolvedHomeDir = path.resolve(homedir());
334
+
335
+ let realWorkspaceDir = resolvedWorkspaceDir;
336
+ try {
337
+ // fs.realpathSync gets the "true" path, resolving any symlinks
338
+ realWorkspaceDir = fs.realpathSync(resolvedWorkspaceDir);
339
+ } catch (_e) {
340
+ // This is okay. The path might not exist yet, and that's a valid state.
341
+ }
342
+
343
+ // We expect homedir to always exist and be resolvable.
344
+ const realHomeDir = fs.realpathSync(resolvedHomeDir);
345
+
346
+ const workspaceSettingsPath = getWorkspaceSettingsPath(workspaceDir);
347
+
348
+ // Load system settings
349
+ try {
350
+ if (fs.existsSync(systemSettingsPath)) {
351
+ const systemContent = fs.readFileSync(systemSettingsPath, 'utf-8');
352
+ systemSettings = JSON.parse(stripJsonComments(systemContent)) as Settings;
353
+ }
354
+ } catch (error: unknown) {
355
+ settingsErrors.push({
356
+ message: getErrorMessage(error),
357
+ path: systemSettingsPath,
358
+ });
359
+ }
360
+
361
+ // Load user settings
362
+ try {
363
+ if (fs.existsSync(USER_SETTINGS_PATH)) {
364
+ const userContent = fs.readFileSync(USER_SETTINGS_PATH, 'utf-8');
365
+ userSettings = JSON.parse(stripJsonComments(userContent)) as Settings;
366
+ // Support legacy theme names
367
+ if (userSettings.theme && userSettings.theme === 'VS') {
368
+ userSettings.theme = DefaultLight.name;
369
+ } else if (userSettings.theme && userSettings.theme === 'VS2015') {
370
+ userSettings.theme = DefaultDark.name;
371
+ }
372
+ }
373
+ } catch (error: unknown) {
374
+ settingsErrors.push({
375
+ message: getErrorMessage(error),
376
+ path: USER_SETTINGS_PATH,
377
+ });
378
+ }
379
+
380
+ if (realWorkspaceDir !== realHomeDir) {
381
+ // Load workspace settings
382
+ try {
383
+ if (fs.existsSync(workspaceSettingsPath)) {
384
+ const projectContent = fs.readFileSync(workspaceSettingsPath, 'utf-8');
385
+ workspaceSettings = JSON.parse(
386
+ stripJsonComments(projectContent),
387
+ ) as Settings;
388
+ if (workspaceSettings.theme && workspaceSettings.theme === 'VS') {
389
+ workspaceSettings.theme = DefaultLight.name;
390
+ } else if (
391
+ workspaceSettings.theme &&
392
+ workspaceSettings.theme === 'VS2015'
393
+ ) {
394
+ workspaceSettings.theme = DefaultDark.name;
395
+ }
396
+ }
397
+ } catch (error: unknown) {
398
+ settingsErrors.push({
399
+ message: getErrorMessage(error),
400
+ path: workspaceSettingsPath,
401
+ });
402
+ }
403
+ }
404
+
405
+ // Create a temporary merged settings object to pass to loadEnvironment.
406
+ const tempMergedSettings = mergeSettings(
407
+ systemSettings,
408
+ userSettings,
409
+ workspaceSettings,
410
+ );
411
+
412
+ // loadEnviroment depends on settings so we have to create a temp version of
413
+ // the settings to avoid a cycle
414
+ loadEnvironment(tempMergedSettings);
415
+
416
+ // Now that the environment is loaded, resolve variables in the settings.
417
+ systemSettings = resolveEnvVarsInObject(systemSettings);
418
+ userSettings = resolveEnvVarsInObject(userSettings);
419
+ workspaceSettings = resolveEnvVarsInObject(workspaceSettings);
420
+
421
+ // Create LoadedSettings first
422
+ const loadedSettings = new LoadedSettings(
423
+ {
424
+ path: systemSettingsPath,
425
+ settings: systemSettings,
426
+ },
427
+ {
428
+ path: USER_SETTINGS_PATH,
429
+ settings: userSettings,
430
+ },
431
+ {
432
+ path: workspaceSettingsPath,
433
+ settings: workspaceSettings,
434
+ },
435
+ settingsErrors,
436
+ );
437
+
438
+ // Validate chatCompression settings
439
+ const chatCompression = loadedSettings.merged.chatCompression;
440
+ const threshold = chatCompression?.contextPercentageThreshold;
441
+ if (
442
+ threshold != null &&
443
+ (typeof threshold !== 'number' || threshold < 0 || threshold > 1)
444
+ ) {
445
+ console.warn(
446
+ `Invalid value for chatCompression.contextPercentageThreshold: "${threshold}". Please use a value between 0 and 1. Using default compression settings.`,
447
+ );
448
+ delete loadedSettings.merged.chatCompression;
449
+ }
450
+
451
+ // Restore authentication settings to process.env for immediate use
452
+ // This ensures that persisted settings are available as environment variables
453
+ const merged = loadedSettings.merged;
454
+
455
+ // Restore OpenAI settings
456
+ if (merged.openaiApiKey && !process.env['OPENAI_API_KEY']) {
457
+ process.env['OPENAI_API_KEY'] = merged.openaiApiKey;
458
+ }
459
+ if (merged.openaiBaseUrl && !process.env['OPENAI_BASE_URL']) {
460
+ process.env['OPENAI_BASE_URL'] = merged.openaiBaseUrl;
461
+ }
462
+ if (merged.openaiModel && !process.env['OPENAI_MODEL']) {
463
+ process.env['OPENAI_MODEL'] = merged.openaiModel;
464
+ }
465
+
466
+ // Restore Ollama settings
467
+ if (merged.ollamaApiKey && !process.env['OLLAMA_API_KEY']) {
468
+ process.env['OLLAMA_API_KEY'] = merged.ollamaApiKey;
469
+ }
470
+ if (merged.ollamaBaseUrl && !process.env['OLLAMA_BASE_URL']) {
471
+ process.env['OLLAMA_BASE_URL'] = merged.ollamaBaseUrl;
472
+ }
473
+ if (merged.ollamaModel && !process.env['OLLAMA_MODEL']) {
474
+ process.env['OLLAMA_MODEL'] = merged.ollamaModel;
475
+ }
476
+
477
+ // Restore LM Studio settings
478
+ if (merged.lmStudioApiKey && !process.env['LM_STUDIO_API_KEY']) {
479
+ process.env['LM_STUDIO_API_KEY'] = merged.lmStudioApiKey;
480
+ }
481
+ if (merged.lmStudioBaseUrl && !process.env['LM_STUDIO_BASE_URL']) {
482
+ process.env['LM_STUDIO_BASE_URL'] = merged.lmStudioBaseUrl;
483
+ }
484
+ if (merged.lmStudioModel && !process.env['LM_STUDIO_MODEL']) {
485
+ process.env['LM_STUDIO_MODEL'] = merged.lmStudioModel;
486
+ }
487
+
488
+ // Initialize database and restore active model environment variables
489
+ // This ensures database-stored model configurations are available immediately
490
+ try {
491
+ const modelManager = getModelManager();
492
+ modelManager.initializeFromStore();
493
+ } catch (error) {
494
+ // Database initialization errors should not prevent settings loading
495
+ console.warn('Failed to initialize model database:', getErrorMessage(error));
496
+ }
497
+
498
+ return loadedSettings;
499
+ }
500
+
501
+ export function saveSettings(settingsFile: SettingsFile): void {
502
+ try {
503
+ // Ensure the directory exists
504
+ const dirPath = path.dirname(settingsFile.path);
505
+ if (!fs.existsSync(dirPath)) {
506
+ fs.mkdirSync(dirPath, { recursive: true });
507
+ }
508
+
509
+ fs.writeFileSync(
510
+ settingsFile.path,
511
+ JSON.stringify(settingsFile.settings, null, 2),
512
+ 'utf-8',
513
+ );
514
+ } catch (error) {
515
+ console.error('Error saving user settings file:', error);
516
+ }
517
+ }