fss-link 1.0.49 → 1.0.51

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (419) hide show
  1. package/dist/index.js +0 -0
  2. package/dist/package.json +2 -2
  3. package/dist/src/config/auth.js +8 -5
  4. package/dist/src/config/auth.js.map +1 -1
  5. package/dist/src/config/database.d.ts +103 -11
  6. package/dist/src/config/database.js +301 -59
  7. package/dist/src/config/database.js.map +1 -1
  8. package/dist/src/config/databaseBackup.d.ts +114 -0
  9. package/dist/src/config/databaseBackup.js +334 -0
  10. package/dist/src/config/databaseBackup.js.map +1 -0
  11. package/dist/src/config/databaseMigrations.d.ts +63 -0
  12. package/dist/src/config/databaseMigrations.js +379 -0
  13. package/dist/src/config/databaseMigrations.js.map +1 -0
  14. package/dist/src/config/databasePool.d.ts +70 -0
  15. package/dist/src/config/databasePool.js +193 -0
  16. package/dist/src/config/databasePool.js.map +1 -0
  17. package/dist/src/config/queryOptimizer.d.ts +127 -0
  18. package/dist/src/config/queryOptimizer.js +309 -0
  19. package/dist/src/config/queryOptimizer.js.map +1 -0
  20. package/dist/src/utils/sandbox.js +2 -8
  21. package/dist/src/utils/sandbox.js.map +1 -1
  22. package/dist/src/validateNonInterActiveAuth.js +3 -7
  23. package/dist/src/validateNonInterActiveAuth.js.map +1 -1
  24. package/dist/tsconfig.tsbuildinfo +1 -1
  25. package/package.json +2 -2
  26. package/dist/commands/mcp/add.test.ts +0 -122
  27. package/dist/commands/mcp/add.ts +0 -222
  28. package/dist/commands/mcp/list.test.ts +0 -154
  29. package/dist/commands/mcp/list.ts +0 -139
  30. package/dist/commands/mcp/remove.test.ts +0 -69
  31. package/dist/commands/mcp/remove.ts +0 -60
  32. package/dist/commands/mcp.test.ts +0 -55
  33. package/dist/commands/mcp.ts +0 -27
  34. package/dist/config/apiValidation.test.ts +0 -118
  35. package/dist/config/auth.test.ts +0 -79
  36. package/dist/config/auth.ts +0 -100
  37. package/dist/config/config.integration.test.ts +0 -407
  38. package/dist/config/config.test.ts +0 -1952
  39. package/dist/config/config.ts +0 -690
  40. package/dist/config/database.test.ts +0 -96
  41. package/dist/config/database.ts +0 -824
  42. package/dist/config/extension.test.ts +0 -236
  43. package/dist/config/extension.ts +0 -180
  44. package/dist/config/keyBindings.test.ts +0 -62
  45. package/dist/config/keyBindings.ts +0 -184
  46. package/dist/config/modelManager.ts +0 -326
  47. package/dist/config/providerManager.ts +0 -244
  48. package/dist/config/providerPersistence.test.ts +0 -377
  49. package/dist/config/providerPersistence.ts +0 -105
  50. package/dist/config/sandboxConfig.ts +0 -107
  51. package/dist/config/settings.test.ts +0 -1424
  52. package/dist/config/settings.ts +0 -517
  53. package/dist/config/settingsSchema.test.ts +0 -252
  54. package/dist/config/settingsSchema.ts +0 -728
  55. package/dist/config/trustedFolders.test.ts +0 -208
  56. package/dist/config/trustedFolders.ts +0 -167
  57. package/dist/gemini.test.tsx +0 -252
  58. package/dist/gemini.tsx +0 -357
  59. package/dist/generated/git-commit.ts +0 -10
  60. package/dist/index.ts +0 -21
  61. package/dist/nonInteractiveCli.test.ts +0 -276
  62. package/dist/nonInteractiveCli.ts +0 -143
  63. package/dist/patches/is-in-ci.ts +0 -17
  64. package/dist/services/BuiltinCommandLoader.test.ts +0 -127
  65. package/dist/services/BuiltinCommandLoader.ts +0 -95
  66. package/dist/services/CommandService.test.ts +0 -352
  67. package/dist/services/CommandService.ts +0 -103
  68. package/dist/services/FileCommandLoader.test.ts +0 -1002
  69. package/dist/services/FileCommandLoader.ts +0 -289
  70. package/dist/services/McpPromptLoader.ts +0 -231
  71. package/dist/services/SearchEngineConfigProvider.ts +0 -100
  72. package/dist/services/prompt-processors/argumentProcessor.test.ts +0 -41
  73. package/dist/services/prompt-processors/argumentProcessor.ts +0 -23
  74. package/dist/services/prompt-processors/shellProcessor.test.ts +0 -709
  75. package/dist/services/prompt-processors/shellProcessor.ts +0 -248
  76. package/dist/services/prompt-processors/types.ts +0 -44
  77. package/dist/services/types.ts +0 -24
  78. package/dist/src/config/apiValidation.test.d.ts +0 -6
  79. package/dist/src/config/apiValidation.test.js +0 -99
  80. package/dist/src/config/apiValidation.test.js.map +0 -1
  81. package/dist/src/config/database.test.d.ts +0 -6
  82. package/dist/src/config/database.test.js +0 -80
  83. package/dist/src/config/database.test.js.map +0 -1
  84. package/dist/src/config/providerManager.d.ts +0 -74
  85. package/dist/src/config/providerManager.js +0 -203
  86. package/dist/src/config/providerManager.js.map +0 -1
  87. package/dist/src/config/providerPersistence.test.d.ts +0 -6
  88. package/dist/src/config/providerPersistence.test.js +0 -283
  89. package/dist/src/config/providerPersistence.test.js.map +0 -1
  90. package/dist/src/ui/components/GeminiKeyDialog.d.ts +0 -11
  91. package/dist/src/ui/components/GeminiKeyDialog.js +0 -156
  92. package/dist/src/ui/components/GeminiKeyDialog.js.map +0 -1
  93. package/dist/src/ui/components/OpenAIEndpointDialog.d.ts +0 -19
  94. package/dist/src/ui/components/OpenAIEndpointDialog.js +0 -163
  95. package/dist/src/ui/components/OpenAIEndpointDialog.js.map +0 -1
  96. package/dist/test-setup.ts +0 -12
  97. package/dist/test-utils/customMatchers.ts +0 -65
  98. package/dist/test-utils/mockCommandContext.test.ts +0 -62
  99. package/dist/test-utils/mockCommandContext.ts +0 -105
  100. package/dist/test-utils/render.tsx +0 -18
  101. package/dist/ui/App.test.tsx +0 -2181
  102. package/dist/ui/App.tsx +0 -1344
  103. package/dist/ui/IdeIntegrationNudge.tsx +0 -98
  104. package/dist/ui/__snapshots__/App.test.tsx.snap +0 -124
  105. package/dist/ui/colors.ts +0 -56
  106. package/dist/ui/commands/aboutCommand.test.ts +0 -153
  107. package/dist/ui/commands/aboutCommand.ts +0 -49
  108. package/dist/ui/commands/authCommand.test.ts +0 -36
  109. package/dist/ui/commands/authCommand.ts +0 -17
  110. package/dist/ui/commands/bugCommand.test.ts +0 -114
  111. package/dist/ui/commands/bugCommand.ts +0 -92
  112. package/dist/ui/commands/chatCommand.test.ts +0 -414
  113. package/dist/ui/commands/chatCommand.ts +0 -280
  114. package/dist/ui/commands/clearCommand.test.ts +0 -100
  115. package/dist/ui/commands/clearCommand.ts +0 -29
  116. package/dist/ui/commands/compressCommand.test.ts +0 -129
  117. package/dist/ui/commands/compressCommand.ts +0 -78
  118. package/dist/ui/commands/contextCommand.ts +0 -132
  119. package/dist/ui/commands/copyCommand.test.ts +0 -296
  120. package/dist/ui/commands/copyCommand.ts +0 -67
  121. package/dist/ui/commands/corgiCommand.test.ts +0 -34
  122. package/dist/ui/commands/corgiCommand.ts +0 -16
  123. package/dist/ui/commands/directoryCommand.test.tsx +0 -185
  124. package/dist/ui/commands/directoryCommand.tsx +0 -179
  125. package/dist/ui/commands/docsCommand.test.ts +0 -99
  126. package/dist/ui/commands/docsCommand.ts +0 -42
  127. package/dist/ui/commands/editorCommand.test.ts +0 -30
  128. package/dist/ui/commands/editorCommand.ts +0 -21
  129. package/dist/ui/commands/extensionsCommand.test.ts +0 -67
  130. package/dist/ui/commands/extensionsCommand.ts +0 -46
  131. package/dist/ui/commands/helpCommand.test.ts +0 -52
  132. package/dist/ui/commands/helpCommand.ts +0 -23
  133. package/dist/ui/commands/ideCommand.test.ts +0 -255
  134. package/dist/ui/commands/ideCommand.ts +0 -283
  135. package/dist/ui/commands/initCommand.test.ts +0 -127
  136. package/dist/ui/commands/initCommand.ts +0 -117
  137. package/dist/ui/commands/mcpCommand.test.ts +0 -1057
  138. package/dist/ui/commands/mcpCommand.ts +0 -531
  139. package/dist/ui/commands/memoryCommand.test.ts +0 -344
  140. package/dist/ui/commands/memoryCommand.ts +0 -305
  141. package/dist/ui/commands/privacyCommand.test.ts +0 -38
  142. package/dist/ui/commands/privacyCommand.ts +0 -17
  143. package/dist/ui/commands/quitCommand.test.ts +0 -55
  144. package/dist/ui/commands/quitCommand.ts +0 -36
  145. package/dist/ui/commands/restoreCommand.test.ts +0 -250
  146. package/dist/ui/commands/restoreCommand.ts +0 -157
  147. package/dist/ui/commands/searchEngineSetupCommand.ts +0 -18
  148. package/dist/ui/commands/settingsCommand.test.ts +0 -36
  149. package/dist/ui/commands/settingsCommand.ts +0 -17
  150. package/dist/ui/commands/setupGithubCommand.test.ts +0 -238
  151. package/dist/ui/commands/setupGithubCommand.ts +0 -212
  152. package/dist/ui/commands/speakCommand.ts +0 -175
  153. package/dist/ui/commands/statsCommand.test.ts +0 -78
  154. package/dist/ui/commands/statsCommand.ts +0 -70
  155. package/dist/ui/commands/terminalSetupCommand.test.ts +0 -85
  156. package/dist/ui/commands/terminalSetupCommand.ts +0 -45
  157. package/dist/ui/commands/themeCommand.test.ts +0 -38
  158. package/dist/ui/commands/themeCommand.ts +0 -17
  159. package/dist/ui/commands/toolsCommand.test.ts +0 -105
  160. package/dist/ui/commands/toolsCommand.ts +0 -71
  161. package/dist/ui/commands/ttsCommand.ts +0 -143
  162. package/dist/ui/commands/types.ts +0 -204
  163. package/dist/ui/commands/vimCommand.ts +0 -25
  164. package/dist/ui/commands/voiceCommand.ts +0 -125
  165. package/dist/ui/components/AboutBox.tsx +0 -133
  166. package/dist/ui/components/AsciiArt.ts +0 -54
  167. package/dist/ui/components/AuthDialog.test.tsx +0 -334
  168. package/dist/ui/components/AuthDialog.tsx +0 -289
  169. package/dist/ui/components/AuthInProgress.tsx +0 -62
  170. package/dist/ui/components/AutoAcceptIndicator.tsx +0 -47
  171. package/dist/ui/components/ConsoleSummaryDisplay.tsx +0 -35
  172. package/dist/ui/components/ContextSummaryDisplay.test.tsx +0 -85
  173. package/dist/ui/components/ContextSummaryDisplay.tsx +0 -120
  174. package/dist/ui/components/ContextUsageDisplay.tsx +0 -77
  175. package/dist/ui/components/DebugProfiler.tsx +0 -36
  176. package/dist/ui/components/DetailedMessagesDisplay.tsx +0 -82
  177. package/dist/ui/components/EditorSettingsDialog.tsx +0 -172
  178. package/dist/ui/components/FolderTrustDialog.test.tsx +0 -36
  179. package/dist/ui/components/FolderTrustDialog.tsx +0 -74
  180. package/dist/ui/components/Footer.test.tsx +0 -159
  181. package/dist/ui/components/Footer.tsx +0 -158
  182. package/dist/ui/components/GeminiKeyDialog.tsx +0 -252
  183. package/dist/ui/components/GeminiRespondingSpinner.tsx +0 -34
  184. package/dist/ui/components/Header.test.tsx +0 -44
  185. package/dist/ui/components/Header.tsx +0 -70
  186. package/dist/ui/components/Help.tsx +0 -174
  187. package/dist/ui/components/HistoryItemDisplay.test.tsx +0 -125
  188. package/dist/ui/components/HistoryItemDisplay.tsx +0 -98
  189. package/dist/ui/components/InputPrompt.test.tsx +0 -1467
  190. package/dist/ui/components/InputPrompt.tsx +0 -641
  191. package/dist/ui/components/LMStudioModelPrompt.tsx +0 -215
  192. package/dist/ui/components/LoadingIndicator.test.tsx +0 -296
  193. package/dist/ui/components/LoadingIndicator.tsx +0 -82
  194. package/dist/ui/components/MemoryUsageDisplay.tsx +0 -36
  195. package/dist/ui/components/ModelStatsDisplay.test.tsx +0 -252
  196. package/dist/ui/components/ModelStatsDisplay.tsx +0 -197
  197. package/dist/ui/components/OllamaModelPrompt.tsx +0 -206
  198. package/dist/ui/components/OpenAIEndpointDialog.tsx +0 -261
  199. package/dist/ui/components/OpenAIKeyPrompt.test.tsx +0 -64
  200. package/dist/ui/components/OpenAIKeyPrompt.tsx +0 -197
  201. package/dist/ui/components/PrepareLabel.tsx +0 -48
  202. package/dist/ui/components/SearchEngineConfigDialog.tsx +0 -280
  203. package/dist/ui/components/SessionSummaryDisplay.test.tsx +0 -75
  204. package/dist/ui/components/SessionSummaryDisplay.tsx +0 -18
  205. package/dist/ui/components/SettingsDialog.test.tsx +0 -865
  206. package/dist/ui/components/SettingsDialog.tsx +0 -753
  207. package/dist/ui/components/ShellConfirmationDialog.test.tsx +0 -53
  208. package/dist/ui/components/ShellConfirmationDialog.tsx +0 -103
  209. package/dist/ui/components/ShellModeIndicator.tsx +0 -18
  210. package/dist/ui/components/ShowMoreLines.tsx +0 -40
  211. package/dist/ui/components/StatsDisplay.test.tsx +0 -401
  212. package/dist/ui/components/StatsDisplay.tsx +0 -273
  213. package/dist/ui/components/SuggestionsDisplay.tsx +0 -102
  214. package/dist/ui/components/ThemeDialog.tsx +0 -310
  215. package/dist/ui/components/Tips.tsx +0 -45
  216. package/dist/ui/components/TodoDisplay.test.tsx +0 -97
  217. package/dist/ui/components/TodoDisplay.tsx +0 -72
  218. package/dist/ui/components/ToolStatsDisplay.test.tsx +0 -180
  219. package/dist/ui/components/ToolStatsDisplay.tsx +0 -208
  220. package/dist/ui/components/UpdateNotification.tsx +0 -23
  221. package/dist/ui/components/WelcomeBackDialog.tsx +0 -290
  222. package/dist/ui/components/__snapshots__/IDEContextDetailDisplay.test.tsx.snap +0 -24
  223. package/dist/ui/components/__snapshots__/ModelStatsDisplay.test.tsx.snap +0 -121
  224. package/dist/ui/components/__snapshots__/SessionSummaryDisplay.test.tsx.snap +0 -30
  225. package/dist/ui/components/__snapshots__/ShellConfirmationDialog.test.tsx.snap +0 -21
  226. package/dist/ui/components/__snapshots__/StatsDisplay.test.tsx.snap +0 -264
  227. package/dist/ui/components/__snapshots__/ToolStatsDisplay.test.tsx.snap +0 -91
  228. package/dist/ui/components/messages/CompressionMessage.tsx +0 -49
  229. package/dist/ui/components/messages/DiffRenderer.test.tsx +0 -365
  230. package/dist/ui/components/messages/DiffRenderer.tsx +0 -358
  231. package/dist/ui/components/messages/ErrorMessage.tsx +0 -31
  232. package/dist/ui/components/messages/GeminiMessage.tsx +0 -43
  233. package/dist/ui/components/messages/GeminiMessageContent.tsx +0 -43
  234. package/dist/ui/components/messages/InfoMessage.tsx +0 -32
  235. package/dist/ui/components/messages/ToolConfirmationMessage.test.tsx +0 -58
  236. package/dist/ui/components/messages/ToolConfirmationMessage.tsx +0 -297
  237. package/dist/ui/components/messages/ToolGroupMessage.tsx +0 -126
  238. package/dist/ui/components/messages/ToolMessage.test.tsx +0 -183
  239. package/dist/ui/components/messages/ToolMessage.tsx +0 -296
  240. package/dist/ui/components/messages/UserMessage.tsx +0 -43
  241. package/dist/ui/components/messages/UserShellMessage.tsx +0 -25
  242. package/dist/ui/components/shared/MaxSizedBox.test.tsx +0 -425
  243. package/dist/ui/components/shared/MaxSizedBox.tsx +0 -624
  244. package/dist/ui/components/shared/RadioButtonSelect.test.tsx +0 -181
  245. package/dist/ui/components/shared/RadioButtonSelect.tsx +0 -234
  246. package/dist/ui/components/shared/__snapshots__/RadioButtonSelect.test.tsx.snap +0 -47
  247. package/dist/ui/components/shared/text-buffer.test.ts +0 -1728
  248. package/dist/ui/components/shared/text-buffer.ts +0 -2227
  249. package/dist/ui/components/shared/vim-buffer-actions.test.ts +0 -1119
  250. package/dist/ui/components/shared/vim-buffer-actions.ts +0 -814
  251. package/dist/ui/constants.ts +0 -17
  252. package/dist/ui/contexts/KeypressContext.test.tsx +0 -391
  253. package/dist/ui/contexts/KeypressContext.tsx +0 -440
  254. package/dist/ui/contexts/OverflowContext.tsx +0 -87
  255. package/dist/ui/contexts/SessionContext.test.tsx +0 -132
  256. package/dist/ui/contexts/SessionContext.tsx +0 -143
  257. package/dist/ui/contexts/SettingsContext.tsx +0 -20
  258. package/dist/ui/contexts/StreamingContext.tsx +0 -22
  259. package/dist/ui/contexts/VimModeContext.tsx +0 -79
  260. package/dist/ui/editors/editorSettingsManager.ts +0 -66
  261. package/dist/ui/hooks/atCommandProcessor.test.ts +0 -1102
  262. package/dist/ui/hooks/atCommandProcessor.ts +0 -485
  263. package/dist/ui/hooks/shellCommandProcessor.test.ts +0 -481
  264. package/dist/ui/hooks/shellCommandProcessor.ts +0 -314
  265. package/dist/ui/hooks/slashCommandProcessor.test.ts +0 -1044
  266. package/dist/ui/hooks/slashCommandProcessor.ts +0 -595
  267. package/dist/ui/hooks/useAtCompletion.test.ts +0 -497
  268. package/dist/ui/hooks/useAtCompletion.ts +0 -244
  269. package/dist/ui/hooks/useAuthCommand.ts +0 -129
  270. package/dist/ui/hooks/useAutoAcceptIndicator.test.ts +0 -300
  271. package/dist/ui/hooks/useAutoAcceptIndicator.ts +0 -52
  272. package/dist/ui/hooks/useBracketedPaste.ts +0 -37
  273. package/dist/ui/hooks/useCommandCompletion.test.ts +0 -518
  274. package/dist/ui/hooks/useCommandCompletion.tsx +0 -238
  275. package/dist/ui/hooks/useCompletion.ts +0 -128
  276. package/dist/ui/hooks/useConsoleMessages.test.ts +0 -147
  277. package/dist/ui/hooks/useConsoleMessages.ts +0 -110
  278. package/dist/ui/hooks/useEditorSettings.test.ts +0 -283
  279. package/dist/ui/hooks/useEditorSettings.ts +0 -75
  280. package/dist/ui/hooks/useFocus.test.ts +0 -119
  281. package/dist/ui/hooks/useFocus.ts +0 -48
  282. package/dist/ui/hooks/useFolderTrust.test.ts +0 -159
  283. package/dist/ui/hooks/useFolderTrust.ts +0 -72
  284. package/dist/ui/hooks/useGeminiStream.test.tsx +0 -1998
  285. package/dist/ui/hooks/useGeminiStream.ts +0 -1017
  286. package/dist/ui/hooks/useGitBranchName.test.ts +0 -280
  287. package/dist/ui/hooks/useGitBranchName.ts +0 -79
  288. package/dist/ui/hooks/useHistoryManager.test.ts +0 -202
  289. package/dist/ui/hooks/useHistoryManager.ts +0 -111
  290. package/dist/ui/hooks/useInputHistory.test.ts +0 -261
  291. package/dist/ui/hooks/useInputHistory.ts +0 -111
  292. package/dist/ui/hooks/useKeypress.test.ts +0 -280
  293. package/dist/ui/hooks/useKeypress.ts +0 -39
  294. package/dist/ui/hooks/useKittyKeyboardProtocol.ts +0 -31
  295. package/dist/ui/hooks/useLoadingIndicator.test.ts +0 -139
  296. package/dist/ui/hooks/useLoadingIndicator.ts +0 -57
  297. package/dist/ui/hooks/useLogger.ts +0 -32
  298. package/dist/ui/hooks/useMessageQueue.test.ts +0 -226
  299. package/dist/ui/hooks/useMessageQueue.ts +0 -69
  300. package/dist/ui/hooks/usePhraseCycler.test.ts +0 -145
  301. package/dist/ui/hooks/usePhraseCycler.ts +0 -198
  302. package/dist/ui/hooks/usePrivacySettings.test.ts +0 -242
  303. package/dist/ui/hooks/usePrivacySettings.ts +0 -150
  304. package/dist/ui/hooks/useReactToolScheduler.ts +0 -309
  305. package/dist/ui/hooks/useRefreshMemoryCommand.ts +0 -7
  306. package/dist/ui/hooks/useReverseSearchCompletion.test.tsx +0 -260
  307. package/dist/ui/hooks/useReverseSearchCompletion.tsx +0 -95
  308. package/dist/ui/hooks/useSettingsCommand.ts +0 -25
  309. package/dist/ui/hooks/useShellHistory.test.ts +0 -219
  310. package/dist/ui/hooks/useShellHistory.ts +0 -133
  311. package/dist/ui/hooks/useShowMemoryCommand.ts +0 -75
  312. package/dist/ui/hooks/useSlashCompletion.test.ts +0 -434
  313. package/dist/ui/hooks/useSlashCompletion.ts +0 -187
  314. package/dist/ui/hooks/useStateAndRef.ts +0 -36
  315. package/dist/ui/hooks/useTerminalSize.ts +0 -32
  316. package/dist/ui/hooks/useThemeCommand.ts +0 -110
  317. package/dist/ui/hooks/useTimer.test.ts +0 -120
  318. package/dist/ui/hooks/useTimer.ts +0 -65
  319. package/dist/ui/hooks/useToolScheduler.test.ts +0 -1123
  320. package/dist/ui/hooks/useWelcomeBack.ts +0 -253
  321. package/dist/ui/hooks/vim.test.ts +0 -1691
  322. package/dist/ui/hooks/vim.ts +0 -784
  323. package/dist/ui/keyMatchers.test.ts +0 -337
  324. package/dist/ui/keyMatchers.ts +0 -105
  325. package/dist/ui/privacy/CloudFreePrivacyNotice.tsx +0 -117
  326. package/dist/ui/privacy/CloudPaidPrivacyNotice.tsx +0 -59
  327. package/dist/ui/privacy/GeminiPrivacyNotice.tsx +0 -62
  328. package/dist/ui/privacy/PrivacyNotice.tsx +0 -42
  329. package/dist/ui/semantic-colors.ts +0 -26
  330. package/dist/ui/themes/ansi-light.ts +0 -150
  331. package/dist/ui/themes/ansi.ts +0 -159
  332. package/dist/ui/themes/atom-one-dark.ts +0 -147
  333. package/dist/ui/themes/ayu-light.ts +0 -139
  334. package/dist/ui/themes/ayu.ts +0 -113
  335. package/dist/ui/themes/color-utils.test.ts +0 -221
  336. package/dist/ui/themes/color-utils.ts +0 -231
  337. package/dist/ui/themes/default-light.ts +0 -108
  338. package/dist/ui/themes/default.ts +0 -151
  339. package/dist/ui/themes/dracula.ts +0 -124
  340. package/dist/ui/themes/fss-code-dark.ts +0 -156
  341. package/dist/ui/themes/fss-dark.ts +0 -113
  342. package/dist/ui/themes/fss-light.ts +0 -139
  343. package/dist/ui/themes/github-dark.ts +0 -147
  344. package/dist/ui/themes/github-light.ts +0 -149
  345. package/dist/ui/themes/googlecode.ts +0 -146
  346. package/dist/ui/themes/no-color.ts +0 -125
  347. package/dist/ui/themes/qwen-dark.ts +0 -118
  348. package/dist/ui/themes/qwen-light.ts +0 -144
  349. package/dist/ui/themes/semantic-tokens.ts +0 -127
  350. package/dist/ui/themes/shades-of-purple.ts +0 -352
  351. package/dist/ui/themes/theme-manager.test.ts +0 -99
  352. package/dist/ui/themes/theme-manager.ts +0 -257
  353. package/dist/ui/themes/theme.test.ts +0 -97
  354. package/dist/ui/themes/theme.ts +0 -451
  355. package/dist/ui/themes/xcode.ts +0 -154
  356. package/dist/ui/types.ts +0 -255
  357. package/dist/ui/utils/CodeColorizer.tsx +0 -217
  358. package/dist/ui/utils/ConsolePatcher.ts +0 -71
  359. package/dist/ui/utils/InlineMarkdownRenderer.tsx +0 -173
  360. package/dist/ui/utils/MarkdownDisplay.test.tsx +0 -244
  361. package/dist/ui/utils/MarkdownDisplay.tsx +0 -415
  362. package/dist/ui/utils/TableRenderer.tsx +0 -159
  363. package/dist/ui/utils/__snapshots__/MarkdownDisplay.test.tsx.snap +0 -93
  364. package/dist/ui/utils/clipboardUtils.test.ts +0 -76
  365. package/dist/ui/utils/clipboardUtils.ts +0 -149
  366. package/dist/ui/utils/commandUtils.test.ts +0 -384
  367. package/dist/ui/utils/commandUtils.ts +0 -106
  368. package/dist/ui/utils/computeStats.test.ts +0 -292
  369. package/dist/ui/utils/computeStats.ts +0 -86
  370. package/dist/ui/utils/displayUtils.test.ts +0 -58
  371. package/dist/ui/utils/displayUtils.ts +0 -32
  372. package/dist/ui/utils/formatters.test.ts +0 -72
  373. package/dist/ui/utils/formatters.ts +0 -63
  374. package/dist/ui/utils/isNarrowWidth.ts +0 -9
  375. package/dist/ui/utils/kittyProtocolDetector.ts +0 -105
  376. package/dist/ui/utils/markdownUtilities.test.ts +0 -50
  377. package/dist/ui/utils/markdownUtilities.ts +0 -125
  378. package/dist/ui/utils/platformConstants.ts +0 -52
  379. package/dist/ui/utils/terminalSetup.ts +0 -342
  380. package/dist/ui/utils/textUtils.ts +0 -40
  381. package/dist/ui/utils/updateCheck.test.ts +0 -163
  382. package/dist/ui/utils/updateCheck.ts +0 -100
  383. package/dist/utils/checks.ts +0 -28
  384. package/dist/utils/cleanup.test.ts +0 -68
  385. package/dist/utils/cleanup.ts +0 -36
  386. package/dist/utils/dialogScopeUtils.ts +0 -64
  387. package/dist/utils/events.ts +0 -14
  388. package/dist/utils/gitUtils.test.ts +0 -149
  389. package/dist/utils/gitUtils.ts +0 -116
  390. package/dist/utils/handleAutoUpdate.test.ts +0 -272
  391. package/dist/utils/handleAutoUpdate.ts +0 -145
  392. package/dist/utils/installationInfo.test.ts +0 -315
  393. package/dist/utils/installationInfo.ts +0 -176
  394. package/dist/utils/package.ts +0 -38
  395. package/dist/utils/readStdin.ts +0 -51
  396. package/dist/utils/resolvePath.ts +0 -21
  397. package/dist/utils/sandbox-macos-permissive-closed.sb +0 -32
  398. package/dist/utils/sandbox-macos-permissive-open.sb +0 -25
  399. package/dist/utils/sandbox-macos-permissive-proxied.sb +0 -37
  400. package/dist/utils/sandbox-macos-restrictive-closed.sb +0 -93
  401. package/dist/utils/sandbox-macos-restrictive-open.sb +0 -96
  402. package/dist/utils/sandbox-macos-restrictive-proxied.sb +0 -98
  403. package/dist/utils/sandbox.ts +0 -962
  404. package/dist/utils/settingsUtils.test.ts +0 -797
  405. package/dist/utils/settingsUtils.ts +0 -489
  406. package/dist/utils/spawnWrapper.ts +0 -9
  407. package/dist/utils/startupWarnings.test.ts +0 -83
  408. package/dist/utils/startupWarnings.ts +0 -40
  409. package/dist/utils/updateEventEmitter.ts +0 -13
  410. package/dist/utils/userStartupWarnings.test.ts +0 -87
  411. package/dist/utils/userStartupWarnings.ts +0 -69
  412. package/dist/utils/version.ts +0 -12
  413. package/dist/validateNonInterActiveAuth.test.ts +0 -260
  414. package/dist/validateNonInterActiveAuth.ts +0 -51
  415. package/dist/vitest.config.ts +0 -37
  416. package/dist/zed-integration/acp.ts +0 -366
  417. package/dist/zed-integration/fileSystemService.ts +0 -47
  418. package/dist/zed-integration/schema.ts +0 -466
  419. package/dist/zed-integration/zedIntegration.ts +0 -944
