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,962 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
-
7
- import { exec, execSync, spawn, type ChildProcess } from 'node:child_process';
8
- import os from 'node:os';
9
- import path from 'node:path';
10
- import fs from 'node:fs';
11
- import { readFile } from 'node:fs/promises';
12
- import { quote, parse } from 'shell-quote';
13
- import {
14
- USER_SETTINGS_DIR,
15
- SETTINGS_DIRECTORY_NAME,
16
- } from '../config/settings.js';
17
- import { promisify } from 'util';
18
- import { Config, SandboxConfig } from 'fss-link-core';
19
- import { ConsolePatcher } from '../ui/utils/ConsolePatcher.js';
20
-
21
- const execAsync = promisify(exec);
22
-
23
- function getContainerPath(hostPath: string): string {
24
- if (os.platform() !== 'win32') {
25
- return hostPath;
26
- }
27
- const withForwardSlashes = hostPath.replace(/\\/g, '/');
28
- const match = withForwardSlashes.match(/^([A-Z]):\/(.*)/i);
29
- if (match) {
30
- return `/${match[1].toLowerCase()}/${match[2]}`;
31
- }
32
- return hostPath;
33
- }
34
-
35
- const LOCAL_DEV_SANDBOX_IMAGE_NAME = 'fss-link-sandbox';
36
- const SANDBOX_NETWORK_NAME = 'fss-link-sandbox';
37
- const SANDBOX_PROXY_NAME = 'fss-link-sandbox-proxy';
38
- const BUILTIN_SEATBELT_PROFILES = [
39
- 'permissive-open',
40
- 'permissive-closed',
41
- 'permissive-proxied',
42
- 'restrictive-open',
43
- 'restrictive-closed',
44
- 'restrictive-proxied',
45
- ];
46
-
47
- /**
48
- * Determines whether the sandbox container should be run with the current user's UID and GID.
49
- * This is often necessary on Linux systems (especially Debian/Ubuntu based) when using
50
- * rootful Docker without userns-remap configured, to avoid permission issues with
51
- * mounted volumes.
52
- *
53
- * The behavior is controlled by the `SANDBOX_SET_UID_GID` environment variable:
54
- * - If `SANDBOX_SET_UID_GID` is "1" or "true", this function returns `true`.
55
- * - If `SANDBOX_SET_UID_GID` is "0" or "false", this function returns `false`.
56
- * - If `SANDBOX_SET_UID_GID` is not set:
57
- * - On Debian/Ubuntu Linux, it defaults to `true`.
58
- * - On other OSes, or if OS detection fails, it defaults to `false`.
59
- *
60
- * For more context on running Docker containers as non-root, see:
61
- * https://medium.com/redbubble/running-a-docker-container-as-a-non-root-user-7d2e00f8ee15
62
- *
63
- * @returns {Promise<boolean>} A promise that resolves to true if the current user's UID/GID should be used, false otherwise.
64
- */
65
- async function shouldUseCurrentUserInSandbox(): Promise<boolean> {
66
- const envVar = process.env['SANDBOX_SET_UID_GID']?.toLowerCase().trim();
67
-
68
- if (envVar === '1' || envVar === 'true') {
69
- return true;
70
- }
71
- if (envVar === '0' || envVar === 'false') {
72
- return false;
73
- }
74
-
75
- // If environment variable is not explicitly set, check for Debian/Ubuntu Linux
76
- if (os.platform() === 'linux') {
77
- try {
78
- const osReleaseContent = await readFile('/etc/os-release', 'utf8');
79
- if (
80
- osReleaseContent.includes('ID=debian') ||
81
- osReleaseContent.includes('ID=ubuntu') ||
82
- osReleaseContent.match(/^ID_LIKE=.*debian.*/m) || // Covers derivatives
83
- osReleaseContent.match(/^ID_LIKE=.*ubuntu.*/m) // Covers derivatives
84
- ) {
85
- // note here and below we use console.error for informational messages on stderr
86
- console.error(
87
- 'INFO: Defaulting to use current user UID/GID for Debian/Ubuntu-based Linux.',
88
- );
89
- return true;
90
- }
91
- } catch (_err) {
92
- // Silently ignore if /etc/os-release is not found or unreadable.
93
- // The default (false) will be applied in this case.
94
- console.warn(
95
- 'Warning: Could not read /etc/os-release to auto-detect Debian/Ubuntu for UID/GID default.',
96
- );
97
- }
98
- }
99
- return false; // Default to false if no other condition is met
100
- }
101
-
102
- // docker does not allow container names to contain ':' or '/', so we
103
- // parse those out to shorten the name
104
- function parseImageName(image: string): string {
105
- const [fullName, tag] = image.split(':');
106
- const name = fullName.split('/').at(-1) ?? 'unknown-image';
107
- return tag ? `${name}-${tag}` : name;
108
- }
109
-
110
- function ports(): string[] {
111
- return (process.env['SANDBOX_PORTS'] ?? '')
112
- .split(',')
113
- .filter((p) => p.trim())
114
- .map((p) => p.trim());
115
- }
116
-
117
- function entrypoint(workdir: string): string[] {
118
- const isWindows = os.platform() === 'win32';
119
- const containerWorkdir = getContainerPath(workdir);
120
- const shellCmds = [];
121
- const pathSeparator = isWindows ? ';' : ':';
122
-
123
- let pathSuffix = '';
124
- if (process.env['PATH']) {
125
- const paths = process.env['PATH'].split(pathSeparator);
126
- for (const p of paths) {
127
- const containerPath = getContainerPath(p);
128
- if (
129
- containerPath.toLowerCase().startsWith(containerWorkdir.toLowerCase())
130
- ) {
131
- pathSuffix += `:${containerPath}`;
132
- }
133
- }
134
- }
135
- if (pathSuffix) {
136
- shellCmds.push(`export PATH="$PATH${pathSuffix}";`);
137
- }
138
-
139
- let pythonPathSuffix = '';
140
- if (process.env['PYTHONPATH']) {
141
- const paths = process.env['PYTHONPATH'].split(pathSeparator);
142
- for (const p of paths) {
143
- const containerPath = getContainerPath(p);
144
- if (
145
- containerPath.toLowerCase().startsWith(containerWorkdir.toLowerCase())
146
- ) {
147
- pythonPathSuffix += `:${containerPath}`;
148
- }
149
- }
150
- }
151
- if (pythonPathSuffix) {
152
- shellCmds.push(`export PYTHONPATH="$PYTHONPATH${pythonPathSuffix}";`);
153
- }
154
-
155
- const projectSandboxBashrc = path.join(
156
- SETTINGS_DIRECTORY_NAME,
157
- 'sandbox.bashrc',
158
- );
159
- if (fs.existsSync(projectSandboxBashrc)) {
160
- shellCmds.push(`source ${getContainerPath(projectSandboxBashrc)};`);
161
- }
162
-
163
- ports().forEach((p) =>
164
- shellCmds.push(
165
- `socat TCP4-LISTEN:${p},bind=$(hostname -i),fork,reuseaddr TCP4:127.0.0.1:${p} 2> /dev/null &`,
166
- ),
167
- );
168
-
169
- const cliArgs = process.argv.slice(2).map((arg) => quote([arg]));
170
- const cliCmd =
171
- process.env['NODE_ENV'] === 'development'
172
- ? process.env['DEBUG']
173
- ? 'npm run debug --'
174
- : 'npm rebuild && npm run start --'
175
- : process.env['DEBUG']
176
- ? `node --inspect-brk=0.0.0.0:${process.env['DEBUG_PORT'] || '9229'} $(which fss-link)`
177
- : 'fss-link';
178
-
179
- const args = [...shellCmds, cliCmd, ...cliArgs];
180
-
181
- return ['bash', '-c', args.join(' ')];
182
- }
183
-
184
- export async function start_sandbox(
185
- config: SandboxConfig,
186
- nodeArgs: string[] = [],
187
- cliConfig?: Config,
188
- ) {
189
- const patcher = new ConsolePatcher({
190
- debugMode: cliConfig?.getDebugMode() || !!process.env['DEBUG'],
191
- stderr: true,
192
- });
193
- patcher.patch();
194
-
195
- try {
196
- if (config.command === 'sandbox-exec') {
197
- // disallow BUILD_SANDBOX
198
- if (process.env['BUILD_SANDBOX']) {
199
- console.error('ERROR: cannot BUILD_SANDBOX when using macOS Seatbelt');
200
- process.exit(1);
201
- }
202
- const profile = (process.env['SEATBELT_PROFILE'] ??= 'permissive-open');
203
- let profileFile = new URL(`sandbox-macos-${profile}.sb`, import.meta.url)
204
- .pathname;
205
- // if profile name is not recognized, then look for file under project settings directory
206
- if (!BUILTIN_SEATBELT_PROFILES.includes(profile)) {
207
- profileFile = path.join(
208
- SETTINGS_DIRECTORY_NAME,
209
- `sandbox-macos-${profile}.sb`,
210
- );
211
- }
212
- if (!fs.existsSync(profileFile)) {
213
- console.error(
214
- `ERROR: missing macos seatbelt profile file '${profileFile}'`,
215
- );
216
- process.exit(1);
217
- }
218
- // Log on STDERR so it doesn't clutter the output on STDOUT
219
- console.error(`using macos seatbelt (profile: ${profile}) ...`);
220
- // if DEBUG is set, convert to --inspect-brk in NODE_OPTIONS
221
- const nodeOptions = [
222
- ...(process.env['DEBUG'] ? ['--inspect-brk'] : []),
223
- ...nodeArgs,
224
- ].join(' ');
225
-
226
- const args = [
227
- '-D',
228
- `TARGET_DIR=${fs.realpathSync(process.cwd())}`,
229
- '-D',
230
- `TMP_DIR=${fs.realpathSync(os.tmpdir())}`,
231
- '-D',
232
- `HOME_DIR=${fs.realpathSync(os.homedir())}`,
233
- '-D',
234
- `CACHE_DIR=${fs.realpathSync(execSync(`getconf DARWIN_USER_CACHE_DIR`).toString().trim())}`,
235
- ];
236
-
237
- // Add included directories from the workspace context
238
- // Always add 5 INCLUDE_DIR parameters to ensure .sb files can reference them
239
- const MAX_INCLUDE_DIRS = 5;
240
- const targetDir = fs.realpathSync(cliConfig?.getTargetDir() || '');
241
- const includedDirs: string[] = [];
242
-
243
- if (cliConfig) {
244
- const workspaceContext = cliConfig.getWorkspaceContext();
245
- const directories = workspaceContext.getDirectories();
246
-
247
- // Filter out TARGET_DIR
248
- for (const dir of directories) {
249
- const realDir = fs.realpathSync(dir);
250
- if (realDir !== targetDir) {
251
- includedDirs.push(realDir);
252
- }
253
- }
254
- }
255
-
256
- for (let i = 0; i < MAX_INCLUDE_DIRS; i++) {
257
- let dirPath = '/dev/null'; // Default to a safe path that won't cause issues
258
-
259
- if (i < includedDirs.length) {
260
- dirPath = includedDirs[i];
261
- }
262
-
263
- args.push('-D', `INCLUDE_DIR_${i}=${dirPath}`);
264
- }
265
-
266
- args.push(
267
- '-f',
268
- profileFile,
269
- 'sh',
270
- '-c',
271
- [
272
- `SANDBOX=sandbox-exec`,
273
- `NODE_OPTIONS="${nodeOptions}"`,
274
- ...process.argv.map((arg) => quote([arg])),
275
- ].join(' '),
276
- );
277
- // start and set up proxy if FSS_LINK_SANDBOX_PROXY_COMMAND is set
278
- const proxyCommand = process.env['FSS_LINK_SANDBOX_PROXY_COMMAND'];
279
- let proxyProcess: ChildProcess | undefined = undefined;
280
- let sandboxProcess: ChildProcess | undefined = undefined;
281
- const sandboxEnv = { ...process.env };
282
- if (proxyCommand) {
283
- const proxy =
284
- process.env['HTTPS_PROXY'] ||
285
- process.env['https_proxy'] ||
286
- process.env['HTTP_PROXY'] ||
287
- process.env['http_proxy'] ||
288
- 'http://localhost:8877';
289
- sandboxEnv['HTTPS_PROXY'] = proxy;
290
- sandboxEnv['https_proxy'] = proxy; // lower-case can be required, e.g. for curl
291
- sandboxEnv['HTTP_PROXY'] = proxy;
292
- sandboxEnv['http_proxy'] = proxy;
293
- const noProxy = process.env['NO_PROXY'] || process.env['no_proxy'];
294
- if (noProxy) {
295
- sandboxEnv['NO_PROXY'] = noProxy;
296
- sandboxEnv['no_proxy'] = noProxy;
297
- }
298
- proxyProcess = spawn(proxyCommand, {
299
- stdio: ['ignore', 'pipe', 'pipe'],
300
- shell: true,
301
- detached: true,
302
- });
303
- // install handlers to stop proxy on exit/signal
304
- const stopProxy = () => {
305
- console.log('stopping proxy ...');
306
- if (proxyProcess?.pid) {
307
- process.kill(-proxyProcess.pid, 'SIGTERM');
308
- }
309
- };
310
- process.on('exit', stopProxy);
311
- process.on('SIGINT', stopProxy);
312
- process.on('SIGTERM', stopProxy);
313
-
314
- // commented out as it disrupts ink rendering
315
- // proxyProcess.stdout?.on('data', (data) => {
316
- // console.info(data.toString());
317
- // });
318
- proxyProcess.stderr?.on('data', (data) => {
319
- console.error(data.toString());
320
- });
321
- proxyProcess.on('close', (code, signal) => {
322
- console.error(
323
- `ERROR: proxy command '${proxyCommand}' exited with code ${code}, signal ${signal}`,
324
- );
325
- if (sandboxProcess?.pid) {
326
- process.kill(-sandboxProcess.pid, 'SIGTERM');
327
- }
328
- process.exit(1);
329
- });
330
- console.log('waiting for proxy to start ...');
331
- await execAsync(
332
- `until timeout 0.25 curl -s http://localhost:8877; do sleep 0.25; done`,
333
- );
334
- }
335
- // spawn child and let it inherit stdio
336
- sandboxProcess = spawn(config.command, args, {
337
- stdio: 'inherit',
338
- });
339
- await new Promise((resolve) => sandboxProcess?.on('close', resolve));
340
- return;
341
- }
342
-
343
- console.error(`hopping into sandbox (command: ${config.command}) ...`);
344
-
345
- // determine full path for fss-link to distinguish linked vs installed setting
346
- const gcPath = fs.realpathSync(process.argv[1]);
347
-
348
- const projectSandboxDockerfile = path.join(
349
- SETTINGS_DIRECTORY_NAME,
350
- 'sandbox.Dockerfile',
351
- );
352
- const isCustomProjectSandbox = fs.existsSync(projectSandboxDockerfile);
353
-
354
- const image = config.image;
355
- const workdir = path.resolve(process.cwd());
356
- const containerWorkdir = getContainerPath(workdir);
357
-
358
- // if BUILD_SANDBOX is set, then call scripts/build_sandbox.js under fss-link repo
359
- //
360
- // note this can only be done with binary linked from fss-link repo
361
- if (process.env['BUILD_SANDBOX']) {
362
- if (!gcPath.includes('fss-link/packages/')) {
363
- console.error(
364
- 'ERROR: cannot build sandbox using installed fss-link binary; ' +
365
- 'run `npm link ./packages/cli` under fss-link repo to switch to linked binary.',
366
- );
367
- process.exit(1);
368
- } else {
369
- console.error('building sandbox ...');
370
- const gcRoot = gcPath.split('/packages/')[0];
371
- // if project folder has sandbox.Dockerfile under project settings folder, use that
372
- let buildArgs = '';
373
- const projectSandboxDockerfile = path.join(
374
- SETTINGS_DIRECTORY_NAME,
375
- 'sandbox.Dockerfile',
376
- );
377
- if (isCustomProjectSandbox) {
378
- console.error(`using ${projectSandboxDockerfile} for sandbox`);
379
- buildArgs += `-f ${path.resolve(projectSandboxDockerfile)} -i ${image}`;
380
- }
381
- execSync(
382
- `cd ${gcRoot} && node scripts/build_sandbox.js -s ${buildArgs}`,
383
- {
384
- stdio: 'inherit',
385
- env: {
386
- ...process.env,
387
- FSS_LINK_SANDBOX: config.command, // in case sandbox is enabled via flags (see config.ts under cli package)
388
- },
389
- },
390
- );
391
- }
392
- }
393
-
394
- // stop if image is missing
395
- if (!(await ensureSandboxImageIsPresent(config.command, image))) {
396
- const remedy =
397
- image === LOCAL_DEV_SANDBOX_IMAGE_NAME
398
- ? 'Try running `npm run build:all` or `npm run build:sandbox` under the fss-link repo to build it locally, or check the image name and your network connection.'
399
- : 'Please check the image name, your network connection, or notify support@fsscoding.com if the issue persists.';
400
- console.error(
401
- `ERROR: Sandbox image '${image}' is missing or could not be pulled. ${remedy}`,
402
- );
403
- process.exit(1);
404
- }
405
-
406
- // use interactive mode and auto-remove container on exit
407
- // run init binary inside container to forward signals & reap zombies
408
- const args = ['run', '-i', '--rm', '--init', '--workdir', containerWorkdir];
409
-
410
- // add custom flags from SANDBOX_FLAGS
411
- if (process.env['SANDBOX_FLAGS']) {
412
- const flags = parse(process.env['SANDBOX_FLAGS'], process.env).filter(
413
- (f): f is string => typeof f === 'string',
414
- );
415
- args.push(...flags);
416
- }
417
-
418
- // add TTY only if stdin is TTY as well, i.e. for piped input don't init TTY in container
419
- if (process.stdin.isTTY) {
420
- args.push('-t');
421
- }
422
-
423
- // mount current directory as working directory in sandbox (set via --workdir)
424
- args.push('--volume', `${workdir}:${containerWorkdir}`);
425
-
426
- // mount user settings directory inside container, after creating if missing
427
- // note user/home changes inside sandbox and we mount at BOTH paths for consistency
428
- const userSettingsDirOnHost = USER_SETTINGS_DIR;
429
- const userSettingsDirInSandbox = getContainerPath(
430
- `/home/node/${SETTINGS_DIRECTORY_NAME}`,
431
- );
432
- if (!fs.existsSync(userSettingsDirOnHost)) {
433
- fs.mkdirSync(userSettingsDirOnHost);
434
- }
435
- args.push(
436
- '--volume',
437
- `${userSettingsDirOnHost}:${userSettingsDirInSandbox}`,
438
- );
439
- if (userSettingsDirInSandbox !== userSettingsDirOnHost) {
440
- args.push(
441
- '--volume',
442
- `${userSettingsDirOnHost}:${getContainerPath(userSettingsDirOnHost)}`,
443
- );
444
- }
445
-
446
- // mount os.tmpdir() as os.tmpdir() inside container
447
- args.push('--volume', `${os.tmpdir()}:${getContainerPath(os.tmpdir())}`);
448
-
449
- // mount gcloud config directory if it exists
450
- const gcloudConfigDir = path.join(os.homedir(), '.config', 'gcloud');
451
- if (fs.existsSync(gcloudConfigDir)) {
452
- args.push(
453
- '--volume',
454
- `${gcloudConfigDir}:${getContainerPath(gcloudConfigDir)}:ro`,
455
- );
456
- }
457
-
458
- // mount ADC file if GOOGLE_APPLICATION_CREDENTIALS is set
459
- if (process.env['GOOGLE_APPLICATION_CREDENTIALS']) {
460
- const adcFile = process.env['GOOGLE_APPLICATION_CREDENTIALS'];
461
- if (fs.existsSync(adcFile)) {
462
- args.push('--volume', `${adcFile}:${getContainerPath(adcFile)}:ro`);
463
- args.push(
464
- '--env',
465
- `GOOGLE_APPLICATION_CREDENTIALS=${getContainerPath(adcFile)}`,
466
- );
467
- }
468
- }
469
-
470
- // mount paths listed in SANDBOX_MOUNTS
471
- if (process.env['SANDBOX_MOUNTS']) {
472
- for (let mount of process.env['SANDBOX_MOUNTS'].split(',')) {
473
- if (mount.trim()) {
474
- // parse mount as from:to:opts
475
- let [from, to, opts] = mount.trim().split(':');
476
- to = to || from; // default to mount at same path inside container
477
- opts = opts || 'ro'; // default to read-only
478
- mount = `${from}:${to}:${opts}`;
479
- // check that from path is absolute
480
- if (!path.isAbsolute(from)) {
481
- console.error(
482
- `ERROR: path '${from}' listed in SANDBOX_MOUNTS must be absolute`,
483
- );
484
- process.exit(1);
485
- }
486
- // check that from path exists on host
487
- if (!fs.existsSync(from)) {
488
- console.error(
489
- `ERROR: missing mount path '${from}' listed in SANDBOX_MOUNTS`,
490
- );
491
- process.exit(1);
492
- }
493
- console.error(`SANDBOX_MOUNTS: ${from} -> ${to} (${opts})`);
494
- args.push('--volume', mount);
495
- }
496
- }
497
- }
498
-
499
- // expose env-specified ports on the sandbox
500
- ports().forEach((p) => args.push('--publish', `${p}:${p}`));
501
-
502
- // if DEBUG is set, expose debugging port
503
- if (process.env['DEBUG']) {
504
- const debugPort = process.env['DEBUG_PORT'] || '9229';
505
- args.push(`--publish`, `${debugPort}:${debugPort}`);
506
- }
507
-
508
- // copy proxy environment variables, replacing localhost with SANDBOX_PROXY_NAME
509
- // copy as both upper-case and lower-case as is required by some utilities
510
- // FSS_LINK_SANDBOX_PROXY_COMMAND implies HTTPS_PROXY unless HTTP_PROXY is set
511
- const proxyCommand = process.env['FSS_LINK_SANDBOX_PROXY_COMMAND'];
512
-
513
- if (proxyCommand) {
514
- let proxy =
515
- process.env['HTTPS_PROXY'] ||
516
- process.env['https_proxy'] ||
517
- process.env['HTTP_PROXY'] ||
518
- process.env['http_proxy'] ||
519
- 'http://localhost:8877';
520
- proxy = proxy.replace('localhost', SANDBOX_PROXY_NAME);
521
- if (proxy) {
522
- args.push('--env', `HTTPS_PROXY=${proxy}`);
523
- args.push('--env', `https_proxy=${proxy}`); // lower-case can be required, e.g. for curl
524
- args.push('--env', `HTTP_PROXY=${proxy}`);
525
- args.push('--env', `http_proxy=${proxy}`);
526
- }
527
- const noProxy = process.env['NO_PROXY'] || process.env['no_proxy'];
528
- if (noProxy) {
529
- args.push('--env', `NO_PROXY=${noProxy}`);
530
- args.push('--env', `no_proxy=${noProxy}`);
531
- }
532
-
533
- // if using proxy, switch to internal networking through proxy
534
- if (proxy) {
535
- execSync(
536
- `${config.command} network inspect ${SANDBOX_NETWORK_NAME} || ${config.command} network create --internal ${SANDBOX_NETWORK_NAME}`,
537
- );
538
- args.push('--network', SANDBOX_NETWORK_NAME);
539
- // if proxy command is set, create a separate network w/ host access (i.e. non-internal)
540
- // we will run proxy in its own container connected to both host network and internal network
541
- // this allows proxy to work even on rootless podman on macos with host<->vm<->container isolation
542
- if (proxyCommand) {
543
- execSync(
544
- `${config.command} network inspect ${SANDBOX_PROXY_NAME} || ${config.command} network create ${SANDBOX_PROXY_NAME}`,
545
- );
546
- }
547
- }
548
- }
549
-
550
- // name container after image, plus numeric suffix to avoid conflicts
551
- const imageName = parseImageName(image);
552
- let index = 0;
553
- const containerNameCheck = execSync(
554
- `${config.command} ps -a --format "{{.Names}}"`,
555
- )
556
- .toString()
557
- .trim();
558
- while (containerNameCheck.includes(`${imageName}-${index}`)) {
559
- index++;
560
- }
561
- const containerName = `${imageName}-${index}`;
562
- args.push('--name', containerName, '--hostname', containerName);
563
-
564
- // copy GEMINI_API_KEY(s)
565
- if (process.env['GEMINI_API_KEY']) {
566
- args.push('--env', `GEMINI_API_KEY=${process.env['GEMINI_API_KEY']}`);
567
- }
568
- if (process.env['GOOGLE_API_KEY']) {
569
- args.push('--env', `GOOGLE_API_KEY=${process.env['GOOGLE_API_KEY']}`);
570
- }
571
-
572
- // copy OPENAI_API_KEY and related env vars for Qwen
573
- if (process.env['OPENAI_API_KEY']) {
574
- args.push('--env', `OPENAI_API_KEY=${process.env['OPENAI_API_KEY']}`);
575
- }
576
- // copy TAVILY_API_KEY for web search tool
577
- if (process.env['TAVILY_API_KEY']) {
578
- args.push('--env', `TAVILY_API_KEY=${process.env['TAVILY_API_KEY']}`);
579
- }
580
- if (process.env['OPENAI_BASE_URL']) {
581
- args.push('--env', `OPENAI_BASE_URL=${process.env['OPENAI_BASE_URL']}`);
582
- }
583
- if (process.env['OPENAI_MODEL']) {
584
- args.push('--env', `OPENAI_MODEL=${process.env['OPENAI_MODEL']}`);
585
- }
586
-
587
- // copy GOOGLE_GENAI_USE_VERTEXAI
588
- if (process.env['GOOGLE_GENAI_USE_VERTEXAI']) {
589
- args.push(
590
- '--env',
591
- `GOOGLE_GENAI_USE_VERTEXAI=${process.env['GOOGLE_GENAI_USE_VERTEXAI']}`,
592
- );
593
- }
594
-
595
- // copy GOOGLE_GENAI_USE_GCA
596
- if (process.env['GOOGLE_GENAI_USE_GCA']) {
597
- args.push(
598
- '--env',
599
- `GOOGLE_GENAI_USE_GCA=${process.env['GOOGLE_GENAI_USE_GCA']}`,
600
- );
601
- }
602
-
603
- // copy GOOGLE_CLOUD_PROJECT
604
- if (process.env['GOOGLE_CLOUD_PROJECT']) {
605
- args.push(
606
- '--env',
607
- `GOOGLE_CLOUD_PROJECT=${process.env['GOOGLE_CLOUD_PROJECT']}`,
608
- );
609
- }
610
-
611
- // copy GOOGLE_CLOUD_LOCATION
612
- if (process.env['GOOGLE_CLOUD_LOCATION']) {
613
- args.push(
614
- '--env',
615
- `GOOGLE_CLOUD_LOCATION=${process.env['GOOGLE_CLOUD_LOCATION']}`,
616
- );
617
- }
618
-
619
- // copy GEMINI_MODEL
620
- if (process.env['GEMINI_MODEL']) {
621
- args.push('--env', `GEMINI_MODEL=${process.env['GEMINI_MODEL']}`);
622
- }
623
-
624
- // copy TERM and COLORTERM to try to maintain terminal setup
625
- if (process.env['TERM']) {
626
- args.push('--env', `TERM=${process.env['TERM']}`);
627
- }
628
- if (process.env['COLORTERM']) {
629
- args.push('--env', `COLORTERM=${process.env['COLORTERM']}`);
630
- }
631
-
632
- // Pass through IDE mode environment variables
633
- for (const envVar of [
634
- 'FSS_LINK_IDE_SERVER_PORT',
635
- 'FSS_LINK_IDE_WORKSPACE_PATH',
636
- 'TERM_PROGRAM',
637
- ]) {
638
- if (process.env[envVar]) {
639
- args.push('--env', `${envVar}=${process.env[envVar]}`);
640
- }
641
- }
642
-
643
- // copy VIRTUAL_ENV if under working directory
644
- // also mount-replace VIRTUAL_ENV directory with <project_settings>/sandbox.venv
645
- // sandbox can then set up this new VIRTUAL_ENV directory using sandbox.bashrc (see below)
646
- // directory will be empty if not set up, which is still preferable to having host binaries
647
- if (
648
- process.env['VIRTUAL_ENV']
649
- ?.toLowerCase()
650
- .startsWith(workdir.toLowerCase())
651
- ) {
652
- const sandboxVenvPath = path.resolve(
653
- SETTINGS_DIRECTORY_NAME,
654
- 'sandbox.venv',
655
- );
656
- if (!fs.existsSync(sandboxVenvPath)) {
657
- fs.mkdirSync(sandboxVenvPath, { recursive: true });
658
- }
659
- args.push(
660
- '--volume',
661
- `${sandboxVenvPath}:${getContainerPath(process.env['VIRTUAL_ENV'])}`,
662
- );
663
- args.push(
664
- '--env',
665
- `VIRTUAL_ENV=${getContainerPath(process.env['VIRTUAL_ENV'])}`,
666
- );
667
- }
668
-
669
- // copy additional environment variables from SANDBOX_ENV
670
- if (process.env['SANDBOX_ENV']) {
671
- for (let env of process.env['SANDBOX_ENV'].split(',')) {
672
- if ((env = env.trim())) {
673
- if (env.includes('=')) {
674
- console.error(`SANDBOX_ENV: ${env}`);
675
- args.push('--env', env);
676
- } else {
677
- console.error(
678
- 'ERROR: SANDBOX_ENV must be a comma-separated list of key=value pairs',
679
- );
680
- process.exit(1);
681
- }
682
- }
683
- }
684
- }
685
-
686
- // copy NODE_OPTIONS
687
- const existingNodeOptions = process.env['NODE_OPTIONS'] || '';
688
- const allNodeOptions = [
689
- ...(existingNodeOptions ? [existingNodeOptions] : []),
690
- ...nodeArgs,
691
- ].join(' ');
692
-
693
- if (allNodeOptions.length > 0) {
694
- args.push('--env', `NODE_OPTIONS="${allNodeOptions}"`);
695
- }
696
-
697
- // set SANDBOX as container name
698
- args.push('--env', `SANDBOX=${containerName}`);
699
-
700
- // for podman only, use empty --authfile to skip unnecessary auth refresh overhead
701
- if (config.command === 'podman') {
702
- const emptyAuthFilePath = path.join(os.tmpdir(), 'empty_auth.json');
703
- fs.writeFileSync(emptyAuthFilePath, '{}', 'utf-8');
704
- args.push('--authfile', emptyAuthFilePath);
705
- }
706
-
707
- // Determine if the current user's UID/GID should be passed to the sandbox.
708
- // See shouldUseCurrentUserInSandbox for more details.
709
- let userFlag = '';
710
- const finalEntrypoint = entrypoint(workdir);
711
-
712
- if (process.env['FSS_LINK_CLI_INTEGRATION_TEST'] === 'true') {
713
- args.push('--user', 'root');
714
- userFlag = '--user root';
715
- } else if (await shouldUseCurrentUserInSandbox()) {
716
- // For the user-creation logic to work, the container must start as root.
717
- // The entrypoint script then handles dropping privileges to the correct user.
718
- args.push('--user', 'root');
719
-
720
- const uid = execSync('id -u').toString().trim();
721
- const gid = execSync('id -g').toString().trim();
722
-
723
- // Instead of passing --user to the main sandbox container, we let it
724
- // start as root, then create a user with the host's UID/GID, and
725
- // finally switch to that user to run the fss-link process. This is
726
- // necessary on Linux to ensure the user exists within the
727
- // container's /etc/passwd file, which is required by os.userInfo().
728
- const username = 'fss-link';
729
- const homeDir = getContainerPath(os.homedir());
730
-
731
- const setupUserCommands = [
732
- // Use -f with groupadd to avoid errors if the group already exists.
733
- `groupadd -f -g ${gid} ${username}`,
734
- // Create user only if it doesn't exist. Use -o for non-unique UID.
735
- `id -u ${username} &>/dev/null || useradd -o -u ${uid} -g ${gid} -d ${homeDir} -s /bin/bash ${username}`,
736
- ].join(' && ');
737
-
738
- const originalCommand = finalEntrypoint[2];
739
- const escapedOriginalCommand = originalCommand.replace(/'/g, "'\\''");
740
-
741
- // Use `su -p` to preserve the environment.
742
- const suCommand = `su -p ${username} -c '${escapedOriginalCommand}'`;
743
-
744
- // The entrypoint is always `['bash', '-c', '<command>']`, so we modify the command part.
745
- finalEntrypoint[2] = `${setupUserCommands} && ${suCommand}`;
746
-
747
- // We still need userFlag for the simpler proxy container, which does not have this issue.
748
- userFlag = `--user ${uid}:${gid}`;
749
- // When forcing a UID in the sandbox, $HOME can be reset to '/', so we copy $HOME as well.
750
- args.push('--env', `HOME=${os.homedir()}`);
751
- }
752
-
753
- // push container image name
754
- args.push(image);
755
-
756
- // push container entrypoint (including args)
757
- args.push(...finalEntrypoint);
758
-
759
- // start and set up proxy if FSS_LINK_SANDBOX_PROXY_COMMAND is set
760
- let proxyProcess: ChildProcess | undefined = undefined;
761
- let sandboxProcess: ChildProcess | undefined = undefined;
762
-
763
- if (proxyCommand) {
764
- // run proxyCommand in its own container
765
- const proxyContainerCommand = `${config.command} run --rm --init ${userFlag} --name ${SANDBOX_PROXY_NAME} --network ${SANDBOX_PROXY_NAME} -p 8877:8877 -v ${process.cwd()}:${workdir} --workdir ${workdir} ${image} ${proxyCommand}`;
766
- proxyProcess = spawn(proxyContainerCommand, {
767
- stdio: ['ignore', 'pipe', 'pipe'],
768
- shell: true,
769
- detached: true,
770
- });
771
- // install handlers to stop proxy on exit/signal
772
- const stopProxy = () => {
773
- console.log('stopping proxy container ...');
774
- execSync(`${config.command} rm -f ${SANDBOX_PROXY_NAME}`);
775
- };
776
- process.on('exit', stopProxy);
777
- process.on('SIGINT', stopProxy);
778
- process.on('SIGTERM', stopProxy);
779
-
780
- // commented out as it disrupts ink rendering
781
- // proxyProcess.stdout?.on('data', (data) => {
782
- // console.info(data.toString());
783
- // });
784
- proxyProcess.stderr?.on('data', (data) => {
785
- console.error(data.toString().trim());
786
- });
787
- proxyProcess.on('close', (code, signal) => {
788
- console.error(
789
- `ERROR: proxy container command '${proxyContainerCommand}' exited with code ${code}, signal ${signal}`,
790
- );
791
- if (sandboxProcess?.pid) {
792
- process.kill(-sandboxProcess.pid, 'SIGTERM');
793
- }
794
- process.exit(1);
795
- });
796
- console.log('waiting for proxy to start ...');
797
- await execAsync(
798
- `until timeout 0.25 curl -s http://localhost:8877; do sleep 0.25; done`,
799
- );
800
- // connect proxy container to sandbox network
801
- // (workaround for older versions of docker that don't support multiple --network args)
802
- await execAsync(
803
- `${config.command} network connect ${SANDBOX_NETWORK_NAME} ${SANDBOX_PROXY_NAME}`,
804
- );
805
- }
806
-
807
- // spawn child and let it inherit stdio
808
- sandboxProcess = spawn(config.command, args, {
809
- stdio: 'inherit',
810
- });
811
-
812
- sandboxProcess.on('error', (err) => {
813
- console.error('Sandbox process error:', err);
814
- });
815
-
816
- await new Promise<void>((resolve) => {
817
- sandboxProcess?.on('close', (code, signal) => {
818
- if (code !== 0) {
819
- console.log(
820
- `Sandbox process exited with code: ${code}, signal: ${signal}`,
821
- );
822
- }
823
- resolve();
824
- });
825
- });
826
- } finally {
827
- patcher.cleanup();
828
- }
829
- }
830
-
831
- // Helper functions to ensure sandbox image is present
832
- async function imageExists(sandbox: string, image: string): Promise<boolean> {
833
- return new Promise((resolve) => {
834
- const args = ['images', '-q', image];
835
- const checkProcess = spawn(sandbox, args);
836
-
837
- let stdoutData = '';
838
- if (checkProcess.stdout) {
839
- checkProcess.stdout.on('data', (data) => {
840
- stdoutData += data.toString();
841
- });
842
- }
843
-
844
- checkProcess.on('error', (err) => {
845
- console.warn(
846
- `Failed to start '${sandbox}' command for image check: ${err.message}`,
847
- );
848
- resolve(false);
849
- });
850
-
851
- checkProcess.on('close', (code) => {
852
- // Non-zero code might indicate docker daemon not running, etc.
853
- // The primary success indicator is non-empty stdoutData.
854
- if (code !== 0) {
855
- // console.warn(`'${sandbox} images -q ${image}' exited with code ${code}.`);
856
- }
857
- resolve(stdoutData.trim() !== '');
858
- });
859
- });
860
- }
861
-
862
- async function pullImage(sandbox: string, image: string): Promise<boolean> {
863
- console.info(`Attempting to pull image ${image} using ${sandbox}...`);
864
- return new Promise((resolve) => {
865
- const args = ['pull', image];
866
- const pullProcess = spawn(sandbox, args, { stdio: 'pipe' });
867
-
868
- let stderrData = '';
869
-
870
- const onStdoutData = (data: Buffer) => {
871
- console.info(data.toString().trim()); // Show pull progress
872
- };
873
-
874
- const onStderrData = (data: Buffer) => {
875
- stderrData += data.toString();
876
- console.error(data.toString().trim()); // Show pull errors/info from the command itself
877
- };
878
-
879
- const onError = (err: Error) => {
880
- console.warn(
881
- `Failed to start '${sandbox} pull ${image}' command: ${err.message}`,
882
- );
883
- cleanup();
884
- resolve(false);
885
- };
886
-
887
- const onClose = (code: number | null) => {
888
- if (code === 0) {
889
- console.info(`Successfully pulled image ${image}.`);
890
- cleanup();
891
- resolve(true);
892
- } else {
893
- console.warn(
894
- `Failed to pull image ${image}. '${sandbox} pull ${image}' exited with code ${code}.`,
895
- );
896
- if (stderrData.trim()) {
897
- // Details already printed by the stderr listener above
898
- }
899
- cleanup();
900
- resolve(false);
901
- }
902
- };
903
-
904
- const cleanup = () => {
905
- if (pullProcess.stdout) {
906
- pullProcess.stdout.removeListener('data', onStdoutData);
907
- }
908
- if (pullProcess.stderr) {
909
- pullProcess.stderr.removeListener('data', onStderrData);
910
- }
911
- pullProcess.removeListener('error', onError);
912
- pullProcess.removeListener('close', onClose);
913
- if (pullProcess.connected) {
914
- pullProcess.disconnect();
915
- }
916
- };
917
-
918
- if (pullProcess.stdout) {
919
- pullProcess.stdout.on('data', onStdoutData);
920
- }
921
- if (pullProcess.stderr) {
922
- pullProcess.stderr.on('data', onStderrData);
923
- }
924
- pullProcess.on('error', onError);
925
- pullProcess.on('close', onClose);
926
- });
927
- }
928
-
929
- async function ensureSandboxImageIsPresent(
930
- sandbox: string,
931
- image: string,
932
- ): Promise<boolean> {
933
- console.info(`Checking for sandbox image: ${image}`);
934
- if (await imageExists(sandbox, image)) {
935
- console.info(`Sandbox image ${image} found locally.`);
936
- return true;
937
- }
938
-
939
- console.info(`Sandbox image ${image} not found locally.`);
940
- if (image === LOCAL_DEV_SANDBOX_IMAGE_NAME) {
941
- // user needs to build the image themselves
942
- return false;
943
- }
944
-
945
- if (await pullImage(sandbox, image)) {
946
- // After attempting to pull, check again to be certain
947
- if (await imageExists(sandbox, image)) {
948
- console.info(`Sandbox image ${image} is now available after pulling.`);
949
- return true;
950
- } else {
951
- console.warn(
952
- `Sandbox image ${image} still not found after a pull attempt. This might indicate an issue with the image name or registry, or the pull command reported success but failed to make the image available.`,
953
- );
954
- return false;
955
- }
956
- }
957
-
958
- console.error(
959
- `Failed to obtain sandbox image ${image} after check and pull attempt.`,
960
- );
961
- return false; // Pull command failed or image still not present
962
- }