@friggframework/devtools 2.0.0--canary.522.cbd3d5a.0 → 2.0.0--canary.517.21b69ac.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 (247) hide show
  1. package/.eslintrc.json +3 -0
  2. package/CHANGELOG.md +132 -0
  3. package/frigg-cli/README.md +1 -1
  4. package/frigg-cli/__tests__/unit/commands/doctor.test.js +2 -0
  5. package/frigg-cli/__tests__/unit/commands/install.test.js +17 -21
  6. package/frigg-cli/doctor-command/index.js +16 -17
  7. package/frigg-cli/index.js +6 -21
  8. package/frigg-cli/index.test.js +1 -7
  9. package/frigg-cli/init-command/backend-first-handler.js +42 -124
  10. package/frigg-cli/init-command/index.js +1 -2
  11. package/frigg-cli/init-command/template-handler.js +3 -13
  12. package/frigg-cli/install-command/backend-js.js +3 -3
  13. package/frigg-cli/install-command/environment-variables.js +19 -16
  14. package/frigg-cli/install-command/environment-variables.test.js +13 -12
  15. package/frigg-cli/install-command/index.js +9 -14
  16. package/frigg-cli/install-command/integration-file.js +3 -3
  17. package/frigg-cli/install-command/logger.js +12 -0
  18. package/frigg-cli/install-command/validate-package.js +9 -5
  19. package/frigg-cli/jest.config.js +1 -4
  20. package/frigg-cli/repair-command/index.js +128 -101
  21. package/frigg-cli/start-command/index.js +2 -246
  22. package/frigg-cli/ui-command/index.js +36 -58
  23. package/frigg-cli/utils/repo-detection.js +37 -85
  24. package/infrastructure/docs/iam-policy-templates.md +1 -1
  25. package/infrastructure/domains/networking/vpc-builder.test.js +4 -2
  26. package/infrastructure/domains/networking/vpc-resolver.test.js +1 -1
  27. package/infrastructure/domains/shared/cloudformation-discovery.test.js +7 -4
  28. package/infrastructure/domains/shared/resource-discovery.js +5 -5
  29. package/infrastructure/domains/shared/types/discovery-result.test.js +1 -1
  30. package/infrastructure/domains/shared/utilities/base-definition-factory.js +2 -25
  31. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +2 -2
  32. package/infrastructure/infrastructure-composer.test.js +2 -2
  33. package/layers/prisma/.build-complete +3 -0
  34. package/layers/prisma/nodejs/package.json +8 -0
  35. package/management-ui/.eslintrc.js +22 -0
  36. package/management-ui/README.md +109 -245
  37. package/management-ui/components.json +21 -0
  38. package/management-ui/docs/phase2-integration-guide.md +320 -0
  39. package/management-ui/index.html +13 -0
  40. package/management-ui/package.json +76 -0
  41. package/management-ui/packages/devtools/frigg-cli/ui-command/index.js +302 -0
  42. package/management-ui/postcss.config.js +6 -0
  43. package/management-ui/server/api/backend.js +256 -0
  44. package/management-ui/server/api/cli.js +315 -0
  45. package/management-ui/server/api/codegen.js +663 -0
  46. package/management-ui/server/api/connections.js +857 -0
  47. package/management-ui/server/api/discovery.js +185 -0
  48. package/management-ui/server/api/environment/index.js +1 -0
  49. package/management-ui/server/api/environment/router.js +378 -0
  50. package/management-ui/server/api/environment.js +328 -0
  51. package/management-ui/server/api/integrations.js +876 -0
  52. package/management-ui/server/api/logs.js +248 -0
  53. package/management-ui/server/api/monitoring.js +282 -0
  54. package/management-ui/server/api/open-ide.js +31 -0
  55. package/management-ui/server/api/project.js +1029 -0
  56. package/management-ui/server/api/users/sessions.js +371 -0
  57. package/management-ui/server/api/users/simulation.js +254 -0
  58. package/management-ui/server/api/users.js +362 -0
  59. package/management-ui/server/api-contract.md +275 -0
  60. package/management-ui/server/index.js +873 -0
  61. package/management-ui/server/middleware/errorHandler.js +93 -0
  62. package/management-ui/server/middleware/security.js +32 -0
  63. package/management-ui/server/processManager.js +296 -0
  64. package/management-ui/server/server.js +346 -0
  65. package/management-ui/server/services/aws-monitor.js +413 -0
  66. package/management-ui/server/services/npm-registry.js +347 -0
  67. package/management-ui/server/services/template-engine.js +538 -0
  68. package/management-ui/server/utils/cliIntegration.js +220 -0
  69. package/management-ui/server/utils/environment/auditLogger.js +471 -0
  70. package/management-ui/server/utils/environment/awsParameterStore.js +275 -0
  71. package/management-ui/server/utils/environment/encryption.js +278 -0
  72. package/management-ui/server/utils/environment/envFileManager.js +286 -0
  73. package/management-ui/server/utils/import-commonjs.js +28 -0
  74. package/management-ui/server/utils/response.js +83 -0
  75. package/management-ui/server/websocket/handler.js +325 -0
  76. package/management-ui/src/App.jsx +25 -0
  77. package/management-ui/src/assets/FriggLogo.svg +1 -0
  78. package/management-ui/src/components/AppRouter.jsx +65 -0
  79. package/management-ui/src/components/Button.jsx +70 -0
  80. package/management-ui/src/components/Card.jsx +97 -0
  81. package/management-ui/src/components/EnvironmentCompare.jsx +400 -0
  82. package/management-ui/src/components/EnvironmentEditor.jsx +372 -0
  83. package/management-ui/src/components/EnvironmentImportExport.jsx +469 -0
  84. package/management-ui/src/components/EnvironmentSchema.jsx +491 -0
  85. package/management-ui/src/components/EnvironmentSecurity.jsx +463 -0
  86. package/management-ui/src/components/ErrorBoundary.jsx +73 -0
  87. package/management-ui/src/components/IntegrationCard.jsx +481 -0
  88. package/management-ui/src/components/IntegrationCardEnhanced.jsx +770 -0
  89. package/management-ui/src/components/IntegrationExplorer.jsx +379 -0
  90. package/management-ui/src/components/IntegrationStatus.jsx +336 -0
  91. package/management-ui/src/components/Layout.jsx +716 -0
  92. package/management-ui/src/components/LoadingSpinner.jsx +113 -0
  93. package/management-ui/src/components/RepositoryPicker.jsx +248 -0
  94. package/management-ui/src/components/SessionMonitor.jsx +350 -0
  95. package/management-ui/src/components/StatusBadge.jsx +208 -0
  96. package/management-ui/src/components/UserContextSwitcher.jsx +212 -0
  97. package/management-ui/src/components/UserSimulation.jsx +327 -0
  98. package/management-ui/src/components/Welcome.jsx +434 -0
  99. package/management-ui/src/components/codegen/APIEndpointGenerator.jsx +637 -0
  100. package/management-ui/src/components/codegen/APIModuleSelector.jsx +227 -0
  101. package/management-ui/src/components/codegen/CodeGenerationWizard.jsx +247 -0
  102. package/management-ui/src/components/codegen/CodePreviewEditor.jsx +316 -0
  103. package/management-ui/src/components/codegen/DynamicModuleForm.jsx +271 -0
  104. package/management-ui/src/components/codegen/FormBuilder.jsx +737 -0
  105. package/management-ui/src/components/codegen/IntegrationGenerator.jsx +855 -0
  106. package/management-ui/src/components/codegen/ProjectScaffoldWizard.jsx +797 -0
  107. package/management-ui/src/components/codegen/SchemaBuilder.jsx +303 -0
  108. package/management-ui/src/components/codegen/TemplateSelector.jsx +586 -0
  109. package/management-ui/src/components/codegen/index.js +10 -0
  110. package/management-ui/src/components/connections/ConnectionConfigForm.jsx +362 -0
  111. package/management-ui/src/components/connections/ConnectionHealthMonitor.jsx +182 -0
  112. package/management-ui/src/components/connections/ConnectionTester.jsx +200 -0
  113. package/management-ui/src/components/connections/EntityRelationshipMapper.jsx +292 -0
  114. package/management-ui/src/components/connections/OAuthFlow.jsx +204 -0
  115. package/management-ui/src/components/connections/index.js +5 -0
  116. package/management-ui/src/components/index.js +21 -0
  117. package/management-ui/src/components/monitoring/APIGatewayMetrics.jsx +222 -0
  118. package/management-ui/src/components/monitoring/LambdaMetrics.jsx +169 -0
  119. package/management-ui/src/components/monitoring/MetricsChart.jsx +197 -0
  120. package/management-ui/src/components/monitoring/MonitoringDashboard.jsx +393 -0
  121. package/management-ui/src/components/monitoring/SQSMetrics.jsx +246 -0
  122. package/management-ui/src/components/monitoring/index.js +6 -0
  123. package/management-ui/src/components/monitoring/monitoring.css +218 -0
  124. package/management-ui/src/components/theme-provider.jsx +52 -0
  125. package/management-ui/src/components/theme-toggle.jsx +39 -0
  126. package/management-ui/src/components/ui/badge.tsx +36 -0
  127. package/management-ui/src/components/ui/button.test.jsx +56 -0
  128. package/management-ui/src/components/ui/button.tsx +57 -0
  129. package/management-ui/src/components/ui/card.tsx +76 -0
  130. package/management-ui/src/components/ui/dropdown-menu.tsx +199 -0
  131. package/management-ui/src/components/ui/select.tsx +157 -0
  132. package/management-ui/src/components/ui/skeleton.jsx +15 -0
  133. package/management-ui/src/hooks/useFrigg.jsx +387 -0
  134. package/management-ui/src/hooks/useSocket.jsx +58 -0
  135. package/management-ui/src/index.css +193 -0
  136. package/management-ui/src/lib/utils.ts +6 -0
  137. package/management-ui/src/main.jsx +10 -0
  138. package/management-ui/src/pages/CodeGeneration.jsx +14 -0
  139. package/management-ui/src/pages/Connections.jsx +252 -0
  140. package/management-ui/src/pages/ConnectionsEnhanced.jsx +633 -0
  141. package/management-ui/src/pages/Dashboard.jsx +311 -0
  142. package/management-ui/src/pages/Environment.jsx +314 -0
  143. package/management-ui/src/pages/IntegrationConfigure.jsx +669 -0
  144. package/management-ui/src/pages/IntegrationDiscovery.jsx +567 -0
  145. package/management-ui/src/pages/IntegrationTest.jsx +742 -0
  146. package/management-ui/src/pages/Integrations.jsx +253 -0
  147. package/management-ui/src/pages/Monitoring.jsx +17 -0
  148. package/management-ui/src/pages/Simulation.jsx +155 -0
  149. package/management-ui/src/pages/Users.jsx +492 -0
  150. package/management-ui/src/services/api.js +41 -0
  151. package/management-ui/src/services/apiModuleService.js +193 -0
  152. package/management-ui/src/services/websocket-handlers.js +120 -0
  153. package/management-ui/src/test/api/project.test.js +273 -0
  154. package/management-ui/src/test/components/Welcome.test.jsx +378 -0
  155. package/management-ui/src/test/mocks/server.js +178 -0
  156. package/management-ui/src/test/setup.js +61 -0
  157. package/management-ui/src/test/utils/test-utils.jsx +134 -0
  158. package/management-ui/src/utils/repository.js +98 -0
  159. package/management-ui/src/utils/repository.test.js +118 -0
  160. package/management-ui/src/workflows/phase2-integration-workflows.js +884 -0
  161. package/management-ui/tailwind.config.js +63 -0
  162. package/management-ui/tsconfig.json +37 -0
  163. package/management-ui/tsconfig.node.json +10 -0
  164. package/management-ui/vite.config.js +26 -0
  165. package/management-ui/vitest.config.js +38 -0
  166. package/package.json +7 -17
  167. package/frigg-cli/__tests__/application/use-cases/AddApiModuleToIntegrationUseCase.test.js +0 -326
  168. package/frigg-cli/__tests__/application/use-cases/CreateApiModuleUseCase.test.js +0 -337
  169. package/frigg-cli/__tests__/domain/entities/ApiModule.test.js +0 -373
  170. package/frigg-cli/__tests__/domain/entities/AppDefinition.test.js +0 -313
  171. package/frigg-cli/__tests__/domain/services/IntegrationValidator.test.js +0 -269
  172. package/frigg-cli/__tests__/domain/value-objects/IntegrationName.test.js +0 -82
  173. package/frigg-cli/__tests__/infrastructure/adapters/IntegrationJsUpdater.test.js +0 -408
  174. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemApiModuleRepository.test.js +0 -583
  175. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemAppDefinitionRepository.test.js +0 -314
  176. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemIntegrationRepository.test.js +0 -430
  177. package/frigg-cli/__tests__/unit/commands/init.test.js +0 -406
  178. package/frigg-cli/__tests__/unit/commands/repair.test.js +0 -275
  179. package/frigg-cli/__tests__/unit/start-command/application/RunPreflightChecksUseCase.test.js +0 -411
  180. package/frigg-cli/__tests__/unit/start-command/infrastructure/DatabaseAdapter.test.js +0 -405
  181. package/frigg-cli/__tests__/unit/start-command/infrastructure/DockerAdapter.test.js +0 -496
  182. package/frigg-cli/__tests__/unit/start-command/presentation/InteractivePromptAdapter.test.js +0 -474
  183. package/frigg-cli/__tests__/unit/utils/output.test.js +0 -196
  184. package/frigg-cli/application/use-cases/AddApiModuleToIntegrationUseCase.js +0 -93
  185. package/frigg-cli/application/use-cases/CreateApiModuleUseCase.js +0 -93
  186. package/frigg-cli/application/use-cases/CreateIntegrationUseCase.js +0 -103
  187. package/frigg-cli/container.js +0 -172
  188. package/frigg-cli/docs/OUTPUT_MIGRATION_GUIDE.md +0 -286
  189. package/frigg-cli/domain/entities/ApiModule.js +0 -272
  190. package/frigg-cli/domain/entities/AppDefinition.js +0 -227
  191. package/frigg-cli/domain/entities/Integration.js +0 -198
  192. package/frigg-cli/domain/exceptions/DomainException.js +0 -24
  193. package/frigg-cli/domain/ports/IApiModuleRepository.js +0 -53
  194. package/frigg-cli/domain/ports/IAppDefinitionRepository.js +0 -43
  195. package/frigg-cli/domain/ports/IIntegrationRepository.js +0 -61
  196. package/frigg-cli/domain/services/IntegrationValidator.js +0 -185
  197. package/frigg-cli/domain/value-objects/IntegrationId.js +0 -42
  198. package/frigg-cli/domain/value-objects/IntegrationName.js +0 -60
  199. package/frigg-cli/domain/value-objects/SemanticVersion.js +0 -70
  200. package/frigg-cli/infrastructure/UnitOfWork.js +0 -46
  201. package/frigg-cli/infrastructure/adapters/BackendJsUpdater.js +0 -197
  202. package/frigg-cli/infrastructure/adapters/FileSystemAdapter.js +0 -224
  203. package/frigg-cli/infrastructure/adapters/IntegrationJsUpdater.js +0 -249
  204. package/frigg-cli/infrastructure/adapters/SchemaValidator.js +0 -92
  205. package/frigg-cli/infrastructure/repositories/FileSystemApiModuleRepository.js +0 -373
  206. package/frigg-cli/infrastructure/repositories/FileSystemAppDefinitionRepository.js +0 -116
  207. package/frigg-cli/infrastructure/repositories/FileSystemIntegrationRepository.js +0 -277
  208. package/frigg-cli/package-lock.json +0 -16226
  209. package/frigg-cli/start-command/application/RunPreflightChecksUseCase.js +0 -376
  210. package/frigg-cli/start-command/infrastructure/DatabaseAdapter.js +0 -591
  211. package/frigg-cli/start-command/infrastructure/DockerAdapter.js +0 -306
  212. package/frigg-cli/start-command/presentation/InteractivePromptAdapter.js +0 -329
  213. package/frigg-cli/templates/backend/.env.example +0 -62
  214. package/frigg-cli/templates/backend/.eslintrc.json +0 -12
  215. package/frigg-cli/templates/backend/.prettierrc +0 -6
  216. package/frigg-cli/templates/backend/docker-compose.yml +0 -22
  217. package/frigg-cli/templates/backend/index.js +0 -96
  218. package/frigg-cli/templates/backend/infrastructure.js +0 -12
  219. package/frigg-cli/templates/backend/jest.config.js +0 -17
  220. package/frigg-cli/templates/backend/package.json +0 -50
  221. package/frigg-cli/templates/backend/src/api-modules/.gitkeep +0 -10
  222. package/frigg-cli/templates/backend/src/base/.gitkeep +0 -7
  223. package/frigg-cli/templates/backend/src/integrations/.gitkeep +0 -10
  224. package/frigg-cli/templates/backend/src/integrations/ExampleIntegration.js +0 -65
  225. package/frigg-cli/templates/backend/src/utils/.gitkeep +0 -7
  226. package/frigg-cli/templates/backend/test/setup.js +0 -30
  227. package/frigg-cli/templates/backend/ui-extensions/.gitkeep +0 -0
  228. package/frigg-cli/templates/backend/ui-extensions/README.md +0 -77
  229. package/frigg-cli/utils/__tests__/repo-detection.test.js +0 -436
  230. package/frigg-cli/utils/output.js +0 -382
  231. package/frigg-cli/validate-command/__tests__/adapters/validate-command.test.js +0 -205
  232. package/frigg-cli/validate-command/__tests__/application/validate-app-use-case.test.js +0 -104
  233. package/frigg-cli/validate-command/__tests__/domain/fix-suggestion.test.js +0 -153
  234. package/frigg-cli/validate-command/__tests__/domain/validation-error.test.js +0 -162
  235. package/frigg-cli/validate-command/__tests__/domain/validation-result.test.js +0 -152
  236. package/frigg-cli/validate-command/__tests__/infrastructure/api-module-validator.test.js +0 -332
  237. package/frigg-cli/validate-command/__tests__/infrastructure/app-definition-validator.test.js +0 -191
  238. package/frigg-cli/validate-command/__tests__/infrastructure/integration-class-validator.test.js +0 -146
  239. package/frigg-cli/validate-command/__tests__/infrastructure/template-validation.test.js +0 -155
  240. package/frigg-cli/validate-command/adapters/cli/validate-command.js +0 -199
  241. package/frigg-cli/validate-command/application/use-cases/validate-app-use-case.js +0 -35
  242. package/frigg-cli/validate-command/domain/entities/validation-result.js +0 -74
  243. package/frigg-cli/validate-command/domain/value-objects/fix-suggestion.js +0 -74
  244. package/frigg-cli/validate-command/domain/value-objects/validation-error.js +0 -68
  245. package/frigg-cli/validate-command/infrastructure/validators/api-module-validator.js +0 -181
  246. package/frigg-cli/validate-command/infrastructure/validators/app-definition-validator.js +0 -128
  247. package/frigg-cli/validate-command/infrastructure/validators/integration-class-validator.js +0 -113