@@ -1,1424 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
-
7
- /// <reference types="vitest/globals" />
8
-
9
- // Mock 'os' first.
10
- import * as osActual from 'os'; // Import for type info for the mock factory
11
- vi.mock('os', async (importOriginal) => {
12
- const actualOs = await importOriginal<typeof osActual>();
13
- return {
14
- ...actualOs,
15
- homedir: vi.fn(() => '/mock/home/user'),
16
- platform: vi.fn(() => 'linux'),
17
- };
18
- });
19
-
20
- // Mock './settings.js' to ensure it uses the mocked 'os.homedir()' for its internal constants.
21
- vi.mock('./settings.js', async (importActual) => {
22
- const originalModule = await importActual<typeof import('./settings.js')>();
23
- return {
24
- __esModule: true, // Ensure correct module shape
25
- ...originalModule, // Re-export all original members
26
- // We are relying on originalModule's USER_SETTINGS_PATH being constructed with mocked os.homedir()
27
- };
28
- });
29
-
30
- // NOW import everything else, including the (now effectively re-exported) settings.js
31
- import * as pathActual from 'path'; // Restored for MOCK_WORKSPACE_SETTINGS_PATH
32
- import {
33
- describe,
34
- it,
35
- expect,
36
- vi,
37
- beforeEach,
38
- afterEach,
39
- type Mocked,
40
- type Mock,
41
- } from 'vitest';
42
- import * as fs from 'fs'; // fs will be mocked separately
43
- import stripJsonComments from 'strip-json-comments'; // Will be mocked separately
44
-
45
- // These imports will get the versions from the vi.mock('./settings.js', ...) factory.
46
- import {
47
- loadSettings,
48
- USER_SETTINGS_PATH, // This IS the mocked path.
49
- getSystemSettingsPath,
50
- SETTINGS_DIRECTORY_NAME, // This is from the original module, but used by the mock.
51
- SettingScope,
52
- } from './settings.js';
53
-
54
- const MOCK_WORKSPACE_DIR = '/mock/workspace';
55
- // Use the (mocked) SETTINGS_DIRECTORY_NAME for consistency
56
- const MOCK_WORKSPACE_SETTINGS_PATH = pathActual.join(
57
- MOCK_WORKSPACE_DIR,
58
- SETTINGS_DIRECTORY_NAME,
59
- 'settings.json',
60
- );
61
-
62
- vi.mock('fs', async (importOriginal) => {
63
- // Get all the functions from the real 'fs' module
64
- const actualFs = await importOriginal<typeof fs>();
65
-
66
- return {
67
- ...actualFs, // Keep all the real functions
68
- // Now, just override the ones we need for the test
69
- existsSync: vi.fn(),
70
- readFileSync: vi.fn(),
71
- writeFileSync: vi.fn(),
72
- mkdirSync: vi.fn(),
73
- realpathSync: (p: string) => p,
74
- };
75
- });
76
-
77
- vi.mock('strip-json-comments', () => ({
78
- default: vi.fn((content) => content),
79
- }));
80
-
81
- describe('Settings Loading and Merging', () => {
82
- let mockFsExistsSync: Mocked<typeof fs.existsSync>;
83
- let mockStripJsonComments: Mocked<typeof stripJsonComments>;
84
- let mockFsMkdirSync: Mocked<typeof fs.mkdirSync>;
85
-
86
- beforeEach(() => {
87
- vi.resetAllMocks();
88
-
89
- mockFsExistsSync = vi.mocked(fs.existsSync);
90
- mockFsMkdirSync = vi.mocked(fs.mkdirSync);
91
- mockStripJsonComments = vi.mocked(stripJsonComments);
92
-
93
- vi.mocked(osActual.homedir).mockReturnValue('/mock/home/user');
94
- (mockStripJsonComments as unknown as Mock).mockImplementation(
95
- (jsonString: string) => jsonString,
96
- );
97
- (mockFsExistsSync as Mock).mockReturnValue(false);
98
- (fs.readFileSync as Mock).mockReturnValue('{}'); // Return valid empty JSON
99
- (mockFsMkdirSync as Mock).mockImplementation(() => undefined);
100
- });
101
-
102
- afterEach(() => {
103
- vi.restoreAllMocks();
104
- });
105
-
106
- describe('loadSettings', () => {
107
- it('should load empty settings if no files exist', () => {
108
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
109
- expect(settings.system.settings).toEqual({});
110
- expect(settings.user.settings).toEqual({});
111
- expect(settings.workspace.settings).toEqual({});
112
- expect(settings.merged).toEqual({
113
- customThemes: {},
114
- mcpServers: {},
115
- includeDirectories: [],
116
- chatCompression: {},
117
- });
118
- expect(settings.errors.length).toBe(0);
119
- });
120
-
121
- it('should load system settings if only system file exists', () => {
122
- (mockFsExistsSync as Mock).mockImplementation(
123
- (p: fs.PathLike) => p === getSystemSettingsPath(),
124
- );
125
- const systemSettingsContent = {
126
- theme: 'system-default',
127
- sandbox: false,
128
- };
129
- (fs.readFileSync as Mock).mockImplementation(
130
- (p: fs.PathOrFileDescriptor) => {
131
- if (p === getSystemSettingsPath())
132
- return JSON.stringify(systemSettingsContent);
133
- return '{}';
134
- },
135
- );
136
-
137
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
138
-
139
- expect(fs.readFileSync).toHaveBeenCalledWith(
140
- getSystemSettingsPath(),
141
- 'utf-8',
142
- );
143
- expect(settings.system.settings).toEqual(systemSettingsContent);
144
- expect(settings.user.settings).toEqual({});
145
- expect(settings.workspace.settings).toEqual({});
146
- expect(settings.merged).toEqual({
147
- ...systemSettingsContent,
148
- customThemes: {},
149
- mcpServers: {},
150
- includeDirectories: [],
151
- chatCompression: {},
152
- });
153
- });
154
-
155
- it('should load user settings if only user file exists', () => {
156
- const expectedUserSettingsPath = USER_SETTINGS_PATH; // Use the path actually resolved by the (mocked) module
157
-
158
- (mockFsExistsSync as Mock).mockImplementation(
159
- (p: fs.PathLike) => p === expectedUserSettingsPath,
160
- );
161
- const userSettingsContent = {
162
- theme: 'dark',
163
- contextFileName: 'USER_CONTEXT.md',
164
- };
165
- (fs.readFileSync as Mock).mockImplementation(
166
- (p: fs.PathOrFileDescriptor) => {
167
- if (p === expectedUserSettingsPath)
168
- return JSON.stringify(userSettingsContent);
169
- return '{}';
170
- },
171
- );
172
-
173
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
174
-
175
- expect(fs.readFileSync).toHaveBeenCalledWith(
176
- expectedUserSettingsPath,
177
- 'utf-8',
178
- );
179
- expect(settings.user.settings).toEqual(userSettingsContent);
180
- expect(settings.workspace.settings).toEqual({});
181
- expect(settings.merged).toEqual({
182
- ...userSettingsContent,
183
- customThemes: {},
184
- mcpServers: {},
185
- includeDirectories: [],
186
- chatCompression: {},
187
- });
188
- });
189
-
190
- it('should load workspace settings if only workspace file exists', () => {
191
- (mockFsExistsSync as Mock).mockImplementation(
192
- (p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
193
- );
194
- const workspaceSettingsContent = {
195
- sandbox: true,
196
- contextFileName: 'WORKSPACE_CONTEXT.md',
197
- };
198
- (fs.readFileSync as Mock).mockImplementation(
199
- (p: fs.PathOrFileDescriptor) => {
200
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
201
- return JSON.stringify(workspaceSettingsContent);
202
- return '';
203
- },
204
- );
205
-
206
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
207
-
208
- expect(fs.readFileSync).toHaveBeenCalledWith(
209
- MOCK_WORKSPACE_SETTINGS_PATH,
210
- 'utf-8',
211
- );
212
- expect(settings.user.settings).toEqual({});
213
- expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
214
- expect(settings.merged).toEqual({
215
- ...workspaceSettingsContent,
216
- customThemes: {},
217
- mcpServers: {},
218
- includeDirectories: [],
219
- chatCompression: {},
220
- });
221
- });
222
-
223
- it('should merge user and workspace settings, with workspace taking precedence', () => {
224
- (mockFsExistsSync as Mock).mockReturnValue(true);
225
- const userSettingsContent = {
226
- theme: 'dark',
227
- sandbox: false,
228
- contextFileName: 'USER_CONTEXT.md',
229
- };
230
- const workspaceSettingsContent = {
231
- sandbox: true,
232
- coreTools: ['tool1'],
233
- contextFileName: 'WORKSPACE_CONTEXT.md',
234
- };
235
-
236
- (fs.readFileSync as Mock).mockImplementation(
237
- (p: fs.PathOrFileDescriptor) => {
238
- if (p === USER_SETTINGS_PATH)
239
- return JSON.stringify(userSettingsContent);
240
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
241
- return JSON.stringify(workspaceSettingsContent);
242
- return '';
243
- },
244
- );
245
-
246
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
247
-
248
- expect(settings.user.settings).toEqual(userSettingsContent);
249
- expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
250
- expect(settings.merged).toEqual({
251
- theme: 'dark',
252
- sandbox: true,
253
- coreTools: ['tool1'],
254
- contextFileName: 'WORKSPACE_CONTEXT.md',
255
- customThemes: {},
256
- mcpServers: {},
257
- includeDirectories: [],
258
- chatCompression: {},
259
- });
260
- });
261
-
262
- it('should merge system, user and workspace settings, with system taking precedence over workspace, and workspace over user', () => {
263
- (mockFsExistsSync as Mock).mockReturnValue(true);
264
- const systemSettingsContent = {
265
- theme: 'system-theme',
266
- sandbox: false,
267
- allowMCPServers: ['server1', 'server2'],
268
- telemetry: { enabled: false },
269
- };
270
- const userSettingsContent = {
271
- theme: 'dark',
272
- sandbox: true,
273
- contextFileName: 'USER_CONTEXT.md',
274
- };
275
- const workspaceSettingsContent = {
276
- sandbox: false,
277
- coreTools: ['tool1'],
278
- contextFileName: 'WORKSPACE_CONTEXT.md',
279
- allowMCPServers: ['server1', 'server2', 'server3'],
280
- };
281
-
282
- (fs.readFileSync as Mock).mockImplementation(
283
- (p: fs.PathOrFileDescriptor) => {
284
- if (p === getSystemSettingsPath())
285
- return JSON.stringify(systemSettingsContent);
286
- if (p === USER_SETTINGS_PATH)
287
- return JSON.stringify(userSettingsContent);
288
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
289
- return JSON.stringify(workspaceSettingsContent);
290
- return '';
291
- },
292
- );
293
-
294
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
295
-
296
- expect(settings.system.settings).toEqual(systemSettingsContent);
297
- expect(settings.user.settings).toEqual(userSettingsContent);
298
- expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
299
- expect(settings.merged).toEqual({
300
- theme: 'system-theme',
301
- sandbox: false,
302
- telemetry: { enabled: false },
303
- coreTools: ['tool1'],
304
- contextFileName: 'WORKSPACE_CONTEXT.md',
305
- allowMCPServers: ['server1', 'server2'],
306
- customThemes: {},
307
- mcpServers: {},
308
- includeDirectories: [],
309
- chatCompression: {},
310
- });
311
- });
312
-
313
- it('should ignore folderTrust from workspace settings', () => {
314
- (mockFsExistsSync as Mock).mockReturnValue(true);
315
- const userSettingsContent = {
316
- folderTrust: true,
317
- };
318
- const workspaceSettingsContent = {
319
- folderTrust: false, // This should be ignored
320
- };
321
- const systemSettingsContent = {
322
- // No folderTrust here
323
- };
324
-
325
- (fs.readFileSync as Mock).mockImplementation(
326
- (p: fs.PathOrFileDescriptor) => {
327
- if (p === getSystemSettingsPath())
328
- return JSON.stringify(systemSettingsContent);
329
- if (p === USER_SETTINGS_PATH)
330
- return JSON.stringify(userSettingsContent);
331
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
332
- return JSON.stringify(workspaceSettingsContent);
333
- return '{}';
334
- },
335
- );
336
-
337
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
338
- expect(settings.merged.folderTrust).toBe(true); // User setting should be used
339
- });
340
-
341
- it('should use system folderTrust over user setting', () => {
342
- (mockFsExistsSync as Mock).mockReturnValue(true);
343
- const userSettingsContent = {
344
- folderTrust: false,
345
- };
346
- const workspaceSettingsContent = {
347
- folderTrust: true, // This should be ignored
348
- };
349
- const systemSettingsContent = {
350
- folderTrust: true,
351
- };
352
-
353
- (fs.readFileSync as Mock).mockImplementation(
354
- (p: fs.PathOrFileDescriptor) => {
355
- if (p === getSystemSettingsPath())
356
- return JSON.stringify(systemSettingsContent);
357
- if (p === USER_SETTINGS_PATH)
358
- return JSON.stringify(userSettingsContent);
359
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
360
- return JSON.stringify(workspaceSettingsContent);
361
- return '{}';
362
- },
363
- );
364
-
365
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
366
- expect(settings.merged.folderTrust).toBe(true); // System setting should be used
367
- });
368
-
369
- it('should handle contextFileName correctly when only in user settings', () => {
370
- (mockFsExistsSync as Mock).mockImplementation(
371
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
372
- );
373
- const userSettingsContent = { contextFileName: 'CUSTOM.md' };
374
- (fs.readFileSync as Mock).mockImplementation(
375
- (p: fs.PathOrFileDescriptor) => {
376
- if (p === USER_SETTINGS_PATH)
377
- return JSON.stringify(userSettingsContent);
378
- return '';
379
- },
380
- );
381
-
382
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
383
- expect(settings.merged.contextFileName).toBe('CUSTOM.md');
384
- });
385
-
386
- it('should handle contextFileName correctly when only in workspace settings', () => {
387
- (mockFsExistsSync as Mock).mockImplementation(
388
- (p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
389
- );
390
- const workspaceSettingsContent = {
391
- contextFileName: 'PROJECT_SPECIFIC.md',
392
- };
393
- (fs.readFileSync as Mock).mockImplementation(
394
- (p: fs.PathOrFileDescriptor) => {
395
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
396
- return JSON.stringify(workspaceSettingsContent);
397
- return '';
398
- },
399
- );
400
-
401
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
402
- expect(settings.merged.contextFileName).toBe('PROJECT_SPECIFIC.md');
403
- });
404
-
405
- it('should handle excludedProjectEnvVars correctly when only in user settings', () => {
406
- (mockFsExistsSync as Mock).mockImplementation(
407
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
408
- );
409
- const userSettingsContent = {
410
- excludedProjectEnvVars: ['DEBUG', 'NODE_ENV', 'CUSTOM_VAR'],
411
- };
412
- (fs.readFileSync as Mock).mockImplementation(
413
- (p: fs.PathOrFileDescriptor) => {
414
- if (p === USER_SETTINGS_PATH)
415
- return JSON.stringify(userSettingsContent);
416
- return '';
417
- },
418
- );
419
-
420
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
421
- expect(settings.merged.excludedProjectEnvVars).toEqual([
422
- 'DEBUG',
423
- 'NODE_ENV',
424
- 'CUSTOM_VAR',
425
- ]);
426
- });
427
-
428
- it('should handle excludedProjectEnvVars correctly when only in workspace settings', () => {
429
- (mockFsExistsSync as Mock).mockImplementation(
430
- (p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
431
- );
432
- const workspaceSettingsContent = {
433
- excludedProjectEnvVars: ['WORKSPACE_DEBUG', 'WORKSPACE_VAR'],
434
- };
435
- (fs.readFileSync as Mock).mockImplementation(
436
- (p: fs.PathOrFileDescriptor) => {
437
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
438
- return JSON.stringify(workspaceSettingsContent);
439
- return '';
440
- },
441
- );
442
-
443
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
444
- expect(settings.merged.excludedProjectEnvVars).toEqual([
445
- 'WORKSPACE_DEBUG',
446
- 'WORKSPACE_VAR',
447
- ]);
448
- });
449
-
450
- it('should merge excludedProjectEnvVars with workspace taking precedence over user', () => {
451
- (mockFsExistsSync as Mock).mockReturnValue(true);
452
- const userSettingsContent = {
453
- excludedProjectEnvVars: ['DEBUG', 'NODE_ENV', 'USER_VAR'],
454
- };
455
- const workspaceSettingsContent = {
456
- excludedProjectEnvVars: ['WORKSPACE_DEBUG', 'WORKSPACE_VAR'],
457
- };
458
-
459
- (fs.readFileSync as Mock).mockImplementation(
460
- (p: fs.PathOrFileDescriptor) => {
461
- if (p === USER_SETTINGS_PATH)
462
- return JSON.stringify(userSettingsContent);
463
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
464
- return JSON.stringify(workspaceSettingsContent);
465
- return '';
466
- },
467
- );
468
-
469
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
470
- expect(settings.user.settings.excludedProjectEnvVars).toEqual([
471
- 'DEBUG',
472
- 'NODE_ENV',
473
- 'USER_VAR',
474
- ]);
475
- expect(settings.workspace.settings.excludedProjectEnvVars).toEqual([
476
- 'WORKSPACE_DEBUG',
477
- 'WORKSPACE_VAR',
478
- ]);
479
- expect(settings.merged.excludedProjectEnvVars).toEqual([
480
- 'WORKSPACE_DEBUG',
481
- 'WORKSPACE_VAR',
482
- ]);
483
- });
484
-
485
- it('should default contextFileName to undefined if not in any settings file', () => {
486
- (mockFsExistsSync as Mock).mockReturnValue(true);
487
- const userSettingsContent = { theme: 'dark' };
488
- const workspaceSettingsContent = { sandbox: true };
489
- (fs.readFileSync as Mock).mockImplementation(
490
- (p: fs.PathOrFileDescriptor) => {
491
- if (p === USER_SETTINGS_PATH)
492
- return JSON.stringify(userSettingsContent);
493
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
494
- return JSON.stringify(workspaceSettingsContent);
495
- return '';
496
- },
497
- );
498
-
499
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
500
- expect(settings.merged.contextFileName).toBeUndefined();
501
- });
502
-
503
- it('should load telemetry setting from user settings', () => {
504
- (mockFsExistsSync as Mock).mockImplementation(
505
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
506
- );
507
- const userSettingsContent = { telemetry: true };
508
- (fs.readFileSync as Mock).mockImplementation(
509
- (p: fs.PathOrFileDescriptor) => {
510
- if (p === USER_SETTINGS_PATH)
511
- return JSON.stringify(userSettingsContent);
512
- return '{}';
513
- },
514
- );
515
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
516
- expect(settings.merged.telemetry).toBe(true);
517
- });
518
-
519
- it('should load telemetry setting from workspace settings', () => {
520
- (mockFsExistsSync as Mock).mockImplementation(
521
- (p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
522
- );
523
- const workspaceSettingsContent = { telemetry: false };
524
- (fs.readFileSync as Mock).mockImplementation(
525
- (p: fs.PathOrFileDescriptor) => {
526
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
527
- return JSON.stringify(workspaceSettingsContent);
528
- return '{}';
529
- },
530
- );
531
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
532
- expect(settings.merged.telemetry).toBe(false);
533
- });
534
-
535
- it('should prioritize workspace telemetry setting over user setting', () => {
536
- (mockFsExistsSync as Mock).mockReturnValue(true);
537
- const userSettingsContent = { telemetry: true };
538
- const workspaceSettingsContent = { telemetry: false };
539
- (fs.readFileSync as Mock).mockImplementation(
540
- (p: fs.PathOrFileDescriptor) => {
541
- if (p === USER_SETTINGS_PATH)
542
- return JSON.stringify(userSettingsContent);
543
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
544
- return JSON.stringify(workspaceSettingsContent);
545
- return '{}';
546
- },
547
- );
548
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
549
- expect(settings.merged.telemetry).toBe(false);
550
- });
551
-
552
- it('should have telemetry as undefined if not in any settings file', () => {
553
- (mockFsExistsSync as Mock).mockReturnValue(false); // No settings files exist
554
- (fs.readFileSync as Mock).mockReturnValue('{}');
555
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
556
- expect(settings.merged.telemetry).toBeUndefined();
557
- expect(settings.merged.customThemes).toEqual({});
558
- expect(settings.merged.mcpServers).toEqual({});
559
- });
560
-
561
- it('should merge MCP servers correctly, with workspace taking precedence', () => {
562
- (mockFsExistsSync as Mock).mockReturnValue(true);
563
- const userSettingsContent = {
564
- mcpServers: {
565
- 'user-server': {
566
- command: 'user-command',
567
- args: ['--user-arg'],
568
- description: 'User MCP server',
569
- },
570
- 'shared-server': {
571
- command: 'user-shared-command',
572
- description: 'User shared server config',
573
- },
574
- },
575
- };
576
- const workspaceSettingsContent = {
577
- mcpServers: {
578
- 'workspace-server': {
579
- command: 'workspace-command',
580
- args: ['--workspace-arg'],
581
- description: 'Workspace MCP server',
582
- },
583
- 'shared-server': {
584
- command: 'workspace-shared-command',
585
- description: 'Workspace shared server config',
586
- },
587
- },
588
- };
589
-
590
- (fs.readFileSync as Mock).mockImplementation(
591
- (p: fs.PathOrFileDescriptor) => {
592
- if (p === USER_SETTINGS_PATH)
593
- return JSON.stringify(userSettingsContent);
594
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
595
- return JSON.stringify(workspaceSettingsContent);
596
- return '';
597
- },
598
- );
599
-
600
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
601
-
602
- expect(settings.user.settings).toEqual(userSettingsContent);
603
- expect(settings.workspace.settings).toEqual(workspaceSettingsContent);
604
- expect(settings.merged.mcpServers).toEqual({
605
- 'user-server': {
606
- command: 'user-command',
607
- args: ['--user-arg'],
608
- description: 'User MCP server',
609
- },
610
- 'workspace-server': {
611
- command: 'workspace-command',
612
- args: ['--workspace-arg'],
613
- description: 'Workspace MCP server',
614
- },
615
- 'shared-server': {
616
- command: 'workspace-shared-command',
617
- description: 'Workspace shared server config',
618
- },
619
- });
620
- });
621
-
622
- it('should handle MCP servers when only in user settings', () => {
623
- (mockFsExistsSync as Mock).mockImplementation(
624
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
625
- );
626
- const userSettingsContent = {
627
- mcpServers: {
628
- 'user-only-server': {
629
- command: 'user-only-command',
630
- description: 'User only server',
631
- },
632
- },
633
- };
634
- (fs.readFileSync as Mock).mockImplementation(
635
- (p: fs.PathOrFileDescriptor) => {
636
- if (p === USER_SETTINGS_PATH)
637
- return JSON.stringify(userSettingsContent);
638
- return '';
639
- },
640
- );
641
-
642
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
643
- expect(settings.merged.mcpServers).toEqual({
644
- 'user-only-server': {
645
- command: 'user-only-command',
646
- description: 'User only server',
647
- },
648
- });
649
- });
650
-
651
- it('should handle MCP servers when only in workspace settings', () => {
652
- (mockFsExistsSync as Mock).mockImplementation(
653
- (p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
654
- );
655
- const workspaceSettingsContent = {
656
- mcpServers: {
657
- 'workspace-only-server': {
658
- command: 'workspace-only-command',
659
- description: 'Workspace only server',
660
- },
661
- },
662
- };
663
- (fs.readFileSync as Mock).mockImplementation(
664
- (p: fs.PathOrFileDescriptor) => {
665
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
666
- return JSON.stringify(workspaceSettingsContent);
667
- return '';
668
- },
669
- );
670
-
671
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
672
- expect(settings.merged.mcpServers).toEqual({
673
- 'workspace-only-server': {
674
- command: 'workspace-only-command',
675
- description: 'Workspace only server',
676
- },
677
- });
678
- });
679
-
680
- it('should have mcpServers as empty object if not in any settings file', () => {
681
- (mockFsExistsSync as Mock).mockReturnValue(false); // No settings files exist
682
- (fs.readFileSync as Mock).mockReturnValue('{}');
683
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
684
- expect(settings.merged.mcpServers).toEqual({});
685
- });
686
-
687
- it('should merge chatCompression settings, with workspace taking precedence', () => {
688
- (mockFsExistsSync as Mock).mockReturnValue(true);
689
- const userSettingsContent = {
690
- chatCompression: { contextPercentageThreshold: 0.5 },
691
- };
692
- const workspaceSettingsContent = {
693
- chatCompression: { contextPercentageThreshold: 0.8 },
694
- };
695
-
696
- (fs.readFileSync as Mock).mockImplementation(
697
- (p: fs.PathOrFileDescriptor) => {
698
- if (p === USER_SETTINGS_PATH)
699
- return JSON.stringify(userSettingsContent);
700
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
701
- return JSON.stringify(workspaceSettingsContent);
702
- return '{}';
703
- },
704
- );
705
-
706
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
707
-
708
- expect(settings.user.settings.chatCompression).toEqual({
709
- contextPercentageThreshold: 0.5,
710
- });
711
- expect(settings.workspace.settings.chatCompression).toEqual({
712
- contextPercentageThreshold: 0.8,
713
- });
714
- expect(settings.merged.chatCompression).toEqual({
715
- contextPercentageThreshold: 0.8,
716
- });
717
- });
718
-
719
- it('should handle chatCompression when only in user settings', () => {
720
- (mockFsExistsSync as Mock).mockImplementation(
721
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
722
- );
723
- const userSettingsContent = {
724
- chatCompression: { contextPercentageThreshold: 0.5 },
725
- };
726
- (fs.readFileSync as Mock).mockImplementation(
727
- (p: fs.PathOrFileDescriptor) => {
728
- if (p === USER_SETTINGS_PATH)
729
- return JSON.stringify(userSettingsContent);
730
- return '{}';
731
- },
732
- );
733
-
734
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
735
- expect(settings.merged.chatCompression).toEqual({
736
- contextPercentageThreshold: 0.5,
737
- });
738
- });
739
-
740
- it('should have chatCompression as an empty object if not in any settings file', () => {
741
- (mockFsExistsSync as Mock).mockReturnValue(false); // No settings files exist
742
- (fs.readFileSync as Mock).mockReturnValue('{}');
743
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
744
- expect(settings.merged.chatCompression).toEqual({});
745
- });
746
-
747
- it('should ignore chatCompression if contextPercentageThreshold is invalid', () => {
748
- const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => {});
749
- (mockFsExistsSync as Mock).mockImplementation(
750
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
751
- );
752
- const userSettingsContent = {
753
- chatCompression: { contextPercentageThreshold: 1.5 },
754
- };
755
- (fs.readFileSync as Mock).mockImplementation(
756
- (p: fs.PathOrFileDescriptor) => {
757
- if (p === USER_SETTINGS_PATH)
758
- return JSON.stringify(userSettingsContent);
759
- return '{}';
760
- },
761
- );
762
-
763
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
764
- expect(settings.merged.chatCompression).toBeUndefined();
765
- expect(warnSpy).toHaveBeenCalledWith(
766
- 'Invalid value for chatCompression.contextPercentageThreshold: "1.5". Please use a value between 0 and 1. Using default compression settings.',
767
- );
768
- warnSpy.mockRestore();
769
- });
770
-
771
- it('should deep merge chatCompression settings', () => {
772
- (mockFsExistsSync as Mock).mockReturnValue(true);
773
- const userSettingsContent = {
774
- chatCompression: { contextPercentageThreshold: 0.5 },
775
- };
776
- const workspaceSettingsContent = {
777
- chatCompression: {},
778
- };
779
-
780
- (fs.readFileSync as Mock).mockImplementation(
781
- (p: fs.PathOrFileDescriptor) => {
782
- if (p === USER_SETTINGS_PATH)
783
- return JSON.stringify(userSettingsContent);
784
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
785
- return JSON.stringify(workspaceSettingsContent);
786
- return '{}';
787
- },
788
- );
789
-
790
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
791
-
792
- expect(settings.merged.chatCompression).toEqual({
793
- contextPercentageThreshold: 0.5,
794
- });
795
- });
796
-
797
- it('should merge includeDirectories from all scopes', () => {
798
- (mockFsExistsSync as Mock).mockReturnValue(true);
799
- const systemSettingsContent = {
800
- includeDirectories: ['/system/dir'],
801
- };
802
- const userSettingsContent = {
803
- includeDirectories: ['/user/dir1', '/user/dir2'],
804
- };
805
- const workspaceSettingsContent = {
806
- includeDirectories: ['/workspace/dir'],
807
- };
808
-
809
- (fs.readFileSync as Mock).mockImplementation(
810
- (p: fs.PathOrFileDescriptor) => {
811
- if (p === getSystemSettingsPath())
812
- return JSON.stringify(systemSettingsContent);
813
- if (p === USER_SETTINGS_PATH)
814
- return JSON.stringify(userSettingsContent);
815
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
816
- return JSON.stringify(workspaceSettingsContent);
817
- return '{}';
818
- },
819
- );
820
-
821
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
822
-
823
- expect(settings.merged.includeDirectories).toEqual([
824
- '/system/dir',
825
- '/user/dir1',
826
- '/user/dir2',
827
- '/workspace/dir',
828
- ]);
829
- });
830
-
831
- it('should handle JSON parsing errors gracefully', () => {
832
- (mockFsExistsSync as Mock).mockReturnValue(true); // Both files "exist"
833
- const invalidJsonContent = 'invalid json';
834
- const userReadError = new SyntaxError(
835
- "Expected ',' or '}' after property value in JSON at position 10",
836
- );
837
- const workspaceReadError = new SyntaxError(
838
- 'Unexpected token i in JSON at position 0',
839
- );
840
-
841
- (fs.readFileSync as Mock).mockImplementation(
842
- (p: fs.PathOrFileDescriptor) => {
843
- if (p === USER_SETTINGS_PATH) {
844
- // Simulate JSON.parse throwing for user settings
845
- vi.spyOn(JSON, 'parse').mockImplementationOnce(() => {
846
- throw userReadError;
847
- });
848
- return invalidJsonContent; // Content that would cause JSON.parse to throw
849
- }
850
- if (p === MOCK_WORKSPACE_SETTINGS_PATH) {
851
- // Simulate JSON.parse throwing for workspace settings
852
- vi.spyOn(JSON, 'parse').mockImplementationOnce(() => {
853
- throw workspaceReadError;
854
- });
855
- return invalidJsonContent;
856
- }
857
- return '{}'; // Default for other reads
858
- },
859
- );
860
-
861
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
862
-
863
- // Check that settings are empty due to parsing errors
864
- expect(settings.user.settings).toEqual({});
865
- expect(settings.workspace.settings).toEqual({});
866
- expect(settings.merged).toEqual({
867
- customThemes: {},
868
- mcpServers: {},
869
- includeDirectories: [],
870
- chatCompression: {},
871
- });
872
-
873
- // Check that error objects are populated in settings.errors
874
- expect(settings.errors).toBeDefined();
875
- // Assuming both user and workspace files cause errors and are added in order
876
- expect(settings.errors.length).toEqual(2);
877
-
878
- const userError = settings.errors.find(
879
- (e) => e.path === USER_SETTINGS_PATH,
880
- );
881
- expect(userError).toBeDefined();
882
- expect(userError?.message).toBe(userReadError.message);
883
-
884
- const workspaceError = settings.errors.find(
885
- (e) => e.path === MOCK_WORKSPACE_SETTINGS_PATH,
886
- );
887
- expect(workspaceError).toBeDefined();
888
- expect(workspaceError?.message).toBe(workspaceReadError.message);
889
-
890
- // Restore JSON.parse mock if it was spied on specifically for this test
891
- vi.restoreAllMocks(); // Or more targeted restore if needed
892
- });
893
-
894
- it('should resolve environment variables in user settings', () => {
895
- process.env['TEST_API_KEY'] = 'user_api_key_from_env';
896
- const userSettingsContent = {
897
- apiKey: '$TEST_API_KEY',
898
- someUrl: 'https://test.com/${TEST_API_KEY}',
899
- };
900
- (mockFsExistsSync as Mock).mockImplementation(
901
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
902
- );
903
- (fs.readFileSync as Mock).mockImplementation(
904
- (p: fs.PathOrFileDescriptor) => {
905
- if (p === USER_SETTINGS_PATH)
906
- return JSON.stringify(userSettingsContent);
907
- return '{}';
908
- },
909
- );
910
-
911
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
912
- // @ts-expect-error: dynamic property for test
913
- expect(settings.user.settings.apiKey).toBe('user_api_key_from_env');
914
- // @ts-expect-error: dynamic property for test
915
- expect(settings.user.settings.someUrl).toBe(
916
- 'https://test.com/user_api_key_from_env',
917
- );
918
- // @ts-expect-error: dynamic property for test
919
- expect(settings.merged.apiKey).toBe('user_api_key_from_env');
920
- delete process.env['TEST_API_KEY'];
921
- });
922
-
923
- it('should resolve environment variables in workspace settings', () => {
924
- process.env['WORKSPACE_ENDPOINT'] = 'workspace_endpoint_from_env';
925
- const workspaceSettingsContent = {
926
- endpoint: '${WORKSPACE_ENDPOINT}/api',
927
- nested: { value: '$WORKSPACE_ENDPOINT' },
928
- };
929
- (mockFsExistsSync as Mock).mockImplementation(
930
- (p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
931
- );
932
- (fs.readFileSync as Mock).mockImplementation(
933
- (p: fs.PathOrFileDescriptor) => {
934
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
935
- return JSON.stringify(workspaceSettingsContent);
936
- return '{}';
937
- },
938
- );
939
-
940
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
941
- expect(settings.workspace.settings.endpoint).toBe(
942
- 'workspace_endpoint_from_env/api',
943
- );
944
- expect(settings.workspace.settings.nested.value).toBe(
945
- 'workspace_endpoint_from_env',
946
- );
947
- // @ts-expect-error: dynamic property for test
948
- expect(settings.merged.endpoint).toBe('workspace_endpoint_from_env/api');
949
- delete process.env['WORKSPACE_ENDPOINT'];
950
- });
951
-
952
- it('should correctly resolve and merge env variables from different scopes', () => {
953
- process.env['SYSTEM_VAR'] = 'system_value';
954
- process.env['USER_VAR'] = 'user_value';
955
- process.env['WORKSPACE_VAR'] = 'workspace_value';
956
- process.env['SHARED_VAR'] = 'final_value';
957
-
958
- const systemSettingsContent = {
959
- configValue: '$SHARED_VAR',
960
- systemOnly: '$SYSTEM_VAR',
961
- };
962
- const userSettingsContent = {
963
- configValue: '$SHARED_VAR',
964
- userOnly: '$USER_VAR',
965
- theme: 'dark',
966
- };
967
- const workspaceSettingsContent = {
968
- configValue: '$SHARED_VAR',
969
- workspaceOnly: '$WORKSPACE_VAR',
970
- theme: 'light',
971
- };
972
-
973
- (mockFsExistsSync as Mock).mockReturnValue(true);
974
- (fs.readFileSync as Mock).mockImplementation(
975
- (p: fs.PathOrFileDescriptor) => {
976
- if (p === getSystemSettingsPath()) {
977
- return JSON.stringify(systemSettingsContent);
978
- }
979
- if (p === USER_SETTINGS_PATH) {
980
- return JSON.stringify(userSettingsContent);
981
- }
982
- if (p === MOCK_WORKSPACE_SETTINGS_PATH) {
983
- return JSON.stringify(workspaceSettingsContent);
984
- }
985
- return '{}';
986
- },
987
- );
988
-
989
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
990
-
991
- // Check resolved values in individual scopes
992
- // @ts-expect-error: dynamic property for test
993
- expect(settings.system.settings.configValue).toBe('final_value');
994
- // @ts-expect-error: dynamic property for test
995
- expect(settings.system.settings.systemOnly).toBe('system_value');
996
- // @ts-expect-error: dynamic property for test
997
- expect(settings.user.settings.configValue).toBe('final_value');
998
- // @ts-expect-error: dynamic property for test
999
- expect(settings.user.settings.userOnly).toBe('user_value');
1000
- // @ts-expect-error: dynamic property for test
1001
- expect(settings.workspace.settings.configValue).toBe('final_value');
1002
- // @ts-expect-error: dynamic property for test
1003
- expect(settings.workspace.settings.workspaceOnly).toBe('workspace_value');
1004
-
1005
- // Check merged values (system > workspace > user)
1006
- // @ts-expect-error: dynamic property for test
1007
- expect(settings.merged.configValue).toBe('final_value');
1008
- // @ts-expect-error: dynamic property for test
1009
- expect(settings.merged.systemOnly).toBe('system_value');
1010
- // @ts-expect-error: dynamic property for test
1011
- expect(settings.merged.userOnly).toBe('user_value');
1012
- // @ts-expect-error: dynamic property for test
1013
- expect(settings.merged.workspaceOnly).toBe('workspace_value');
1014
- expect(settings.merged.theme).toBe('light'); // workspace overrides user
1015
-
1016
- delete process.env['SYSTEM_VAR'];
1017
- delete process.env['USER_VAR'];
1018
- delete process.env['WORKSPACE_VAR'];
1019
- delete process.env['SHARED_VAR'];
1020
- });
1021
-
1022
- it('should correctly merge dnsResolutionOrder with workspace taking precedence', () => {
1023
- (mockFsExistsSync as Mock).mockReturnValue(true);
1024
- const userSettingsContent = {
1025
- dnsResolutionOrder: 'ipv4first',
1026
- };
1027
- const workspaceSettingsContent = {
1028
- dnsResolutionOrder: 'verbatim',
1029
- };
1030
-
1031
- (fs.readFileSync as Mock).mockImplementation(
1032
- (p: fs.PathOrFileDescriptor) => {
1033
- if (p === USER_SETTINGS_PATH)
1034
- return JSON.stringify(userSettingsContent);
1035
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
1036
- return JSON.stringify(workspaceSettingsContent);
1037
- return '{}';
1038
- },
1039
- );
1040
-
1041
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1042
- expect(settings.merged.dnsResolutionOrder).toBe('verbatim');
1043
- });
1044
-
1045
- it('should use user dnsResolutionOrder if workspace is not defined', () => {
1046
- (mockFsExistsSync as Mock).mockImplementation(
1047
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
1048
- );
1049
- const userSettingsContent = {
1050
- dnsResolutionOrder: 'verbatim',
1051
- };
1052
- (fs.readFileSync as Mock).mockImplementation(
1053
- (p: fs.PathOrFileDescriptor) => {
1054
- if (p === USER_SETTINGS_PATH)
1055
- return JSON.stringify(userSettingsContent);
1056
- return '{}';
1057
- },
1058
- );
1059
-
1060
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1061
- expect(settings.merged.dnsResolutionOrder).toBe('verbatim');
1062
- });
1063
-
1064
- it('should leave unresolved environment variables as is', () => {
1065
- const userSettingsContent = { apiKey: '$UNDEFINED_VAR' };
1066
- (mockFsExistsSync as Mock).mockImplementation(
1067
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
1068
- );
1069
- (fs.readFileSync as Mock).mockImplementation(
1070
- (p: fs.PathOrFileDescriptor) => {
1071
- if (p === USER_SETTINGS_PATH)
1072
- return JSON.stringify(userSettingsContent);
1073
- return '{}';
1074
- },
1075
- );
1076
-
1077
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1078
- expect(settings.user.settings.apiKey).toBe('$UNDEFINED_VAR');
1079
- expect(settings.merged.apiKey).toBe('$UNDEFINED_VAR');
1080
- });
1081
-
1082
- it('should resolve multiple environment variables in a single string', () => {
1083
- process.env['VAR_A'] = 'valueA';
1084
- process.env['VAR_B'] = 'valueB';
1085
- const userSettingsContent = { path: '/path/$VAR_A/${VAR_B}/end' };
1086
- (mockFsExistsSync as Mock).mockImplementation(
1087
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
1088
- );
1089
- (fs.readFileSync as Mock).mockImplementation(
1090
- (p: fs.PathOrFileDescriptor) => {
1091
- if (p === USER_SETTINGS_PATH)
1092
- return JSON.stringify(userSettingsContent);
1093
- return '{}';
1094
- },
1095
- );
1096
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1097
- expect(settings.user.settings.path).toBe('/path/valueA/valueB/end');
1098
- delete process.env['VAR_A'];
1099
- delete process.env['VAR_B'];
1100
- });
1101
-
1102
- it('should resolve environment variables in arrays', () => {
1103
- process.env['ITEM_1'] = 'item1_env';
1104
- process.env['ITEM_2'] = 'item2_env';
1105
- const userSettingsContent = { list: ['$ITEM_1', '${ITEM_2}', 'literal'] };
1106
- (mockFsExistsSync as Mock).mockImplementation(
1107
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
1108
- );
1109
- (fs.readFileSync as Mock).mockImplementation(
1110
- (p: fs.PathOrFileDescriptor) => {
1111
- if (p === USER_SETTINGS_PATH)
1112
- return JSON.stringify(userSettingsContent);
1113
- return '{}';
1114
- },
1115
- );
1116
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1117
- expect(settings.user.settings.list).toEqual([
1118
- 'item1_env',
1119
- 'item2_env',
1120
- 'literal',
1121
- ]);
1122
- delete process.env['ITEM_1'];
1123
- delete process.env['ITEM_2'];
1124
- });
1125
-
1126
- it('should correctly pass through null, boolean, and number types, and handle undefined properties', () => {
1127
- process.env['MY_ENV_STRING'] = 'env_string_value';
1128
- process.env['MY_ENV_STRING_NESTED'] = 'env_string_nested_value';
1129
-
1130
- const userSettingsContent = {
1131
- nullVal: null,
1132
- trueVal: true,
1133
- falseVal: false,
1134
- numberVal: 123.45,
1135
- stringVal: '$MY_ENV_STRING',
1136
- nestedObj: {
1137
- nestedNull: null,
1138
- nestedBool: true,
1139
- nestedNum: 0,
1140
- nestedString: 'literal',
1141
- anotherEnv: '${MY_ENV_STRING_NESTED}',
1142
- },
1143
- };
1144
-
1145
- (mockFsExistsSync as Mock).mockImplementation(
1146
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
1147
- );
1148
- (fs.readFileSync as Mock).mockImplementation(
1149
- (p: fs.PathOrFileDescriptor) => {
1150
- if (p === USER_SETTINGS_PATH)
1151
- return JSON.stringify(userSettingsContent);
1152
- return '{}';
1153
- },
1154
- );
1155
-
1156
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1157
-
1158
- expect(settings.user.settings.nullVal).toBeNull();
1159
- expect(settings.user.settings.trueVal).toBe(true);
1160
- expect(settings.user.settings.falseVal).toBe(false);
1161
- expect(settings.user.settings.numberVal).toBe(123.45);
1162
- expect(settings.user.settings.stringVal).toBe('env_string_value');
1163
- expect(settings.user.settings.undefinedVal).toBeUndefined();
1164
-
1165
- expect(settings.user.settings.nestedObj.nestedNull).toBeNull();
1166
- expect(settings.user.settings.nestedObj.nestedBool).toBe(true);
1167
- expect(settings.user.settings.nestedObj.nestedNum).toBe(0);
1168
- expect(settings.user.settings.nestedObj.nestedString).toBe('literal');
1169
- expect(settings.user.settings.nestedObj.anotherEnv).toBe(
1170
- 'env_string_nested_value',
1171
- );
1172
-
1173
- delete process.env['MY_ENV_STRING'];
1174
- delete process.env['MY_ENV_STRING_NESTED'];
1175
- });
1176
-
1177
- it('should resolve multiple concatenated environment variables in a single string value', () => {
1178
- process.env['TEST_HOST'] = 'myhost';
1179
- process.env['TEST_PORT'] = '9090';
1180
- const userSettingsContent = {
1181
- serverAddress: '${TEST_HOST}:${TEST_PORT}/api',
1182
- };
1183
- (mockFsExistsSync as Mock).mockImplementation(
1184
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
1185
- );
1186
- (fs.readFileSync as Mock).mockImplementation(
1187
- (p: fs.PathOrFileDescriptor) => {
1188
- if (p === USER_SETTINGS_PATH)
1189
- return JSON.stringify(userSettingsContent);
1190
- return '{}';
1191
- },
1192
- );
1193
-
1194
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1195
- expect(settings.user.settings.serverAddress).toBe('myhost:9090/api');
1196
-
1197
- delete process.env['TEST_HOST'];
1198
- delete process.env['TEST_PORT'];
1199
- });
1200
-
1201
- describe('when FSS_LINK_CLI_SYSTEM_SETTINGS_PATH is set', () => {
1202
- const MOCK_ENV_SYSTEM_SETTINGS_PATH = '/mock/env/system/settings.json';
1203
-
1204
- beforeEach(() => {
1205
- process.env['FSS_LINK_CLI_SYSTEM_SETTINGS_PATH'] =
1206
- MOCK_ENV_SYSTEM_SETTINGS_PATH;
1207
- });
1208
-
1209
- afterEach(() => {
1210
- delete process.env['FSS_LINK_CLI_SYSTEM_SETTINGS_PATH'];
1211
- });
1212
-
1213
- it('should load system settings from the path specified in the environment variable', () => {
1214
- (mockFsExistsSync as Mock).mockImplementation(
1215
- (p: fs.PathLike) => p === MOCK_ENV_SYSTEM_SETTINGS_PATH,
1216
- );
1217
- const systemSettingsContent = {
1218
- theme: 'env-var-theme',
1219
- sandbox: true,
1220
- };
1221
- (fs.readFileSync as Mock).mockImplementation(
1222
- (p: fs.PathOrFileDescriptor) => {
1223
- if (p === MOCK_ENV_SYSTEM_SETTINGS_PATH)
1224
- return JSON.stringify(systemSettingsContent);
1225
- return '{}';
1226
- },
1227
- );
1228
-
1229
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1230
-
1231
- expect(fs.readFileSync).toHaveBeenCalledWith(
1232
- MOCK_ENV_SYSTEM_SETTINGS_PATH,
1233
- 'utf-8',
1234
- );
1235
- expect(settings.system.path).toBe(MOCK_ENV_SYSTEM_SETTINGS_PATH);
1236
- expect(settings.system.settings).toEqual(systemSettingsContent);
1237
- expect(settings.merged).toEqual({
1238
- ...systemSettingsContent,
1239
- customThemes: {},
1240
- mcpServers: {},
1241
- includeDirectories: [],
1242
- chatCompression: {},
1243
- });
1244
- });
1245
- });
1246
- });
1247
-
1248
- describe('LoadedSettings class', () => {
1249
- it('setValue should update the correct scope and recompute merged settings', () => {
1250
- (mockFsExistsSync as Mock).mockReturnValue(false);
1251
- const loadedSettings = loadSettings(MOCK_WORKSPACE_DIR);
1252
-
1253
- vi.mocked(fs.writeFileSync).mockImplementation(() => {});
1254
- // mkdirSync is mocked in beforeEach to return undefined, which is fine for void usage
1255
-
1256
- loadedSettings.setValue(SettingScope.User, 'theme', 'matrix');
1257
- expect(loadedSettings.user.settings.theme).toBe('matrix');
1258
- expect(loadedSettings.merged.theme).toBe('matrix');
1259
- expect(fs.writeFileSync).toHaveBeenCalledWith(
1260
- USER_SETTINGS_PATH,
1261
- JSON.stringify({ theme: 'matrix' }, null, 2),
1262
- 'utf-8',
1263
- );
1264
-
1265
- loadedSettings.setValue(
1266
- SettingScope.Workspace,
1267
- 'contextFileName',
1268
- 'MY_AGENTS.md',
1269
- );
1270
- expect(loadedSettings.workspace.settings.contextFileName).toBe(
1271
- 'MY_AGENTS.md',
1272
- );
1273
- expect(loadedSettings.merged.contextFileName).toBe('MY_AGENTS.md');
1274
- expect(loadedSettings.merged.theme).toBe('matrix'); // User setting should still be there
1275
- expect(fs.writeFileSync).toHaveBeenCalledWith(
1276
- MOCK_WORKSPACE_SETTINGS_PATH,
1277
- JSON.stringify({ contextFileName: 'MY_AGENTS.md' }, null, 2),
1278
- 'utf-8',
1279
- );
1280
-
1281
- // System theme overrides user and workspace themes
1282
- loadedSettings.setValue(SettingScope.System, 'theme', 'ocean');
1283
-
1284
- expect(loadedSettings.system.settings.theme).toBe('ocean');
1285
- expect(loadedSettings.merged.theme).toBe('ocean');
1286
- });
1287
- });
1288
-
1289
- describe('excludedProjectEnvVars integration', () => {
1290
- const originalEnv = { ...process.env };
1291
-
1292
- beforeEach(() => {
1293
- process.env = { ...originalEnv };
1294
- });
1295
-
1296
- afterEach(() => {
1297
- process.env = originalEnv;
1298
- });
1299
-
1300
- it('should exclude DEBUG and DEBUG_MODE from project .env files by default', () => {
1301
- // Create a workspace settings file with excludedProjectEnvVars
1302
- const workspaceSettingsContent = {
1303
- excludedProjectEnvVars: ['DEBUG', 'DEBUG_MODE'],
1304
- };
1305
-
1306
- (mockFsExistsSync as Mock).mockImplementation(
1307
- (p: fs.PathLike) => p === MOCK_WORKSPACE_SETTINGS_PATH,
1308
- );
1309
-
1310
- (fs.readFileSync as Mock).mockImplementation(
1311
- (p: fs.PathOrFileDescriptor) => {
1312
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
1313
- return JSON.stringify(workspaceSettingsContent);
1314
- return '{}';
1315
- },
1316
- );
1317
-
1318
- // Mock findEnvFile to return a project .env file
1319
- const originalFindEnvFile = (
1320
- loadSettings as unknown as { findEnvFile: () => string }
1321
- ).findEnvFile;
1322
- (loadSettings as unknown as { findEnvFile: () => string }).findEnvFile =
1323
- () => '/mock/project/.env';
1324
-
1325
- // Mock fs.readFileSync for .env file content
1326
- const originalReadFileSync = fs.readFileSync;
1327
- (fs.readFileSync as Mock).mockImplementation(
1328
- (p: fs.PathOrFileDescriptor) => {
1329
- if (p === '/mock/project/.env') {
1330
- return 'DEBUG=true\nDEBUG_MODE=1\nGEMINI_API_KEY=test-key';
1331
- }
1332
- if (p === MOCK_WORKSPACE_SETTINGS_PATH) {
1333
- return JSON.stringify(workspaceSettingsContent);
1334
- }
1335
- return '{}';
1336
- },
1337
- );
1338
-
1339
- try {
1340
- // This will call loadEnvironment internally with the merged settings
1341
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1342
-
1343
- // Verify the settings were loaded correctly
1344
- expect(settings.merged.excludedProjectEnvVars).toEqual([
1345
- 'DEBUG',
1346
- 'DEBUG_MODE',
1347
- ]);
1348
-
1349
- // Note: We can't directly test process.env changes here because the mocking
1350
- // prevents the actual file system operations, but we can verify the settings
1351
- // are correctly merged and passed to loadEnvironment
1352
- } finally {
1353
- (loadSettings as unknown as { findEnvFile: () => string }).findEnvFile =
1354
- originalFindEnvFile;
1355
- (fs.readFileSync as Mock).mockImplementation(originalReadFileSync);
1356
- }
1357
- });
1358
-
1359
- it('should respect custom excludedProjectEnvVars from user settings', () => {
1360
- const userSettingsContent = {
1361
- excludedProjectEnvVars: ['NODE_ENV', 'DEBUG'],
1362
- };
1363
-
1364
- (mockFsExistsSync as Mock).mockImplementation(
1365
- (p: fs.PathLike) => p === USER_SETTINGS_PATH,
1366
- );
1367
-
1368
- (fs.readFileSync as Mock).mockImplementation(
1369
- (p: fs.PathOrFileDescriptor) => {
1370
- if (p === USER_SETTINGS_PATH)
1371
- return JSON.stringify(userSettingsContent);
1372
- return '{}';
1373
- },
1374
- );
1375
-
1376
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1377
- expect(settings.user.settings.excludedProjectEnvVars).toEqual([
1378
- 'NODE_ENV',
1379
- 'DEBUG',
1380
- ]);
1381
- expect(settings.merged.excludedProjectEnvVars).toEqual([
1382
- 'NODE_ENV',
1383
- 'DEBUG',
1384
- ]);
1385
- });
1386
-
1387
- it('should merge excludedProjectEnvVars with workspace taking precedence', () => {
1388
- const userSettingsContent = {
1389
- excludedProjectEnvVars: ['DEBUG', 'NODE_ENV', 'USER_VAR'],
1390
- };
1391
- const workspaceSettingsContent = {
1392
- excludedProjectEnvVars: ['WORKSPACE_DEBUG', 'WORKSPACE_VAR'],
1393
- };
1394
-
1395
- (mockFsExistsSync as Mock).mockReturnValue(true);
1396
-
1397
- (fs.readFileSync as Mock).mockImplementation(
1398
- (p: fs.PathOrFileDescriptor) => {
1399
- if (p === USER_SETTINGS_PATH)
1400
- return JSON.stringify(userSettingsContent);
1401
- if (p === MOCK_WORKSPACE_SETTINGS_PATH)
1402
- return JSON.stringify(workspaceSettingsContent);
1403
- return '{}';
1404
- },
1405
- );
1406
-
1407
- const settings = loadSettings(MOCK_WORKSPACE_DIR);
1408
-
1409
- expect(settings.user.settings.excludedProjectEnvVars).toEqual([
1410
- 'DEBUG',
1411
- 'NODE_ENV',
1412
- 'USER_VAR',
1413
- ]);
1414
- expect(settings.workspace.settings.excludedProjectEnvVars).toEqual([
1415
- 'WORKSPACE_DEBUG',
1416
- 'WORKSPACE_VAR',
1417
- ]);
1418
- expect(settings.merged.excludedProjectEnvVars).toEqual([
1419
- 'WORKSPACE_DEBUG',
1420
- 'WORKSPACE_VAR',
1421
- ]);
1422
- });
1423
- });
1424
- });