codex-linux 1.0.0

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 (367) hide show
  1. package/.claude/settings.local.json +10 -0
  2. package/.eslintrc.json +27 -0
  3. package/.github/workflows/ci.yml +156 -0
  4. package/.huskyrc +7 -0
  5. package/.lintstagedrc +13 -0
  6. package/.prettierrc +12 -0
  7. package/CLAUDE.md +163 -0
  8. package/DESIGN_SUPERIOR.md +73 -0
  9. package/Dockerfile +64 -0
  10. package/INSTALLATION.md +152 -0
  11. package/LICENSE +21 -0
  12. package/README.md +245 -0
  13. package/assets/skills/code-review/instructions.md +102 -0
  14. package/assets/skills/code-review/skill.yaml +15 -0
  15. package/assets/skills/refactoring/instructions.md +149 -0
  16. package/assets/skills/refactoring/skill.yaml +15 -0
  17. package/assets/skills/testing/skill.yaml +15 -0
  18. package/commitlint.config.js +23 -0
  19. package/dist/main/DatabaseManager.js +763 -0
  20. package/dist/main/DatabaseManager.js.map +1 -0
  21. package/dist/main/SettingsManager.js +61 -0
  22. package/dist/main/SettingsManager.js.map +1 -0
  23. package/dist/main/agents/AgentOrchestrator.js +787 -0
  24. package/dist/main/agents/AgentOrchestrator.js.map +1 -0
  25. package/dist/main/agents/AgentSDK.js +219 -0
  26. package/dist/main/agents/AgentSDK.js.map +1 -0
  27. package/dist/main/agents/AgentTools.js +348 -0
  28. package/dist/main/agents/AgentTools.js.map +1 -0
  29. package/dist/main/agents/CodeIndex.js +233 -0
  30. package/dist/main/agents/CodeIndex.js.map +1 -0
  31. package/dist/main/agents/EmbeddingService.js +80 -0
  32. package/dist/main/agents/EmbeddingService.js.map +1 -0
  33. package/dist/main/agents/NativeToolCalling.js +206 -0
  34. package/dist/main/agents/NativeToolCalling.js.map +1 -0
  35. package/dist/main/api/APIServer.js +278 -0
  36. package/dist/main/api/APIServer.js.map +1 -0
  37. package/dist/main/api/RateLimiter.js +138 -0
  38. package/dist/main/api/RateLimiter.js.map +1 -0
  39. package/dist/main/api/WebSocketManager.js +300 -0
  40. package/dist/main/api/WebSocketManager.js.map +1 -0
  41. package/dist/main/assistant/ContextOptimizer.js +192 -0
  42. package/dist/main/assistant/ContextOptimizer.js.map +1 -0
  43. package/dist/main/assistant/PredictedOutputManager.js +172 -0
  44. package/dist/main/assistant/PredictedOutputManager.js.map +1 -0
  45. package/dist/main/assistant/PromptCacheManager.js +193 -0
  46. package/dist/main/assistant/PromptCacheManager.js.map +1 -0
  47. package/dist/main/assistant/PromptOptimizer.js +626 -0
  48. package/dist/main/assistant/PromptOptimizer.js.map +1 -0
  49. package/dist/main/assistant/SmartCodeAssistant.js +224 -0
  50. package/dist/main/assistant/SmartCodeAssistant.js.map +1 -0
  51. package/dist/main/auth/SessionManager.js +300 -0
  52. package/dist/main/auth/SessionManager.js.map +1 -0
  53. package/dist/main/automations/AdvancedWebhookSystem.js +212 -0
  54. package/dist/main/automations/AdvancedWebhookSystem.js.map +1 -0
  55. package/dist/main/automations/AutomationScheduler.js +269 -0
  56. package/dist/main/automations/AutomationScheduler.js.map +1 -0
  57. package/dist/main/automations/BatchProcessingSystem.js +159 -0
  58. package/dist/main/automations/BatchProcessingSystem.js.map +1 -0
  59. package/dist/main/automations/BrowserAutomationManager.js +195 -0
  60. package/dist/main/automations/BrowserAutomationManager.js.map +1 -0
  61. package/dist/main/automations/GitHubActionsManager.js +129 -0
  62. package/dist/main/automations/GitHubActionsManager.js.map +1 -0
  63. package/dist/main/automations/GitLabCIManager.js +122 -0
  64. package/dist/main/automations/GitLabCIManager.js.map +1 -0
  65. package/dist/main/automations/PriorityQueueManager.js +240 -0
  66. package/dist/main/automations/PriorityQueueManager.js.map +1 -0
  67. package/dist/main/background/BackgroundModeManager.js +117 -0
  68. package/dist/main/background/BackgroundModeManager.js.map +1 -0
  69. package/dist/main/backup/BackupManager.js +254 -0
  70. package/dist/main/backup/BackupManager.js.map +1 -0
  71. package/dist/main/backup/MigrationManager.js +114 -0
  72. package/dist/main/backup/MigrationManager.js.map +1 -0
  73. package/dist/main/commands/SlashCommandManager.js +399 -0
  74. package/dist/main/commands/SlashCommandManager.js.map +1 -0
  75. package/dist/main/config/ClaudeMdParser.js +519 -0
  76. package/dist/main/config/ClaudeMdParser.js.map +1 -0
  77. package/dist/main/config/CustomizationManager.js +381 -0
  78. package/dist/main/config/CustomizationManager.js.map +1 -0
  79. package/dist/main/config/LaunchConfigManager.js +211 -0
  80. package/dist/main/config/LaunchConfigManager.js.map +1 -0
  81. package/dist/main/config/SettingsManager.js +166 -0
  82. package/dist/main/config/SettingsManager.js.map +1 -0
  83. package/dist/main/connectors/ConnectorManager.js +151 -0
  84. package/dist/main/connectors/ConnectorManager.js.map +1 -0
  85. package/dist/main/connectors/DatabaseConnector.js +222 -0
  86. package/dist/main/connectors/DatabaseConnector.js.map +1 -0
  87. package/dist/main/cowork/CoworkManager.js +324 -0
  88. package/dist/main/cowork/CoworkManager.js.map +1 -0
  89. package/dist/main/evals/AgentEvalFramework.js +538 -0
  90. package/dist/main/evals/AgentEvalFramework.js.map +1 -0
  91. package/dist/main/evals/GraderManager.js +285 -0
  92. package/dist/main/evals/GraderManager.js.map +1 -0
  93. package/dist/main/git/GitWorktreeManager.js +214 -0
  94. package/dist/main/git/GitWorktreeManager.js.map +1 -0
  95. package/dist/main/github/GitHubPRMonitor.js +244 -0
  96. package/dist/main/github/GitHubPRMonitor.js.map +1 -0
  97. package/dist/main/ide/ContinueInManager.js +181 -0
  98. package/dist/main/ide/ContinueInManager.js.map +1 -0
  99. package/dist/main/ide/IDEIntegration.js +277 -0
  100. package/dist/main/ide/IDEIntegration.js.map +1 -0
  101. package/dist/main/integrations/LinearManager.js +252 -0
  102. package/dist/main/integrations/LinearManager.js.map +1 -0
  103. package/dist/main/integrations/SlackBotManager.js +247 -0
  104. package/dist/main/integrations/SlackBotManager.js.map +1 -0
  105. package/dist/main/lsp/LSPManager.js +394 -0
  106. package/dist/main/lsp/LSPManager.js.map +1 -0
  107. package/dist/main/main.js +1087 -0
  108. package/dist/main/main.js.map +1 -0
  109. package/dist/main/mcp/MCPConfigurationManager.js +281 -0
  110. package/dist/main/mcp/MCPConfigurationManager.js.map +1 -0
  111. package/dist/main/mcp/MCPManager.js +710 -0
  112. package/dist/main/mcp/MCPManager.js.map +1 -0
  113. package/dist/main/mcp/MCPRegistry.js +272 -0
  114. package/dist/main/mcp/MCPRegistry.js.map +1 -0
  115. package/dist/main/monitoring/ErrorRecoveryManager.js +268 -0
  116. package/dist/main/monitoring/ErrorRecoveryManager.js.map +1 -0
  117. package/dist/main/monitoring/ErrorTracker.js +57 -0
  118. package/dist/main/monitoring/ErrorTracker.js.map +1 -0
  119. package/dist/main/monitoring/MetricsCollector.js +155 -0
  120. package/dist/main/monitoring/MetricsCollector.js.map +1 -0
  121. package/dist/main/monitoring/TraceGradingSystem.js +148 -0
  122. package/dist/main/monitoring/TraceGradingSystem.js.map +1 -0
  123. package/dist/main/notifications/NotificationManager.js +67 -0
  124. package/dist/main/notifications/NotificationManager.js.map +1 -0
  125. package/dist/main/pair/AIPairProgramming.js +200 -0
  126. package/dist/main/pair/AIPairProgramming.js.map +1 -0
  127. package/dist/main/plugins/PluginManager.js +222 -0
  128. package/dist/main/plugins/PluginManager.js.map +1 -0
  129. package/dist/main/plugins/PluginMarketplace.js +237 -0
  130. package/dist/main/plugins/PluginMarketplace.js.map +1 -0
  131. package/dist/main/preload.js +189 -0
  132. package/dist/main/preload.js.map +1 -0
  133. package/dist/main/preview/PreviewSessionManager.js +170 -0
  134. package/dist/main/preview/PreviewSessionManager.js.map +1 -0
  135. package/dist/main/providers/AIProviderManager.js +327 -0
  136. package/dist/main/providers/AIProviderManager.js.map +1 -0
  137. package/dist/main/providers/FineTuningManager.js +276 -0
  138. package/dist/main/providers/FineTuningManager.js.map +1 -0
  139. package/dist/main/providers/FreeModelsProvider.js +1104 -0
  140. package/dist/main/providers/FreeModelsProvider.js.map +1 -0
  141. package/dist/main/realtime/RealtimeManager.js +116 -0
  142. package/dist/main/realtime/RealtimeManager.js.map +1 -0
  143. package/dist/main/remote/CloudEnvironmentManager.js +232 -0
  144. package/dist/main/remote/CloudEnvironmentManager.js.map +1 -0
  145. package/dist/main/remote/RemoteSessionManager.js +255 -0
  146. package/dist/main/remote/RemoteSessionManager.js.map +1 -0
  147. package/dist/main/search/DeepResearchManager.js +335 -0
  148. package/dist/main/search/DeepResearchManager.js.map +1 -0
  149. package/dist/main/search/WebSearchIntegration.js +147 -0
  150. package/dist/main/search/WebSearchIntegration.js.map +1 -0
  151. package/dist/main/security/AdminConsoleManager.js +223 -0
  152. package/dist/main/security/AdminConsoleManager.js.map +1 -0
  153. package/dist/main/security/AuditLogger.js +136 -0
  154. package/dist/main/security/AuditLogger.js.map +1 -0
  155. package/dist/main/security/PermissionManager.js +144 -0
  156. package/dist/main/security/PermissionManager.js.map +1 -0
  157. package/dist/main/security/SSOManager.js +173 -0
  158. package/dist/main/security/SSOManager.js.map +1 -0
  159. package/dist/main/security/SecurityManager.js +152 -0
  160. package/dist/main/security/SecurityManager.js.map +1 -0
  161. package/dist/main/skills/SkillsManager.js +223 -0
  162. package/dist/main/skills/SkillsManager.js.map +1 -0
  163. package/dist/main/ssh/SSHManager.js +65 -0
  164. package/dist/main/ssh/SSHManager.js.map +1 -0
  165. package/dist/main/streaming/StreamingManager.js +225 -0
  166. package/dist/main/streaming/StreamingManager.js.map +1 -0
  167. package/dist/main/sync/CloudSyncManager.js +422 -0
  168. package/dist/main/sync/CloudSyncManager.js.map +1 -0
  169. package/dist/main/types.js +28 -0
  170. package/dist/main/types.js.map +1 -0
  171. package/dist/main/verification/AutoVerifyManager.js +235 -0
  172. package/dist/main/verification/AutoVerifyManager.js.map +1 -0
  173. package/dist/main/vision/ComputerUseManager.js +376 -0
  174. package/dist/main/vision/ComputerUseManager.js.map +1 -0
  175. package/dist/main/vision/ImageVideoGenerationManager.js +401 -0
  176. package/dist/main/vision/ImageVideoGenerationManager.js.map +1 -0
  177. package/dist/main/vision/VisionManager.js +172 -0
  178. package/dist/main/vision/VisionManager.js.map +1 -0
  179. package/dist/renderer/assets/main-DJlZQBCA.js +304 -0
  180. package/dist/renderer/assets/main-N33ZXEr8.css +1 -0
  181. package/dist/renderer/index.html +21 -0
  182. package/dist/renderer/manifest.json +42 -0
  183. package/dist/renderer/sw.ts +109 -0
  184. package/dist/shared/types.js +35 -0
  185. package/dist/shared/types.js.map +1 -0
  186. package/docker-compose.yml +65 -0
  187. package/docs/API.md +307 -0
  188. package/docs/USER_GUIDE.md +476 -0
  189. package/examples/plugins/sample-plugin/package.json +41 -0
  190. package/examples/plugins/sample-plugin/src/index.ts +75 -0
  191. package/index.html +20 -0
  192. package/jest.config.js +39 -0
  193. package/package.json +180 -0
  194. package/packages/cli/package.json +29 -0
  195. package/packages/cli/src/commands/agents.ts +199 -0
  196. package/packages/cli/src/commands/tasks.ts +61 -0
  197. package/packages/cli/src/index.ts +91 -0
  198. package/packages/cli/src/utils/api.ts +45 -0
  199. package/packages/cli/src/utils/config.ts +61 -0
  200. package/packages/npm-installer/bin/codex-linux +126 -0
  201. package/packages/npm-installer/lib/download.js +273 -0
  202. package/packages/npm-installer/package.json +42 -0
  203. package/packages/vscode-extension/package.json +167 -0
  204. package/packages/vscode-extension/src/api.ts +68 -0
  205. package/packages/vscode-extension/src/extension.ts +161 -0
  206. package/packages/vscode-extension/src/panels/chatPanel.ts +265 -0
  207. package/packages/vscode-extension/src/panels/createAgentPanel.ts +227 -0
  208. package/packages/vscode-extension/src/providers/agentsProvider.ts +80 -0
  209. package/postcss.config.js +6 -0
  210. package/public/manifest.json +42 -0
  211. package/public/sw.ts +109 -0
  212. package/scripts/install-dev.sh +103 -0
  213. package/scripts/install.sh +275 -0
  214. package/src/main/DatabaseManager.ts +950 -0
  215. package/src/main/SettingsManager.ts +63 -0
  216. package/src/main/agents/AgentOrchestrator.ts +930 -0
  217. package/src/main/agents/AgentSDK.ts +269 -0
  218. package/src/main/agents/AgentTools.ts +380 -0
  219. package/src/main/agents/CodeIndex.ts +240 -0
  220. package/src/main/agents/EmbeddingService.ts +88 -0
  221. package/src/main/agents/NativeToolCalling.ts +245 -0
  222. package/src/main/api/APIServer.ts +316 -0
  223. package/src/main/api/RateLimiter.ts +165 -0
  224. package/src/main/api/WebSocketManager.ts +398 -0
  225. package/src/main/assistant/ContextOptimizer.ts +214 -0
  226. package/src/main/assistant/PredictedOutputManager.ts +265 -0
  227. package/src/main/assistant/PromptCacheManager.ts +280 -0
  228. package/src/main/assistant/PromptOptimizer.ts +746 -0
  229. package/src/main/assistant/SmartCodeAssistant.ts +234 -0
  230. package/src/main/auth/SessionManager.ts +415 -0
  231. package/src/main/automations/AdvancedWebhookSystem.ts +281 -0
  232. package/src/main/automations/AutomationScheduler.ts +272 -0
  233. package/src/main/automations/BatchProcessingSystem.ts +207 -0
  234. package/src/main/automations/BrowserAutomationManager.ts +203 -0
  235. package/src/main/automations/GitHubActionsManager.ts +151 -0
  236. package/src/main/automations/GitLabCIManager.ts +206 -0
  237. package/src/main/automations/PriorityQueueManager.ts +328 -0
  238. package/src/main/background/BackgroundModeManager.ts +130 -0
  239. package/src/main/backup/BackupManager.ts +287 -0
  240. package/src/main/backup/MigrationManager.ts +132 -0
  241. package/src/main/commands/SlashCommandManager.ts +407 -0
  242. package/src/main/config/ClaudeMdParser.ts +539 -0
  243. package/src/main/config/CustomizationManager.ts +493 -0
  244. package/src/main/config/LaunchConfigManager.ts +212 -0
  245. package/src/main/config/SettingsManager.ts +163 -0
  246. package/src/main/connectors/ConnectorManager.ts +175 -0
  247. package/src/main/connectors/DatabaseConnector.ts +212 -0
  248. package/src/main/cowork/CoworkManager.ts +431 -0
  249. package/src/main/evals/AgentEvalFramework.ts +665 -0
  250. package/src/main/evals/GraderManager.ts +417 -0
  251. package/src/main/git/GitWorktreeManager.ts +211 -0
  252. package/src/main/github/GitHubPRMonitor.ts +317 -0
  253. package/src/main/ide/ContinueInManager.ts +180 -0
  254. package/src/main/ide/IDEIntegration.ts +288 -0
  255. package/src/main/integrations/LinearManager.ts +327 -0
  256. package/src/main/integrations/SlackBotManager.ts +312 -0
  257. package/src/main/lsp/LSPManager.ts +445 -0
  258. package/src/main/main.ts +1221 -0
  259. package/src/main/mcp/MCPConfigurationManager.ts +281 -0
  260. package/src/main/mcp/MCPManager.ts +799 -0
  261. package/src/main/mcp/MCPRegistry.ts +273 -0
  262. package/src/main/monitoring/ErrorRecoveryManager.ts +359 -0
  263. package/src/main/monitoring/ErrorTracker.ts +60 -0
  264. package/src/main/monitoring/MetricsCollector.ts +196 -0
  265. package/src/main/monitoring/TraceGradingSystem.ts +196 -0
  266. package/src/main/notifications/NotificationManager.ts +96 -0
  267. package/src/main/pair/AIPairProgramming.ts +290 -0
  268. package/src/main/plugins/PluginManager.ts +266 -0
  269. package/src/main/plugins/PluginMarketplace.ts +318 -0
  270. package/src/main/preload.ts +215 -0
  271. package/src/main/preview/PreviewSessionManager.ts +186 -0
  272. package/src/main/providers/AIProviderManager.ts +394 -0
  273. package/src/main/providers/FineTuningManager.ts +390 -0
  274. package/src/main/providers/FreeModelsProvider.ts +1156 -0
  275. package/src/main/realtime/RealtimeManager.ts +147 -0
  276. package/src/main/remote/CloudEnvironmentManager.ts +253 -0
  277. package/src/main/remote/RemoteSessionManager.ts +323 -0
  278. package/src/main/search/DeepResearchManager.ts +458 -0
  279. package/src/main/search/WebSearchIntegration.ts +203 -0
  280. package/src/main/security/AdminConsoleManager.ts +244 -0
  281. package/src/main/security/AuditLogger.ts +143 -0
  282. package/src/main/security/PermissionManager.ts +184 -0
  283. package/src/main/security/SSOManager.ts +241 -0
  284. package/src/main/security/SecurityManager.ts +139 -0
  285. package/src/main/skills/SkillsManager.ts +218 -0
  286. package/src/main/ssh/SSHManager.ts +86 -0
  287. package/src/main/streaming/StreamingManager.ts +306 -0
  288. package/src/main/sync/CloudSyncManager.ts +532 -0
  289. package/src/main/verification/AutoVerifyManager.ts +285 -0
  290. package/src/main/vision/ComputerUseManager.ts +475 -0
  291. package/src/main/vision/ImageVideoGenerationManager.ts +526 -0
  292. package/src/main/vision/VisionManager.ts +186 -0
  293. package/src/renderer/App.tsx +314 -0
  294. package/src/renderer/components/AdvancedSettingsPanel.tsx +225 -0
  295. package/src/renderer/components/AgentPanel.tsx +760 -0
  296. package/src/renderer/components/AppPreview.tsx +220 -0
  297. package/src/renderer/components/AuditTrailPanel.tsx +148 -0
  298. package/src/renderer/components/AutomationPanel.tsx +220 -0
  299. package/src/renderer/components/ChatInterface.tsx +595 -0
  300. package/src/renderer/components/ChatTab.tsx +296 -0
  301. package/src/renderer/components/CodeEditor.tsx +257 -0
  302. package/src/renderer/components/CodeReviewPanel.tsx +256 -0
  303. package/src/renderer/components/CodeWorkspace.tsx +192 -0
  304. package/src/renderer/components/CodebaseDashboard.tsx +295 -0
  305. package/src/renderer/components/ComputerUsePanel.tsx +262 -0
  306. package/src/renderer/components/ConnectorsPanel.tsx +471 -0
  307. package/src/renderer/components/ContextMenu.tsx +155 -0
  308. package/src/renderer/components/ContextUsageDisplay.tsx +248 -0
  309. package/src/renderer/components/CoworkPanel.tsx +415 -0
  310. package/src/renderer/components/DiffViewer.tsx +452 -0
  311. package/src/renderer/components/ErrorBoundary.tsx +273 -0
  312. package/src/renderer/components/ExtendedThinkingToggle.tsx +244 -0
  313. package/src/renderer/components/FileAttachments.tsx +247 -0
  314. package/src/renderer/components/FileExplorer.tsx +242 -0
  315. package/src/renderer/components/FileExplorerPanel.tsx +302 -0
  316. package/src/renderer/components/GitPanel.tsx +154 -0
  317. package/src/renderer/components/Header.tsx +113 -0
  318. package/src/renderer/components/MCPPanel.tsx +326 -0
  319. package/src/renderer/components/MentionAutocomplete.tsx +239 -0
  320. package/src/renderer/components/PermissionPanel.tsx +159 -0
  321. package/src/renderer/components/PermissionSelector.tsx +203 -0
  322. package/src/renderer/components/PluginMarketplace.tsx +325 -0
  323. package/src/renderer/components/PromptOptimizerPanel.tsx +399 -0
  324. package/src/renderer/components/SearchPanel.tsx +173 -0
  325. package/src/renderer/components/SearchReplace.tsx +284 -0
  326. package/src/renderer/components/SessionSidebar.tsx +367 -0
  327. package/src/renderer/components/SettingsPanel.tsx +426 -0
  328. package/src/renderer/components/Sidebar.tsx +100 -0
  329. package/src/renderer/components/SkillsPanel.tsx +245 -0
  330. package/src/renderer/components/SplitPane.tsx +173 -0
  331. package/src/renderer/components/Terminal.tsx +190 -0
  332. package/src/renderer/components/VoiceCommand.tsx +129 -0
  333. package/src/renderer/components/WorktreePanel.tsx +163 -0
  334. package/src/renderer/components/ui/AriaComponents.tsx +193 -0
  335. package/src/renderer/components/ui/Button.tsx +68 -0
  336. package/src/renderer/components/ui/Card.tsx +102 -0
  337. package/src/renderer/components/ui/Input.tsx +44 -0
  338. package/src/renderer/components/ui/Skeleton.tsx +55 -0
  339. package/src/renderer/components/ui/VirtualList.tsx +196 -0
  340. package/src/renderer/i18n/I18nProvider.tsx +101 -0
  341. package/src/renderer/i18n/de.ts +161 -0
  342. package/src/renderer/i18n/en.ts +163 -0
  343. package/src/renderer/i18n/es.ts +161 -0
  344. package/src/renderer/i18n/fr.ts +161 -0
  345. package/src/renderer/i18n/index.ts +44 -0
  346. package/src/renderer/index.css +129 -0
  347. package/src/renderer/lib/accessibility.tsx +287 -0
  348. package/src/renderer/lib/hooks.ts +304 -0
  349. package/src/renderer/lib/utils.ts +6 -0
  350. package/src/renderer/main.tsx +25 -0
  351. package/src/renderer/styles/minimalist.css +539 -0
  352. package/src/renderer/sw.ts +180 -0
  353. package/src/renderer/types.d.ts +138 -0
  354. package/src/shared/types.ts +813 -0
  355. package/supabase/schema.sql +234 -0
  356. package/tailwind.config.js +78 -0
  357. package/tests/e2e/package.json +15 -0
  358. package/tests/e2e/playwright.config.ts +31 -0
  359. package/tests/e2e/specs/app.spec.ts +194 -0
  360. package/tests/setup.ts +99 -0
  361. package/tests/unit/AgentOrchestrator.test.ts +274 -0
  362. package/tests/unit/DatabaseManager.test.ts +262 -0
  363. package/tests/unit/GitWorktreeManager.test.ts +150 -0
  364. package/tests/unit/SecurityManager.test.ts +110 -0
  365. package/tsconfig.main.json +22 -0
  366. package/tsconfig.renderer.json +27 -0
  367. package/vite.config.ts +28 -0
