create-fluxstack 1.0.12 → 1.0.14

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 (215) hide show
  1. package/.env.example +29 -29
  2. package/app/client/README.md +69 -69
  3. package/app/client/index.html +14 -13
  4. package/app/client/src/App.tsx +157 -524
  5. package/app/client/src/components/ErrorBoundary.tsx +107 -0
  6. package/app/client/src/components/ErrorDisplay.css +365 -0
  7. package/app/client/src/components/ErrorDisplay.tsx +258 -0
  8. package/app/client/src/components/FluxStackConfig.tsx +1321 -0
  9. package/app/client/src/components/HybridLiveCounter.tsx +140 -0
  10. package/app/client/src/components/LiveClock.tsx +286 -0
  11. package/app/client/src/components/MainLayout.tsx +390 -0
  12. package/app/client/src/components/SidebarNavigation.tsx +391 -0
  13. package/app/client/src/components/StateDemo.tsx +178 -0
  14. package/app/client/src/components/SystemMonitor.tsx +1038 -0
  15. package/app/client/src/components/Teste.tsx +104 -0
  16. package/app/client/src/components/UserProfile.tsx +809 -0
  17. package/app/client/src/hooks/useAuth.ts +39 -0
  18. package/app/client/src/hooks/useNotifications.ts +56 -0
  19. package/app/client/src/lib/eden-api.ts +189 -53
  20. package/app/client/src/lib/errors.ts +340 -0
  21. package/app/client/src/lib/hooks/useErrorHandler.ts +258 -0
  22. package/app/client/src/lib/index.ts +45 -0
  23. package/app/client/src/main.tsx +3 -2
  24. package/app/client/src/pages/ApiDocs.tsx +182 -0
  25. package/app/client/src/pages/Demo.tsx +174 -0
  26. package/app/client/src/pages/HybridLive.tsx +263 -0
  27. package/app/client/src/pages/Overview.tsx +155 -0
  28. package/app/client/src/store/README.md +43 -0
  29. package/app/client/src/store/index.ts +16 -0
  30. package/app/client/src/store/slices/uiSlice.ts +151 -0
  31. package/app/client/src/store/slices/userSlice.ts +161 -0
  32. package/app/client/src/test/README.md +257 -0
  33. package/app/client/src/test/setup.ts +70 -0
  34. package/app/client/src/test/types.ts +12 -0
  35. package/app/client/src/vite-env.d.ts +1 -1
  36. package/app/client/tsconfig.app.json +44 -43
  37. package/app/client/tsconfig.json +7 -7
  38. package/app/client/tsconfig.node.json +25 -25
  39. package/app/client/zustand-setup.md +65 -0
  40. package/app/server/controllers/users.controller.ts +68 -68
  41. package/app/server/index.ts +9 -1
  42. package/app/server/live/CounterComponent.ts +191 -0
  43. package/app/server/live/FluxStackConfig.ts +529 -0
  44. package/app/server/live/LiveClockComponent.ts +214 -0
  45. package/app/server/live/SidebarNavigation.ts +156 -0
  46. package/app/server/live/SystemMonitor.ts +594 -0
  47. package/app/server/live/SystemMonitorIntegration.ts +151 -0
  48. package/app/server/live/TesteComponent.ts +87 -0
  49. package/app/server/live/UserProfileComponent.ts +135 -0
  50. package/app/server/live/register-components.ts +28 -0
  51. package/app/server/middleware/auth.ts +136 -0
  52. package/app/server/middleware/errorHandling.ts +250 -0
  53. package/app/server/middleware/index.ts +10 -0
  54. package/app/server/middleware/rateLimit.ts +193 -0
  55. package/app/server/middleware/requestLogging.ts +215 -0
  56. package/app/server/middleware/validation.ts +270 -0
  57. package/app/server/routes/index.ts +14 -2
  58. package/app/server/routes/upload.ts +92 -0
  59. package/app/server/routes/users.routes.ts +2 -9
  60. package/app/server/services/NotificationService.ts +302 -0
  61. package/app/server/services/UserService.ts +222 -0
  62. package/app/server/services/index.ts +46 -0
  63. package/core/cli/commands/plugin-deps.ts +263 -0
  64. package/core/cli/generators/README.md +339 -0
  65. package/core/cli/generators/component.ts +770 -0
  66. package/core/cli/generators/controller.ts +299 -0
  67. package/core/cli/generators/index.ts +144 -0
  68. package/core/cli/generators/interactive.ts +228 -0
  69. package/core/cli/generators/prompts.ts +83 -0
  70. package/core/cli/generators/route.ts +513 -0
  71. package/core/cli/generators/service.ts +465 -0
  72. package/core/cli/generators/template-engine.ts +154 -0
  73. package/core/cli/generators/types.ts +71 -0
  74. package/core/cli/generators/utils.ts +192 -0
  75. package/core/cli/index.ts +69 -0
  76. package/core/cli/plugin-discovery.ts +16 -85
  77. package/core/client/fluxstack.ts +17 -0
  78. package/core/client/hooks/index.ts +7 -0
  79. package/core/client/hooks/state-validator.ts +130 -0
  80. package/core/client/hooks/useAuth.ts +49 -0
  81. package/core/client/hooks/useChunkedUpload.ts +258 -0
  82. package/core/client/hooks/useHybridLiveComponent.ts +967 -0
  83. package/core/client/hooks/useWebSocket.ts +373 -0
  84. package/core/client/index.ts +47 -0
  85. package/core/client/state/createStore.ts +193 -0
  86. package/core/client/state/index.ts +15 -0
  87. package/core/config/env-dynamic.ts +1 -1
  88. package/core/config/env.ts +2 -1
  89. package/core/config/loader.ts +8 -32
  90. package/core/config/runtime-config.ts +3 -3
  91. package/core/config/schema.ts +84 -49
  92. package/core/framework/server.ts +30 -0
  93. package/core/index.ts +25 -0
  94. package/core/live/ComponentRegistry.ts +399 -0
  95. package/core/live/types.ts +164 -0
  96. package/core/plugins/built-in/live-components/commands/create-live-component.ts +1201 -0
  97. package/core/plugins/built-in/live-components/index.ts +27 -0
  98. package/core/plugins/built-in/logger/index.ts +1 -1
  99. package/core/plugins/built-in/monitoring/index.ts +1 -1
  100. package/core/plugins/built-in/static/index.ts +1 -1
  101. package/core/plugins/built-in/swagger/index.ts +1 -1
  102. package/core/plugins/built-in/vite/index.ts +1 -1
  103. package/core/plugins/dependency-manager.ts +384 -0
  104. package/core/plugins/index.ts +5 -1
  105. package/core/plugins/manager.ts +7 -3
  106. package/core/plugins/registry.ts +88 -10
  107. package/core/plugins/types.ts +11 -11
  108. package/core/server/framework.ts +43 -0
  109. package/core/server/index.ts +11 -1
  110. package/core/server/live/ComponentRegistry.ts +1017 -0
  111. package/core/server/live/FileUploadManager.ts +272 -0
  112. package/core/server/live/LiveComponentPerformanceMonitor.ts +930 -0
  113. package/core/server/live/SingleConnectionManager.ts +0 -0
  114. package/core/server/live/StateSignature.ts +644 -0
  115. package/core/server/live/WebSocketConnectionManager.ts +688 -0
  116. package/core/server/live/websocket-plugin.ts +435 -0
  117. package/core/server/middleware/errorHandling.ts +141 -0
  118. package/core/server/middleware/index.ts +16 -0
  119. package/core/server/plugins/static-files-plugin.ts +232 -0
  120. package/core/server/services/BaseService.ts +95 -0
  121. package/core/server/services/ServiceContainer.ts +144 -0
  122. package/core/server/services/index.ts +9 -0
  123. package/core/templates/create-project.ts +46 -2
  124. package/core/testing/index.ts +10 -0
  125. package/core/testing/setup.ts +74 -0
  126. package/core/types/build.ts +38 -14
  127. package/core/types/types.ts +319 -0
  128. package/core/utils/env-runtime.ts +7 -0
  129. package/core/utils/errors/handlers.ts +264 -39
  130. package/core/utils/errors/index.ts +528 -18
  131. package/core/utils/errors/middleware.ts +114 -0
  132. package/core/utils/logger/formatters.ts +222 -0
  133. package/core/utils/logger/index.ts +167 -48
  134. package/core/utils/logger/middleware.ts +253 -0
  135. package/core/utils/logger/performance.ts +384 -0
  136. package/core/utils/logger/transports.ts +365 -0
  137. package/create-fluxstack.ts +296 -296
  138. package/fluxstack.config.ts +17 -1
  139. package/package-template.json +66 -66
  140. package/package.json +31 -6
  141. package/public/README.md +16 -0
  142. package/vite.config.ts +29 -14
  143. package/.claude/settings.local.json +0 -74
  144. package/.github/workflows/ci-build-tests.yml +0 -480
  145. package/.github/workflows/dependency-management.yml +0 -324
  146. package/.github/workflows/release-validation.yml +0 -355
  147. package/.kiro/specs/fluxstack-architecture-optimization/design.md +0 -700
  148. package/.kiro/specs/fluxstack-architecture-optimization/requirements.md +0 -127
  149. package/.kiro/specs/fluxstack-architecture-optimization/tasks.md +0 -330
  150. package/CLAUDE.md +0 -200
  151. package/Dockerfile +0 -58
  152. package/Dockerfile.backend +0 -52
  153. package/Dockerfile.frontend +0 -54
  154. package/README-Docker.md +0 -85
  155. package/ai-context/00-QUICK-START.md +0 -86
  156. package/ai-context/README.md +0 -88
  157. package/ai-context/development/eden-treaty-guide.md +0 -362
  158. package/ai-context/development/patterns.md +0 -382
  159. package/ai-context/development/plugins-guide.md +0 -572
  160. package/ai-context/examples/crud-complete.md +0 -626
  161. package/ai-context/project/architecture.md +0 -399
  162. package/ai-context/project/overview.md +0 -213
  163. package/ai-context/recent-changes/eden-treaty-refactor.md +0 -281
  164. package/ai-context/recent-changes/type-inference-fix.md +0 -223
  165. package/ai-context/reference/environment-vars.md +0 -384
  166. package/ai-context/reference/troubleshooting.md +0 -407
  167. package/app/client/src/components/TestPage.tsx +0 -453
  168. package/bun.lock +0 -1063
  169. package/bunfig.toml +0 -16
  170. package/core/__tests__/integration.test.ts +0 -227
  171. package/core/build/index.ts +0 -186
  172. package/core/config/__tests__/config-loader.test.ts +0 -591
  173. package/core/config/__tests__/config-merger.test.ts +0 -657
  174. package/core/config/__tests__/env-converter.test.ts +0 -372
  175. package/core/config/__tests__/env-processor.test.ts +0 -431
  176. package/core/config/__tests__/env.test.ts +0 -452
  177. package/core/config/__tests__/integration.test.ts +0 -418
  178. package/core/config/__tests__/loader.test.ts +0 -331
  179. package/core/config/__tests__/schema.test.ts +0 -129
  180. package/core/config/__tests__/validator.test.ts +0 -318
  181. package/core/framework/__tests__/server.test.ts +0 -233
  182. package/core/plugins/__tests__/built-in.test.ts.disabled +0 -366
  183. package/core/plugins/__tests__/manager.test.ts +0 -398
  184. package/core/plugins/__tests__/monitoring.test.ts +0 -401
  185. package/core/plugins/__tests__/registry.test.ts +0 -335
  186. package/core/utils/__tests__/errors.test.ts +0 -139
  187. package/core/utils/__tests__/helpers.test.ts +0 -297
  188. package/core/utils/__tests__/logger.test.ts +0 -141
  189. package/create-test-app.ts +0 -156
  190. package/docker-compose.microservices.yml +0 -75
  191. package/docker-compose.simple.yml +0 -57
  192. package/docker-compose.yml +0 -71
  193. package/eslint.config.js +0 -23
  194. package/flux-cli.ts +0 -214
  195. package/nginx-lb.conf +0 -37
  196. package/publish.sh +0 -63
  197. package/run-clean.ts +0 -26
  198. package/run-env-tests.ts +0 -313
  199. package/tailwind.config.js +0 -34
  200. package/tests/__mocks__/api.ts +0 -56
  201. package/tests/fixtures/users.ts +0 -69
  202. package/tests/integration/api/users.routes.test.ts +0 -221
  203. package/tests/setup.ts +0 -29
  204. package/tests/unit/app/client/App-simple.test.tsx +0 -56
  205. package/tests/unit/app/client/App.test.tsx.skip +0 -237
  206. package/tests/unit/app/client/eden-api.test.ts +0 -186
  207. package/tests/unit/app/client/simple.test.tsx +0 -23
  208. package/tests/unit/app/controllers/users.controller.test.ts +0 -150
  209. package/tests/unit/core/create-project.test.ts.skip +0 -95
  210. package/tests/unit/core/framework.test.ts +0 -144
  211. package/tests/unit/core/plugins/logger.test.ts.skip +0 -268
  212. package/tests/unit/core/plugins/vite.test.ts.disabled +0 -188
  213. package/tests/utils/test-helpers.ts +0 -61
  214. package/vitest.config.ts +0 -50
  215. package/workspace.json +0 -6