@@ -0,0 +1,328 @@
1
+ import express from 'express'
2
+ import path from 'path'
3
+ import fs from 'fs-extra'
4
+ import { createStandardResponse, createErrorResponse, ERROR_CODES, asyncHandler } from '../utils/response.js'
5
+ import { wsHandler } from '../websocket/handler.js'
6
+
7
+ const router = express.Router();
8
+
9
+ // Helper to get .env file path
10
+ async function getEnvFilePath() {
11
+ const possiblePaths = [
12
+ path.join(process.cwd(), '../../../backend/.env'),
13
+ path.join(process.cwd(), '../../backend/.env'),
14
+ path.join(process.cwd(), 'backend/.env'),
15
+ path.join(process.cwd(), '.env')
16
+ ];
17
+
18
+ for (const envPath of possiblePaths) {
19
+ if (await fs.pathExists(envPath)) {
20
+ return envPath;
21
+ }
22
+ }
23
+
24
+ // If no .env exists, create one in the most likely location
25
+ const defaultPath = possiblePaths[0];
26
+ await fs.ensureFile(defaultPath);
27
+ return defaultPath;
28
+ }
29
+
30
+ // Parse .env file content
31
+ function parseEnvFile(content) {
32
+ const lines = content.split('\n');
33
+ const variables = [];
34
+
35
+ lines.forEach((line, index) => {
36
+ const trimmedLine = line.trim();
37
+
38
+ // Skip empty lines and comments
39
+ if (!trimmedLine || trimmedLine.startsWith('#')) {
40
+ return;
41
+ }
42
+
43
+ const equalIndex = trimmedLine.indexOf('=');
44
+ if (equalIndex > 0) {
45
+ const key = trimmedLine.substring(0, equalIndex).trim();
46
+ const value = trimmedLine.substring(equalIndex + 1).trim();
47
+
48
+ variables.push({
49
+ key,
50
+ value: value.replace(/^["']|["']$/g, ''), // Remove quotes
51
+ line: index + 1
52
+ });
53
+ }
54
+ });
55
+
56
+ return variables;
57
+ }
58
+
59
+ // Build .env file content from variables
60
+ function buildEnvContent(variables) {
61
+ return variables
62
+ .map(({ key, value }) => {
63
+ // Add quotes if value contains spaces or special characters
64
+ if (value && (value.includes(' ') || value.includes('#'))) {
65
+ return `${key}="${value}"`;
66
+ }
67
+ return `${key}=${value}`;
68
+ })
69
+ .join('\n');
70
+ }
71
+
72
+ // Get all environment variables
73
+ router.get('/', async (req, res) => {
74
+ try {
75
+ const envPath = await getEnvFilePath();
76
+ const content = await fs.readFile(envPath, 'utf8');
77
+ const variables = parseEnvFile(content);
78
+
79
+ // Mask sensitive values
80
+ const maskedVariables = variables.map(variable => {
81
+ const isSensitive = [
82
+ 'KEY', 'SECRET', 'PASSWORD', 'TOKEN', 'API', 'PRIVATE'
83
+ ].some(keyword => variable.key.toUpperCase().includes(keyword));
84
+
85
+ return {
86
+ ...variable,
87
+ value: isSensitive ? '***' : variable.value,
88
+ masked: isSensitive
89
+ };
90
+ });
91
+
92
+ res.json({
93
+ variables: maskedVariables,
94
+ path: envPath
95
+ });
96
+ } catch (error) {
97
+ res.status(500).json({
98
+ error: error.message,
99
+ details: 'Failed to read environment variables'
100
+ });
101
+ }
102
+ });
103
+
104
+ // Get specific environment variable
105
+ router.get('/:key', async (req, res) => {
106
+ const { key } = req.params;
107
+
108
+ try {
109
+ const envPath = await getEnvFilePath();
110
+ const content = await fs.readFile(envPath, 'utf8');
111
+ const variables = parseEnvFile(content);
112
+
113
+ const variable = variables.find(v => v.key === key);
114
+
115
+ if (!variable) {
116
+ return res.status(404).json({
117
+ error: `Environment variable ${key} not found`
118
+ });
119
+ }
120
+
121
+ // Check if it's sensitive
122
+ const isSensitive = [
123
+ 'KEY', 'SECRET', 'PASSWORD', 'TOKEN', 'API', 'PRIVATE'
124
+ ].some(keyword => key.toUpperCase().includes(keyword));
125
+
126
+ res.json({
127
+ key: variable.key,
128
+ value: isSensitive ? '***' : variable.value,
129
+ masked: isSensitive
130
+ });
131
+ } catch (error) {
132
+ res.status(500).json({
133
+ error: error.message,
134
+ details: 'Failed to read environment variable'
135
+ });
136
+ }
137
+ });
138
+
139
+ // Set environment variable
140
+ router.post('/', async (req, res) => {
141
+ const { key, value } = req.body;
142
+
143
+ if (!key) {
144
+ return res.status(400).json({
145
+ error: 'Key is required'
146
+ });
147
+ }
148
+
149
+ try {
150
+ const envPath = await getEnvFilePath();
151
+ const content = await fs.readFile(envPath, 'utf8');
152
+ const variables = parseEnvFile(content);
153
+
154
+ // Check if variable exists
155
+ const existingIndex = variables.findIndex(v => v.key === key);
156
+
157
+ if (existingIndex >= 0) {
158
+ variables[existingIndex].value = value;
159
+ } else {
160
+ variables.push({ key, value });
161
+ }
162
+
163
+ // Write back to file
164
+ const newContent = buildEnvContent(variables);
165
+ await fs.writeFile(envPath, newContent);
166
+
167
+ // Broadcast update
168
+ wsHandler.broadcast('env-update', {
169
+ action: existingIndex >= 0 ? 'updated' : 'created',
170
+ key,
171
+ timestamp: new Date().toISOString()
172
+ });
173
+
174
+ res.json({
175
+ status: 'success',
176
+ message: `Environment variable ${key} ${existingIndex >= 0 ? 'updated' : 'created'}`,
177
+ key,
178
+ value: value.includes('SECRET') || value.includes('KEY') ? '***' : value
179
+ });
180
+ } catch (error) {
181
+ res.status(500).json({
182
+ error: error.message,
183
+ details: 'Failed to set environment variable'
184
+ });
185
+ }
186
+ });
187
+
188
+ // Update multiple environment variables
189
+ router.put('/batch', async (req, res) => {
190
+ const { variables } = req.body;
191
+
192
+ if (!Array.isArray(variables)) {
193
+ return res.status(400).json({
194
+ error: 'Variables must be an array'
195
+ });
196
+ }
197
+
198
+ try {
199
+ const envPath = await getEnvFilePath();
200
+ const content = await fs.readFile(envPath, 'utf8');
201
+ const existingVariables = parseEnvFile(content);
202
+
203
+ // Update or add variables
204
+ variables.forEach(({ key, value }) => {
205
+ const existingIndex = existingVariables.findIndex(v => v.key === key);
206
+
207
+ if (existingIndex >= 0) {
208
+ existingVariables[existingIndex].value = value;
209
+ } else {
210
+ existingVariables.push({ key, value });
211
+ }
212
+ });
213
+
214
+ // Write back to file
215
+ const newContent = buildEnvContent(existingVariables);
216
+ await fs.writeFile(envPath, newContent);
217
+
218
+ // Broadcast update
219
+ wsHandler.broadcast('env-update', {
220
+ action: 'batch-update',
221
+ count: variables.length,
222
+ timestamp: new Date().toISOString()
223
+ });
224
+
225
+ res.json({
226
+ status: 'success',
227
+ message: `Updated ${variables.length} environment variables`,
228
+ updated: variables.length
229
+ });
230
+ } catch (error) {
231
+ res.status(500).json({
232
+ error: error.message,
233
+ details: 'Failed to update environment variables'
234
+ });
235
+ }
236
+ });
237
+
238
+ // Delete environment variable
239
+ router.delete('/:key', async (req, res) => {
240
+ const { key } = req.params;
241
+
242
+ try {
243
+ const envPath = await getEnvFilePath();
244
+ const content = await fs.readFile(envPath, 'utf8');
245
+ const variables = parseEnvFile(content);
246
+
247
+ const filteredVariables = variables.filter(v => v.key !== key);
248
+
249
+ if (filteredVariables.length === variables.length) {
250
+ return res.status(404).json({
251
+ error: `Environment variable ${key} not found`
252
+ });
253
+ }
254
+
255
+ // Write back to file
256
+ const newContent = buildEnvContent(filteredVariables);
257
+ await fs.writeFile(envPath, newContent);
258
+
259
+ // Broadcast update
260
+ wsHandler.broadcast('env-update', {
261
+ action: 'deleted',
262
+ key,
263
+ timestamp: new Date().toISOString()
264
+ });
265
+
266
+ res.json({
267
+ status: 'success',
268
+ message: `Environment variable ${key} deleted`
269
+ });
270
+ } catch (error) {
271
+ res.status(500).json({
272
+ error: error.message,
273
+ details: 'Failed to delete environment variable'
274
+ });
275
+ }
276
+ });
277
+
278
+ // Validate environment variables
279
+ router.post('/validate', async (req, res) => {
280
+ try {
281
+ const envPath = await getEnvFilePath();
282
+ const content = await fs.readFile(envPath, 'utf8');
283
+ const variables = parseEnvFile(content);
284
+
285
+ const issues = [];
286
+
287
+ // Check for required variables
288
+ const requiredVars = [
289
+ 'DATABASE_URL',
290
+ 'JWT_SECRET',
291
+ 'NODE_ENV'
292
+ ];
293
+
294
+ requiredVars.forEach(reqVar => {
295
+ if (!variables.find(v => v.key === reqVar)) {
296
+ issues.push({
297
+ type: 'missing',
298
+ key: reqVar,
299
+ message: `Required variable ${reqVar} is missing`
300
+ });
301
+ }
302
+ });
303
+
304
+ // Check for empty values
305
+ variables.forEach(variable => {
306
+ if (!variable.value || variable.value.trim() === '') {
307
+ issues.push({
308
+ type: 'empty',
309
+ key: variable.key,
310
+ message: `Variable ${variable.key} has an empty value`
311
+ });
312
+ }
313
+ });
314
+
315
+ res.json({
316
+ valid: issues.length === 0,
317
+ issues,
318
+ totalVariables: variables.length
319
+ });
320
+ } catch (error) {
321
+ res.status(500).json({
322
+ error: error.message,
323
+ details: 'Failed to validate environment variables'
324
+ });
325
+ }
326
+ });
327
+
328
+ export default router