@@ -0,0 +1,493 @@
1
+ import { EventEmitter } from 'events';
2
+ import { v4 as uuidv4 } from 'uuid';
3
+ import log from 'electron-log';
4
+
5
+ export type ThemeMode = 'light' | 'dark' | 'system';
6
+ export type LayoutMode = 'default' | 'compact' | 'focus' | 'custom';
7
+ export type FontSize = 'small' | 'medium' | 'large';
8
+
9
+ export interface Theme {
10
+ id: string;
11
+ name: string;
12
+ mode: 'light' | 'dark';
13
+ colors: ThemeColors;
14
+ isCustom: boolean;
15
+ }
16
+
17
+ export interface ThemeColors {
18
+ primary: string;
19
+ secondary: string;
20
+ accent: string;
21
+ background: string;
22
+ foreground: string;
23
+ muted: string;
24
+ border: string;
25
+ success: string;
26
+ warning: string;
27
+ error: string;
28
+ }
29
+
30
+ export interface Layout {
31
+ id: string;
32
+ name: string;
33
+ sidebarPosition: 'left' | 'right' | 'hidden';
34
+ sidebarWidth: number;
35
+ showHeader: boolean;
36
+ showStatusBar: boolean;
37
+ showTabs: boolean;
38
+ terminalPosition: 'bottom' | 'right' | 'hidden';
39
+ }
40
+
41
+ export interface KeyboardShortcut {
42
+ id: string;
43
+ action: string;
44
+ keys: string;
45
+ description: string;
46
+ category: string;
47
+ }
48
+
49
+ export interface UserPreferences {
50
+ theme: ThemeMode;
51
+ customThemeId?: string;
52
+ layout: LayoutMode;
53
+ customLayout?: Partial<Layout>;
54
+ fontSize: FontSize;
55
+ fontFamily: string;
56
+ lineHeight: number;
57
+ tabSize: number;
58
+ wordWrap: boolean;
59
+ minimap: boolean;
60
+ lineNumbers: boolean;
61
+ cursorBlinking: boolean;
62
+ cursorStyle: 'line' | 'block' | 'underline';
63
+ smoothScrolling: boolean;
64
+ mouseWheelZoom: boolean;
65
+ }
66
+
67
+ export interface CustomizationConfig {
68
+ enableCustomThemes: boolean;
69
+ enableCustomLayouts: boolean;
70
+ enableKeyboardShortcuts: boolean;
71
+ }
72
+
73
+ const DEFAULT_THEMES: Theme[] = [
74
+ {
75
+ id: 'light',
76
+ name: 'Light',
77
+ mode: 'light',
78
+ isCustom: false,
79
+ colors: {
80
+ primary: '#3b82f6',
81
+ secondary: '#64748b',
82
+ accent: '#8b5cf6',
83
+ background: '#ffffff',
84
+ foreground: '#0f172a',
85
+ muted: '#f1f5f9',
86
+ border: '#e2e8f0',
87
+ success: '#22c55e',
88
+ warning: '#f59e0b',
89
+ error: '#ef4444',
90
+ },
91
+ },
92
+ {
93
+ id: 'dark',
94
+ name: 'Dark',
95
+ mode: 'dark',
96
+ isCustom: false,
97
+ colors: {
98
+ primary: '#3b82f6',
99
+ secondary: '#94a3b8',
100
+ accent: '#8b5cf6',
101
+ background: '#0f172a',
102
+ foreground: '#f8fafc',
103
+ muted: '#1e293b',
104
+ border: '#334155',
105
+ success: '#22c55e',
106
+ warning: '#f59e0b',
107
+ error: '#ef4444',
108
+ },
109
+ },
110
+ {
111
+ id: 'midnight',
112
+ name: 'Midnight',
113
+ mode: 'dark',
114
+ isCustom: false,
115
+ colors: {
116
+ primary: '#6366f1',
117
+ secondary: '#818cf8',
118
+ accent: '#a78bfa',
119
+ background: '#020617',
120
+ foreground: '#e2e8f0',
121
+ muted: '#0f172a',
122
+ border: '#1e293b',
123
+ success: '#10b981',
124
+ warning: '#fbbf24',
125
+ error: '#f43f5e',
126
+ },
127
+ },
128
+ {
129
+ id: 'dracula',
130
+ name: 'Dracula',
131
+ mode: 'dark',
132
+ isCustom: false,
133
+ colors: {
134
+ primary: '#bd93f9',
135
+ secondary: '#6272a4',
136
+ accent: '#ff79c6',
137
+ background: '#282a36',
138
+ foreground: '#f8f8f2',
139
+ muted: '#44475a',
140
+ border: '#6272a4',
141
+ success: '#50fa7b',
142
+ warning: '#f1fa8c',
143
+ error: '#ff5555',
144
+ },
145
+ },
146
+ ];
147
+
148
+ const DEFAULT_LAYOUTS: Layout[] = [
149
+ {
150
+ id: 'default',
151
+ name: 'Default',
152
+ sidebarPosition: 'left',
153
+ sidebarWidth: 280,
154
+ showHeader: true,
155
+ showStatusBar: true,
156
+ showTabs: true,
157
+ terminalPosition: 'bottom',
158
+ },
159
+ {
160
+ id: 'compact',
161
+ name: 'Compact',
162
+ sidebarPosition: 'left',
163
+ sidebarWidth: 220,
164
+ showHeader: false,
165
+ showStatusBar: true,
166
+ showTabs: false,
167
+ terminalPosition: 'bottom',
168
+ },
169
+ {
170
+ id: 'focus',
171
+ name: 'Focus',
172
+ sidebarPosition: 'hidden',
173
+ sidebarWidth: 0,
174
+ showHeader: false,
175
+ showStatusBar: false,
176
+ showTabs: false,
177
+ terminalPosition: 'hidden',
178
+ },
179
+ ];
180
+
181
+ const DEFAULT_SHORTCUTS: KeyboardShortcut[] = [
182
+ { id: '1', action: 'files.new', keys: 'Ctrl+N', description: 'New file', category: 'File' },
183
+ { id: '2', action: 'files.open', keys: 'Ctrl+O', description: 'Open file', category: 'File' },
184
+ { id: '3', action: 'files.save', keys: 'Ctrl+S', description: 'Save file', category: 'File' },
185
+ { id: '4', action: 'files.saveAll', keys: 'Ctrl+Shift+S', description: 'Save all files', category: 'File' },
186
+ { id: '5', action: 'files.close', keys: 'Ctrl+W', description: 'Close file', category: 'File' },
187
+ { id: '6', action: 'edit.undo', keys: 'Ctrl+Z', description: 'Undo', category: 'Edit' },
188
+ { id: '7', action: 'edit.redo', keys: 'Ctrl+Y', description: 'Redo', category: 'Edit' },
189
+ { id: '8', action: 'edit.cut', keys: 'Ctrl+X', description: 'Cut', category: 'Edit' },
190
+ { id: '9', action: 'edit.copy', keys: 'Ctrl+C', description: 'Copy', category: 'Edit' },
191
+ { id: '10', action: 'edit.paste', keys: 'Ctrl+V', description: 'Paste', category: 'Edit' },
192
+ { id: '11', action: 'edit.find', keys: 'Ctrl+F', description: 'Find', category: 'Edit' },
193
+ { id: '12', action: 'edit.replace', keys: 'Ctrl+H', description: 'Find and replace', category: 'Edit' },
194
+ { id: '13', action: 'view.sidebar', keys: 'Ctrl+B', description: 'Toggle sidebar', category: 'View' },
195
+ { id: '14', action: 'view.terminal', keys: 'Ctrl+`', description: 'Toggle terminal', category: 'View' },
196
+ { id: '15', action: 'view.fullscreen', keys: 'F11', description: 'Toggle fullscreen', category: 'View' },
197
+ { id: '16', action: 'view.zoomIn', keys: 'Ctrl++', description: 'Zoom in', category: 'View' },
198
+ { id: '17', action: 'view.zoomOut', keys: 'Ctrl+-', description: 'Zoom out', category: 'View' },
199
+ { id: '18', action: 'agent.new', keys: 'Ctrl+Shift+N', description: 'New agent', category: 'Agent' },
200
+ { id: '19', action: 'agent.run', keys: 'Ctrl+Enter', description: 'Run agent', category: 'Agent' },
201
+ { id: '20', action: 'agent.stop', keys: 'Ctrl+Shift+.', description: 'Stop agent', category: 'Agent' },
202
+ ];
203
+
204
+ const DEFAULT_PREFERENCES: UserPreferences = {
205
+ theme: 'system',
206
+ layout: 'default',
207
+ fontSize: 'medium',
208
+ fontFamily: 'JetBrains Mono, Consolas, monospace',
209
+ lineHeight: 1.5,
210
+ tabSize: 2,
211
+ wordWrap: true,
212
+ minimap: true,
213
+ lineNumbers: true,
214
+ cursorBlinking: true,
215
+ cursorStyle: 'line',
216
+ smoothScrolling: true,
217
+ mouseWheelZoom: true,
218
+ };
219
+
220
+ export class CustomizationManager extends EventEmitter {
221
+ private themes: Map<string, Theme> = new Map();
222
+ private layouts: Map<string, Layout> = new Map();
223
+ private shortcuts: Map<string, KeyboardShortcut> = new Map();
224
+ private preferences: UserPreferences;
225
+ private config: CustomizationConfig;
226
+
227
+ constructor(config?: Partial<CustomizationConfig>) {
228
+ super();
229
+ this.config = {
230
+ enableCustomThemes: config?.enableCustomThemes ?? true,
231
+ enableCustomLayouts: config?.enableCustomLayouts ?? true,
232
+ enableKeyboardShortcuts: config?.enableKeyboardShortcuts ?? true,
233
+ };
234
+ this.preferences = { ...DEFAULT_PREFERENCES };
235
+ this.initializeDefaults();
236
+ }
237
+
238
+ private initializeDefaults(): void {
239
+ DEFAULT_THEMES.forEach((theme) => this.themes.set(theme.id, theme));
240
+ DEFAULT_LAYOUTS.forEach((layout) => this.layouts.set(layout.id, layout));
241
+ DEFAULT_SHORTCUTS.forEach((shortcut) => this.shortcuts.set(shortcut.action, shortcut));
242
+ log.info('CustomizationManager initialized with defaults');
243
+ }
244
+
245
+ configure(config: Partial<CustomizationConfig>): void {
246
+ this.config = { ...this.config, ...config };
247
+ }
248
+
249
+ getConfig(): CustomizationConfig {
250
+ return { ...this.config };
251
+ }
252
+
253
+ getPreferences(): UserPreferences {
254
+ return { ...this.preferences };
255
+ }
256
+
257
+ updatePreferences(updates: Partial<UserPreferences>): UserPreferences {
258
+ this.preferences = { ...this.preferences, ...updates };
259
+ this.emit('preferences:updated', this.preferences);
260
+ log.info('Preferences updated', updates);
261
+ return this.preferences;
262
+ }
263
+
264
+ setPreferences(preferences: UserPreferences): void {
265
+ this.preferences = preferences;
266
+ this.emit('preferences:updated', this.preferences);
267
+ }
268
+
269
+ resetPreferences(): UserPreferences {
270
+ this.preferences = { ...DEFAULT_PREFERENCES };
271
+ this.emit('preferences:reset', this.preferences);
272
+ return this.preferences;
273
+ }
274
+
275
+ getThemes(): Theme[] {
276
+ return Array.from(this.themes.values());
277
+ }
278
+
279
+ getTheme(id: string): Theme | undefined {
280
+ return this.themes.get(id);
281
+ }
282
+
283
+ getCurrentTheme(): Theme {
284
+ const { theme, customThemeId } = this.preferences;
285
+
286
+ if (theme === 'system') {
287
+ return this.themes.get('dark')!;
288
+ }
289
+
290
+ if (customThemeId) {
291
+ return this.themes.get(customThemeId) || this.themes.get('dark')!;
292
+ }
293
+
294
+ return this.themes.get(theme) || this.themes.get('dark')!;
295
+ }
296
+
297
+ createTheme(name: string, colors: ThemeColors, mode: 'light' | 'dark'): Theme {
298
+ if (!this.config.enableCustomThemes) {
299
+ throw new Error('Custom themes are disabled');
300
+ }
301
+
302
+ const theme: Theme = {
303
+ id: uuidv4(),
304
+ name,
305
+ mode,
306
+ colors,
307
+ isCustom: true,
308
+ };
309
+
310
+ this.themes.set(theme.id, theme);
311
+ this.emit('theme:created', theme);
312
+ log.info(`Custom theme created: ${name}`);
313
+ return theme;
314
+ }
315
+
316
+ updateTheme(id: string, updates: Partial<ThemeColors>): Theme | null {
317
+ const theme = this.themes.get(id);
318
+ if (!theme || !theme.isCustom) return null;
319
+
320
+ theme.colors = { ...theme.colors, ...updates };
321
+ this.emit('theme:updated', theme);
322
+ return theme;
323
+ }
324
+
325
+ deleteTheme(id: string): boolean {
326
+ const theme = this.themes.get(id);
327
+ if (!theme || !theme.isCustom) return false;
328
+
329
+ this.themes.delete(id);
330
+ this.emit('theme:deleted', { id });
331
+ return true;
332
+ }
333
+
334
+ getLayouts(): Layout[] {
335
+ return Array.from(this.layouts.values());
336
+ }
337
+
338
+ getLayout(id: string): Layout | undefined {
339
+ return this.layouts.get(id);
340
+ }
341
+
342
+ getCurrentLayout(): Layout {
343
+ const { layout, customLayout } = this.preferences;
344
+ const defaultLayout = this.layouts.get(layout);
345
+
346
+ if (customLayout && layout === 'custom') {
347
+ return { ...defaultLayout!, ...customLayout };
348
+ }
349
+
350
+ return defaultLayout || this.layouts.get('default')!;
351
+ }
352
+
353
+ createLayout(name: string, config: Omit<Layout, 'id' | 'name'>): Layout {
354
+ if (!this.config.enableCustomLayouts) {
355
+ throw new Error('Custom layouts are disabled');
356
+ }
357
+
358
+ const layout: Layout = {
359
+ id: uuidv4(),
360
+ name,
361
+ ...config,
362
+ };
363
+
364
+ this.layouts.set(layout.id, layout);
365
+ this.emit('layout:created', layout);
366
+ return layout;
367
+ }
368
+
369
+ updateLayout(id: string, updates: Partial<Layout>): Layout | null {
370
+ const layout = this.layouts.get(id);
371
+ if (!layout) return null;
372
+
373
+ Object.assign(layout, updates);
374
+ this.emit('layout:updated', layout);
375
+ return layout;
376
+ }
377
+
378
+ deleteLayout(id: string): boolean {
379
+ const layout = this.layouts.get(id);
380
+ if (!layout || layout.id === 'default') return false;
381
+
382
+ this.layouts.delete(id);
383
+ this.emit('layout:deleted', { id });
384
+ return true;
385
+ }
386
+
387
+ getShortcuts(): KeyboardShortcut[] {
388
+ return Array.from(this.shortcuts.values());
389
+ }
390
+
391
+ getShortcut(action: string): KeyboardShortcut | undefined {
392
+ return this.shortcuts.get(action);
393
+ }
394
+
395
+ getShortcutsByCategory(): Record<string, KeyboardShortcut[]> {
396
+ const byCategory: Record<string, KeyboardShortcut[]> = {};
397
+
398
+ this.shortcuts.forEach((shortcut) => {
399
+ if (!byCategory[shortcut.category]) {
400
+ byCategory[shortcut.category] = [];
401
+ }
402
+ byCategory[shortcut.category].push(shortcut);
403
+ });
404
+
405
+ return byCategory;
406
+ }
407
+
408
+ createShortcut(shortcut: Omit<KeyboardShortcut, 'id'>): KeyboardShortcut {
409
+ if (!this.config.enableKeyboardShortcuts) {
410
+ throw new Error('Keyboard shortcuts are disabled');
411
+ }
412
+
413
+ const newShortcut: KeyboardShortcut = {
414
+ id: uuidv4(),
415
+ ...shortcut,
416
+ };
417
+
418
+ this.shortcuts.set(newShortcut.action, newShortcut);
419
+ this.emit('shortcut:created', newShortcut);
420
+ return newShortcut;
421
+ }
422
+
423
+ updateShortcut(action: string, updates: Partial<KeyboardShortcut>): KeyboardShortcut | null {
424
+ const shortcut = this.shortcuts.get(action);
425
+ if (!shortcut) return null;
426
+
427
+ Object.assign(shortcut, updates);
428
+ this.emit('shortcut:updated', shortcut);
429
+ return shortcut;
430
+ }
431
+
432
+ deleteShortcut(action: string): boolean {
433
+ const shortcut = this.shortcuts.get(action);
434
+ if (!shortcut) return false;
435
+
436
+ this.shortcuts.delete(action);
437
+ this.emit('shortcut:deleted', { action });
438
+ return true;
439
+ }
440
+
441
+ resetShortcuts(): void {
442
+ this.shortcuts.clear();
443
+ DEFAULT_SHORTCUTS.forEach((shortcut) => this.shortcuts.set(shortcut.action, shortcut));
444
+ this.emit('shortcuts:reset');
445
+ }
446
+
447
+ exportPreferences(): string {
448
+ return JSON.stringify({
449
+ preferences: this.preferences,
450
+ customThemes: Array.from(this.themes.values()).filter((t) => t.isCustom),
451
+ customLayouts: Array.from(this.layouts.values()).filter((l) => l.id !== 'default' && l.id !== 'compact' && l.id !== 'focus'),
452
+ customShortcuts: Array.from(this.shortcuts.values()),
453
+ }, null, 2);
454
+ }
455
+
456
+ importPreferences(json: string): boolean {
457
+ try {
458
+ const data = JSON.parse(json);
459
+
460
+ if (data.preferences) {
461
+ this.preferences = { ...DEFAULT_PREFERENCES, ...data.preferences };
462
+ }
463
+
464
+ if (data.customThemes) {
465
+ data.customThemes.forEach((theme: Theme) => {
466
+ this.themes.set(theme.id, theme);
467
+ });
468
+ }
469
+
470
+ if (data.customLayouts) {
471
+ data.customLayouts.forEach((layout: Layout) => {
472
+ this.layouts.set(layout.id, layout);
473
+ });
474
+ }
475
+
476
+ this.emit('preferences:imported');
477
+ return true;
478
+ } catch (error) {
479
+ log.error('Failed to import preferences:', error);
480
+ return false;
481
+ }
482
+ }
483
+
484
+ cleanup(): void {
485
+ this.themes.clear();
486
+ this.layouts.clear();
487
+ this.shortcuts.clear();
488
+ this.removeAllListeners();
489
+ log.info('CustomizationManager cleaned up');
490
+ }
491
+ }
492
+
493
+ export default CustomizationManager;
@@ -0,0 +1,212 @@
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ import { EventEmitter } from 'events';
4
+ import log from 'electron-log';
5
+
6
+ export interface PreviewServerConfig {
7
+ name: string;
8
+ runtimeExecutable: string;
9
+ runtimeArgs: string[];
10
+ port: number;
11
+ cwd?: string;
12
+ env?: Record<string, string>;
13
+ autoPort: boolean;
14
+ program?: string;
15
+ args?: string[];
16
+ }
17
+
18
+ export interface LaunchConfig {
19
+ version: string;
20
+ autoVerify?: boolean;
21
+ configurations: PreviewServerConfig[];
22
+ }
23
+
24
+ export class LaunchConfigManager extends EventEmitter {
25
+ private configPath: string;
26
+ private config: LaunchConfig | null = null;
27
+ private projectRoot: string;
28
+
29
+ constructor(projectRoot: string) {
30
+ super();
31
+ this.projectRoot = projectRoot;
32
+ this.configPath = path.join(projectRoot, '.claude', 'launch.json');
33
+ }
34
+
35
+ async load(): Promise<LaunchConfig | null> {
36
+ try {
37
+ if (!fs.existsSync(this.configPath)) {
38
+ log.info('No launch.json found, will auto-detect');
39
+ return null;
40
+ }
41
+
42
+ const content = fs.readFileSync(this.configPath, 'utf-8');
43
+ this.config = JSON.parse(content) as LaunchConfig;
44
+
45
+ log.info(`Loaded launch config with ${this.config.configurations.length} server(s)`);
46
+ this.emit('config:loaded', this.config);
47
+
48
+ return this.config;
49
+ } catch (error) {
50
+ log.error('Failed to load launch.json:', error);
51
+ return null;
52
+ }
53
+ }
54
+
55
+ async save(config: LaunchConfig): Promise<void> {
56
+ try {
57
+ const dir = path.dirname(this.configPath);
58
+ if (!fs.existsSync(dir)) {
59
+ fs.mkdirSync(dir, { recursive: true });
60
+ }
61
+
62
+ fs.writeFileSync(this.configPath, JSON.stringify(config, null, 2));
63
+ this.config = config;
64
+
65
+ log.info('Saved launch.json');
66
+ this.emit('config:saved', config);
67
+ } catch (error) {
68
+ log.error('Failed to save launch.json:', error);
69
+ throw error;
70
+ }
71
+ }
72
+
73
+ async autoDetect(projectRoot: string): Promise<PreviewServerConfig[]> {
74
+ const servers: PreviewServerConfig[] = [];
75
+
76
+ try {
77
+ const packageJsonPath = path.join(projectRoot, 'package.json');
78
+
79
+ if (fs.existsSync(packageJsonPath)) {
80
+ const pkg = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
81
+
82
+ if (pkg.scripts?.dev) {
83
+ servers.push({
84
+ name: 'dev',
85
+ runtimeExecutable: 'npm',
86
+ runtimeArgs: ['run', 'dev'],
87
+ port: 3000,
88
+ autoPort: true
89
+ });
90
+ }
91
+
92
+ if (pkg.scripts?.start && pkg.scripts.start !== pkg.scripts?.dev) {
93
+ servers.push({
94
+ name: 'start',
95
+ runtimeExecutable: 'npm',
96
+ runtimeArgs: ['run', 'start'],
97
+ port: 3000,
98
+ autoPort: true
99
+ });
100
+ }
101
+
102
+ if (pkg.scripts?.serve) {
103
+ servers.push({
104
+ name: 'serve',
105
+ runtimeExecutable: 'npm',
106
+ runtimeArgs: ['run', 'serve'],
107
+ port: 3000,
108
+ autoPort: true
109
+ });
110
+ }
111
+
112
+ if (pkg.scripts?.next && pkg.scripts.next.startsWith('next')) {
113
+ servers.push({
114
+ name: 'next',
115
+ runtimeExecutable: 'npm',
116
+ runtimeArgs: ['run', 'next', 'dev'],
117
+ port: 3000,
118
+ autoPort: true
119
+ });
120
+ }
121
+ }
122
+
123
+ const nextConfigPath = path.join(projectRoot, 'next.config.js');
124
+ if (fs.existsSync(nextConfigPath)) {
125
+ const existing = servers.find(s => s.name === 'next');
126
+ if (existing) {
127
+ existing.port = 3000;
128
+ }
129
+ }
130
+
131
+ const viteConfigPath = path.join(projectRoot, 'vite.config.ts');
132
+ if (fs.existsSync(viteConfigPath)) {
133
+ const existing = servers.find(s => s.name === 'dev');
134
+ if (existing) {
135
+ existing.port = 5173;
136
+ } else {
137
+ servers.push({
138
+ name: 'vite',
139
+ runtimeExecutable: 'npm',
140
+ runtimeArgs: ['run', 'dev'],
141
+ port: 5173,
142
+ autoPort: true
143
+ });
144
+ }
145
+ }
146
+
147
+ const webpackConfigPath = path.join(projectRoot, 'webpack.config.js');
148
+ if (fs.existsSync(webpackConfigPath)) {
149
+ servers.push({
150
+ name: 'webpack',
151
+ runtimeExecutable: 'npm',
152
+ runtimeArgs: ['run', 'dev'],
153
+ port: 8080,
154
+ autoPort: true
155
+ });
156
+ }
157
+
158
+ const expressPath = path.join(projectRoot, 'server.js');
159
+ const expressIndexPath = path.join(projectRoot, 'index.js');
160
+ if (fs.existsSync(expressPath) || fs.existsSync(expressIndexPath)) {
161
+ servers.push({
162
+ name: 'server',
163
+ runtimeExecutable: 'node',
164
+ runtimeArgs: [fs.existsSync(expressPath) ? 'server.js' : 'index.js'],
165
+ port: 3000,
166
+ autoPort: true
167
+ });
168
+ }
169
+
170
+ } catch (error) {
171
+ log.error('Failed to auto-detect servers:', error);
172
+ }
173
+
174
+ return servers;
175
+ }
176
+
177
+ getConfig(): LaunchConfig | null {
178
+ return this.config;
179
+ }
180
+
181
+ getServer(name: string): PreviewServerConfig | undefined {
182
+ return this.config?.configurations.find(c => c.name === name);
183
+ }
184
+
185
+ getAllServers(): PreviewServerConfig[] {
186
+ return this.config?.configurations || [];
187
+ }
188
+
189
+ addServer(server: PreviewServerConfig): void {
190
+ if (!this.config) {
191
+ this.config = {
192
+ version: '0.0.1',
193
+ configurations: []
194
+ };
195
+ }
196
+
197
+ const existing = this.config.configurations.findIndex(c => c.name === server.name);
198
+ if (existing >= 0) {
199
+ this.config.configurations[existing] = server;
200
+ } else {
201
+ this.config.configurations.push(server);
202
+ }
203
+ }
204
+
205
+ removeServer(name: string): void {
206
+ if (this.config) {
207
+ this.config.configurations = this.config.configurations.filter(c => c.name !== name);
208
+ }
209
+ }
210
+ }
211
+
212
+ export default LaunchConfigManager;