@@ -1,431 +0,0 @@
1
- /**
2
- * Tests for EnvironmentProcessor
3
- * Tests environment variable processing, type conversion, and precedence handling
4
- */
5
-
6
- import { describe, test, expect, beforeEach, afterEach, vi } from 'vitest'
7
- import {
8
- EnvironmentProcessor,
9
- EnvConverter,
10
- getEnvironmentInfo
11
- } from '../env'
12
-
13
- describe('EnvConverter', () => {
14
- describe('toNumber', () => {
15
- test('converts valid number strings', () => {
16
- expect(EnvConverter.toNumber('123', 0)).toBe(123)
17
- expect(EnvConverter.toNumber('0', 10)).toBe(0)
18
- expect(EnvConverter.toNumber('-456', 0)).toBe(-456)
19
- })
20
-
21
- test('returns default for invalid values', () => {
22
- expect(EnvConverter.toNumber('invalid', 42)).toBe(42)
23
- expect(EnvConverter.toNumber('', 42)).toBe(42)
24
- expect(EnvConverter.toNumber(undefined, 42)).toBe(42)
25
- expect(EnvConverter.toNumber('12.5', 42)).toBe(12) // parseInt behavior
26
- })
27
-
28
- test('handles edge cases', () => {
29
- expect(EnvConverter.toNumber('0x10', 0)).toBe(0) // parseInt(value, 10) doesn't parse hex
30
- expect(EnvConverter.toNumber('010', 0)).toBe(10) // parseInt with base 10
31
- expect(EnvConverter.toNumber('Infinity', 42)).toBe(42)
32
- expect(EnvConverter.toNumber('NaN', 42)).toBe(42)
33
- })
34
- })
35
-
36
- describe('toBoolean', () => {
37
- test('converts truthy values', () => {
38
- expect(EnvConverter.toBoolean('true', false)).toBe(true)
39
- expect(EnvConverter.toBoolean('1', false)).toBe(true)
40
- expect(EnvConverter.toBoolean('yes', false)).toBe(true)
41
- expect(EnvConverter.toBoolean('on', false)).toBe(true)
42
- expect(EnvConverter.toBoolean('TRUE', false)).toBe(true)
43
- expect(EnvConverter.toBoolean('YES', false)).toBe(true)
44
- })
45
-
46
- test('converts falsy values', () => {
47
- expect(EnvConverter.toBoolean('false', true)).toBe(false)
48
- expect(EnvConverter.toBoolean('0', true)).toBe(false)
49
- expect(EnvConverter.toBoolean('no', true)).toBe(false)
50
- expect(EnvConverter.toBoolean('off', true)).toBe(false)
51
- expect(EnvConverter.toBoolean('', true)).toBe(true) // Empty string returns default
52
- expect(EnvConverter.toBoolean('invalid', true)).toBe(false)
53
- })
54
-
55
- test('returns default for undefined', () => {
56
- expect(EnvConverter.toBoolean(undefined, true)).toBe(true)
57
- expect(EnvConverter.toBoolean(undefined, false)).toBe(false)
58
- })
59
- })
60
-
61
- describe('toArray', () => {
62
- test('converts comma-separated values', () => {
63
- expect(EnvConverter.toArray('a,b,c')).toEqual(['a', 'b', 'c'])
64
- expect(EnvConverter.toArray('one, two , three')).toEqual(['one', 'two', 'three'])
65
- expect(EnvConverter.toArray('single')).toEqual(['single'])
66
- })
67
-
68
- test('handles empty and invalid values', () => {
69
- expect(EnvConverter.toArray('')).toEqual([])
70
- expect(EnvConverter.toArray(',,')).toEqual([])
71
- expect(EnvConverter.toArray(undefined)).toEqual([])
72
- expect(EnvConverter.toArray(undefined, ['default'])).toEqual(['default'])
73
- })
74
-
75
- test('filters empty strings', () => {
76
- expect(EnvConverter.toArray('a,,b,,')).toEqual(['a', 'b'])
77
- expect(EnvConverter.toArray(' , , ')).toEqual([])
78
- })
79
- })
80
-
81
- describe('toLogLevel', () => {
82
- test('converts valid log levels', () => {
83
- expect(EnvConverter.toLogLevel('debug', 'info')).toBe('debug')
84
- expect(EnvConverter.toLogLevel('info', 'debug')).toBe('info')
85
- expect(EnvConverter.toLogLevel('warn', 'info')).toBe('warn')
86
- expect(EnvConverter.toLogLevel('error', 'info')).toBe('error')
87
- expect(EnvConverter.toLogLevel('DEBUG', 'info')).toBe('debug') // case insensitive
88
- })
89
-
90
- test('returns default for invalid values', () => {
91
- expect(EnvConverter.toLogLevel('invalid', 'info')).toBe('info')
92
- expect(EnvConverter.toLogLevel('', 'warn')).toBe('warn')
93
- expect(EnvConverter.toLogLevel(undefined, 'error')).toBe('error')
94
- })
95
- })
96
-
97
- describe('toBuildTarget', () => {
98
- test('converts valid build targets', () => {
99
- expect(EnvConverter.toBuildTarget('bun', 'node')).toBe('bun')
100
- expect(EnvConverter.toBuildTarget('node', 'bun')).toBe('node')
101
- expect(EnvConverter.toBuildTarget('docker', 'bun')).toBe('docker')
102
- expect(EnvConverter.toBuildTarget('BUN', 'node')).toBe('bun') // case insensitive
103
- })
104
-
105
- test('returns default for invalid values', () => {
106
- expect(EnvConverter.toBuildTarget('invalid', 'bun')).toBe('bun')
107
- expect(EnvConverter.toBuildTarget('', 'node')).toBe('node')
108
- expect(EnvConverter.toBuildTarget(undefined, 'docker')).toBe('docker')
109
- })
110
- })
111
-
112
- describe('toObject', () => {
113
- test('parses valid JSON', () => {
114
- expect(EnvConverter.toObject('{"key": "value"}', {})).toEqual({ key: 'value' })
115
- expect(EnvConverter.toObject('[1,2,3]', {})).toEqual([1, 2, 3])
116
- expect(EnvConverter.toObject('null', {})).toBe(null)
117
- })
118
-
119
- test('returns default for invalid JSON', () => {
120
- const defaultObj = { default: true }
121
- expect(EnvConverter.toObject('invalid json', defaultObj)).toBe(defaultObj)
122
- expect(EnvConverter.toObject('', defaultObj)).toBe(defaultObj)
123
- expect(EnvConverter.toObject(undefined, defaultObj)).toBe(defaultObj)
124
- })
125
- })
126
- })
127
-
128
- describe('EnvironmentProcessor', () => {
129
- let processor: EnvironmentProcessor
130
- let originalEnv: NodeJS.ProcessEnv
131
-
132
- beforeEach(() => {
133
- processor = new EnvironmentProcessor()
134
- originalEnv = { ...process.env }
135
- // Clear environment variables
136
- for (const key in process.env) {
137
- if (key.startsWith('FLUXSTACK_') || key.startsWith('PORT') || key.startsWith('HOST') ||
138
- key.startsWith('CORS_') || key.startsWith('LOG_') || key.startsWith('BUILD_') ||
139
- key.startsWith('DATABASE_') || key.startsWith('JWT_') || key.startsWith('SMTP_') ||
140
- key.startsWith('VITE_') || key.startsWith('API_') || key.startsWith('CLIENT_') ||
141
- key.startsWith('MONITORING_') || key.startsWith('METRICS_') || key.startsWith('PROFILING_')) {
142
- delete process.env[key]
143
- }
144
- }
145
- })
146
-
147
- afterEach(() => {
148
- process.env = originalEnv
149
- })
150
-
151
- describe('processEnvironmentVariables', () => {
152
- test('processes server configuration', () => {
153
- process.env.PORT = '8080'
154
- process.env.HOST = 'example.com'
155
- process.env.FLUXSTACK_API_PREFIX = '/v1'
156
-
157
- const config = processor.processEnvironmentVariables()
158
-
159
- expect(config.server?.port).toBe(8080)
160
- expect(config.server?.host).toBe('example.com')
161
- expect(config.server?.apiPrefix).toBe('/v1')
162
- })
163
-
164
- test('processes CORS configuration', () => {
165
- process.env.CORS_ORIGINS = 'http://localhost:3000,https://example.com'
166
- process.env.CORS_METHODS = 'GET,POST,PUT'
167
- process.env.CORS_HEADERS = 'Content-Type,Authorization'
168
- process.env.CORS_CREDENTIALS = 'true'
169
- process.env.CORS_MAX_AGE = '86400'
170
-
171
- const config = processor.processEnvironmentVariables()
172
-
173
- expect(config.server?.cors?.origins).toEqual(['http://localhost:3000', 'https://example.com'])
174
- expect(config.server?.cors?.methods).toEqual(['GET', 'POST', 'PUT'])
175
- expect(config.server?.cors?.headers).toEqual(['Content-Type', 'Authorization'])
176
- expect(config.server?.cors?.credentials).toBe(true)
177
- expect(config.server?.cors?.maxAge).toBe(86400)
178
- })
179
-
180
- test('processes client configuration', () => {
181
- process.env.VITE_PORT = '5174'
182
- process.env.VITE_API_URL = 'http://localhost:4000'
183
- process.env.FLUXSTACK_CLIENT_SOURCEMAPS = 'false'
184
- process.env.FLUXSTACK_CLIENT_MINIFY = 'true'
185
-
186
- const config = processor.processEnvironmentVariables()
187
-
188
- expect(config.client?.port).toBe(5174)
189
- expect(config.client?.proxy?.target).toBe('http://localhost:4000')
190
- expect(config.client?.build?.sourceMaps).toBe(false)
191
- expect(config.client?.build?.minify).toBe(true)
192
- })
193
-
194
- test('processes build configuration', () => {
195
- process.env.BUILD_TARGET = 'docker'
196
- process.env.BUILD_OUTDIR = 'build'
197
- process.env.BUILD_SOURCEMAPS = 'true'
198
- process.env.BUILD_MINIFY = 'false'
199
- process.env.BUILD_TREESHAKE = 'true'
200
-
201
- const config = processor.processEnvironmentVariables()
202
-
203
- expect(config.build?.target).toBe('docker')
204
- expect(config.build?.outDir).toBe('build')
205
- expect(config.build?.sourceMaps).toBe(true)
206
- expect(config.build?.optimization?.minify).toBe(false)
207
- expect(config.build?.optimization?.treeshake).toBe(true)
208
- })
209
-
210
- test('processes logging configuration', () => {
211
- process.env.LOG_LEVEL = 'debug'
212
- process.env.LOG_FORMAT = 'json'
213
-
214
- const config = processor.processEnvironmentVariables()
215
-
216
- expect(config.logging?.level).toBe('debug')
217
- expect(config.logging?.format).toBe('json')
218
- })
219
-
220
- test('processes monitoring configuration', () => {
221
- process.env.MONITORING_ENABLED = 'true'
222
- process.env.METRICS_ENABLED = 'false'
223
- process.env.METRICS_INTERVAL = '30000'
224
- process.env.PROFILING_ENABLED = 'true'
225
- process.env.PROFILING_SAMPLE_RATE = '0.1'
226
-
227
- const config = processor.processEnvironmentVariables()
228
-
229
- expect(config.monitoring?.enabled).toBe(true)
230
- expect(config.monitoring?.metrics?.enabled).toBe(false)
231
- expect(config.monitoring?.metrics?.collectInterval).toBe(30000)
232
- expect(config.monitoring?.profiling?.enabled).toBe(true)
233
- expect(config.monitoring?.profiling?.sampleRate).toBe(0) // parseInt converts '0.1' to 0
234
- })
235
-
236
- test('processes database configuration', () => {
237
- process.env.DATABASE_URL = 'postgresql://user:pass@localhost:5432/db'
238
- process.env.DATABASE_HOST = 'db.example.com'
239
- process.env.DATABASE_PORT = '5433'
240
- process.env.DATABASE_NAME = 'myapp'
241
- process.env.DATABASE_SSL = 'true'
242
- process.env.DATABASE_POOL_SIZE = '20'
243
-
244
- const config = processor.processEnvironmentVariables()
245
-
246
- expect(config.database?.url).toBe('postgresql://user:pass@localhost:5432/db')
247
- expect(config.database?.host).toBe('db.example.com')
248
- expect(config.database?.port).toBe(5433)
249
- expect(config.database?.database).toBe('myapp')
250
- expect(config.database?.ssl).toBe(true)
251
- expect(config.database?.poolSize).toBe(20)
252
- })
253
-
254
- test('processes auth configuration', () => {
255
- process.env.JWT_SECRET = 'my-secret-key'
256
- process.env.JWT_EXPIRES_IN = '7d'
257
- process.env.JWT_ALGORITHM = 'HS256'
258
- process.env.JWT_ISSUER = 'fluxstack'
259
-
260
- const config = processor.processEnvironmentVariables()
261
-
262
- expect(config.auth?.secret).toBe('my-secret-key')
263
- expect(config.auth?.expiresIn).toBe('7d')
264
- expect(config.auth?.algorithm).toBe('HS256')
265
- expect(config.auth?.issuer).toBe('fluxstack')
266
- })
267
-
268
- test('processes email configuration', () => {
269
- process.env.SMTP_HOST = 'smtp.gmail.com'
270
- process.env.SMTP_PORT = '587'
271
- process.env.SMTP_USER = 'user@example.com'
272
- process.env.SMTP_SECURE = 'true'
273
- process.env.SMTP_FROM = 'noreply@example.com'
274
-
275
- const config = processor.processEnvironmentVariables()
276
-
277
- expect(config.email?.host).toBe('smtp.gmail.com')
278
- expect(config.email?.port).toBe(587)
279
- expect(config.email?.user).toBe('user@example.com')
280
- expect(config.email?.secure).toBe(true)
281
- expect(config.email?.from).toBe('noreply@example.com')
282
- })
283
-
284
- test('processes plugin configuration', () => {
285
- process.env.FLUXSTACK_PLUGINS_ENABLED = 'plugin1,plugin2,plugin3'
286
- process.env.FLUXSTACK_PLUGINS_DISABLED = 'plugin4,plugin5'
287
-
288
- const config = processor.processEnvironmentVariables()
289
-
290
- expect(config.plugins?.enabled).toEqual(['plugin1', 'plugin2', 'plugin3'])
291
- expect(config.plugins?.disabled).toEqual(['plugin4', 'plugin5'])
292
- })
293
-
294
- test('handles precedence with both standard and FluxStack prefixed variables', () => {
295
- // In current implementation, the last one processed wins (FLUXSTACK_ comes after standard)
296
- process.env.PORT = '3000'
297
- process.env.FLUXSTACK_PORT = '4000'
298
- process.env.API_PREFIX = '/api'
299
- process.env.FLUXSTACK_API_PREFIX = '/v2'
300
-
301
- const config = processor.processEnvironmentVariables()
302
-
303
- // The implementation uses || operator, so FLUXSTACK_PORT takes precedence only if PORT is not set
304
- expect(config.server?.port).toBe(3000) // PORT takes precedence when both are set
305
- expect(config.server?.apiPrefix).toBe('/v2') // FLUXSTACK_API_PREFIX wins
306
- })
307
-
308
- test('cleans empty objects from result', () => {
309
- // No environment variables set
310
- const config = processor.processEnvironmentVariables()
311
-
312
- // Should not contain empty nested objects
313
- expect(config.server).toBeUndefined()
314
- expect(config.client).toBeUndefined()
315
- expect(config.database).toBeUndefined()
316
- })
317
-
318
- test('tracks precedence information', () => {
319
- process.env.PORT = '3000'
320
- process.env.LOG_LEVEL = 'debug'
321
-
322
- processor.processEnvironmentVariables()
323
- const precedence = processor.getPrecedenceInfo()
324
-
325
- expect(precedence.has('server.port')).toBe(true)
326
- expect(precedence.has('logging.level')).toBe(true)
327
-
328
- const portPrecedence = precedence.get('server.port')
329
- expect(portPrecedence?.source).toBe('environment')
330
- expect(portPrecedence?.value).toBe(3000)
331
- expect(portPrecedence?.priority).toBe(3)
332
- })
333
- })
334
-
335
- describe('precedence tracking', () => {
336
- test('tracks precedence for all set values', () => {
337
- process.env.PORT = '3000'
338
- process.env.BUILD_TARGET = 'docker'
339
- process.env.CORS_ORIGINS = 'localhost'
340
-
341
- processor.processEnvironmentVariables()
342
- const precedence = processor.getPrecedenceInfo()
343
-
344
- expect(precedence.size).toBeGreaterThan(0)
345
- expect(precedence.get('server.port')?.priority).toBe(3)
346
- expect(precedence.get('build.target')?.priority).toBe(3)
347
- expect(precedence.get('server.cors.origins')?.priority).toBe(3)
348
- })
349
-
350
- test('clears precedence tracking', () => {
351
- process.env.PORT = '3000'
352
- processor.processEnvironmentVariables()
353
-
354
- expect(processor.getPrecedenceInfo().size).toBeGreaterThan(0)
355
-
356
- processor.clearPrecedence()
357
- expect(processor.getPrecedenceInfo().size).toBe(0)
358
- })
359
- })
360
- })
361
-
362
- describe('getEnvironmentInfo', () => {
363
- let originalNodeEnv: string | undefined
364
-
365
- beforeEach(() => {
366
- originalNodeEnv = process.env.NODE_ENV
367
- })
368
-
369
- afterEach(() => {
370
- if (originalNodeEnv) {
371
- process.env.NODE_ENV = originalNodeEnv
372
- } else {
373
- delete process.env.NODE_ENV
374
- }
375
- })
376
-
377
- test('detects development environment', () => {
378
- process.env.NODE_ENV = 'development'
379
- const info = getEnvironmentInfo()
380
-
381
- expect(info.name).toBe('development')
382
- expect(info.nodeEnv).toBe('development')
383
- expect(info.isDevelopment).toBe(true)
384
- expect(info.isProduction).toBe(false)
385
- expect(info.isTest).toBe(false)
386
- })
387
-
388
- test('detects production environment', () => {
389
- process.env.NODE_ENV = 'production'
390
- const info = getEnvironmentInfo()
391
-
392
- expect(info.name).toBe('production')
393
- expect(info.nodeEnv).toBe('production')
394
- expect(info.isDevelopment).toBe(false)
395
- expect(info.isProduction).toBe(true)
396
- expect(info.isTest).toBe(false)
397
- })
398
-
399
- test('detects test environment', () => {
400
- process.env.NODE_ENV = 'test'
401
- const info = getEnvironmentInfo()
402
-
403
- expect(info.name).toBe('test')
404
- expect(info.nodeEnv).toBe('test')
405
- expect(info.isDevelopment).toBe(false)
406
- expect(info.isProduction).toBe(false)
407
- expect(info.isTest).toBe(true)
408
- })
409
-
410
- test('defaults to development when NODE_ENV is not set', () => {
411
- delete process.env.NODE_ENV
412
- const info = getEnvironmentInfo()
413
-
414
- expect(info.name).toBe('development')
415
- expect(info.nodeEnv).toBe('development')
416
- expect(info.isDevelopment).toBe(true)
417
- expect(info.isProduction).toBe(false)
418
- expect(info.isTest).toBe(false)
419
- })
420
-
421
- test('handles custom environment names', () => {
422
- process.env.NODE_ENV = 'staging'
423
- const info = getEnvironmentInfo()
424
-
425
- expect(info.name).toBe('staging')
426
- expect(info.nodeEnv).toBe('staging')
427
- expect(info.isDevelopment).toBe(false)
428
- expect(info.isProduction).toBe(false)
429
- expect(info.isTest).toBe(false)
430
- })
431
- })