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,335 +0,0 @@
1
- /**
2
- * Tests for Plugin Registry
3
- */
4
-
5
- import { describe, it, expect, beforeEach, vi } from 'vitest'
6
- import { PluginRegistry } from '../registry'
7
- import type { Plugin, PluginManifest } from '../types'
8
- import type { Logger } from '../../utils/logger/index'
9
-
10
- // Mock logger
11
- const mockLogger: Logger = {
12
- debug: vi.fn(),
13
- info: vi.fn(),
14
- warn: vi.fn(),
15
- error: vi.fn(),
16
- child: vi.fn(() => mockLogger),
17
- time: vi.fn(),
18
- timeEnd: vi.fn(),
19
- request: vi.fn()
20
- }
21
-
22
- describe('PluginRegistry', () => {
23
- let registry: PluginRegistry
24
-
25
- beforeEach(() => {
26
- registry = new PluginRegistry({ logger: mockLogger })
27
- vi.clearAllMocks()
28
- })
29
-
30
- describe('Plugin Registration', () => {
31
- it('should register a plugin successfully', async () => {
32
- const plugin: Plugin = {
33
- name: 'test-plugin',
34
- version: '1.0.0'
35
- }
36
-
37
- await registry.register(plugin)
38
- expect(registry.get('test-plugin')).toBe(plugin)
39
- expect(registry.has('test-plugin')).toBe(true)
40
- })
41
-
42
- it('should register a plugin with manifest', async () => {
43
- const plugin: Plugin = {
44
- name: 'test-plugin',
45
- version: '1.0.0'
46
- }
47
-
48
- const manifest: PluginManifest = {
49
- name: 'test-plugin',
50
- version: '1.0.0',
51
- description: 'Test plugin',
52
- author: 'Test Author',
53
- license: 'MIT',
54
- keywords: ['test'],
55
- dependencies: {},
56
- fluxstack: {
57
- version: '1.0.0',
58
- hooks: ['setup']
59
- }
60
- }
61
-
62
- await registry.register(plugin, manifest)
63
- expect(registry.getManifest('test-plugin')).toBe(manifest)
64
- })
65
-
66
- it('should throw error when registering duplicate plugin', async () => {
67
- const plugin: Plugin = {
68
- name: 'duplicate-plugin'
69
- }
70
-
71
- await registry.register(plugin)
72
- await expect(registry.register(plugin)).rejects.toThrow("Plugin 'duplicate-plugin' is already registered")
73
- })
74
-
75
- it('should validate plugin structure', async () => {
76
- const invalidPlugin = {
77
- // Missing name property
78
- version: '1.0.0'
79
- } as Plugin
80
-
81
- await expect(registry.register(invalidPlugin)).rejects.toThrow('Plugin must have a valid name property')
82
- })
83
-
84
- it('should unregister a plugin successfully', async () => {
85
- const plugin: Plugin = {
86
- name: 'removable-plugin'
87
- }
88
-
89
- await registry.register(plugin)
90
- expect(registry.get('removable-plugin')).toBe(plugin)
91
-
92
- registry.unregister('removable-plugin')
93
- expect(registry.get('removable-plugin')).toBeUndefined()
94
- expect(registry.has('removable-plugin')).toBe(false)
95
- })
96
-
97
- it('should throw error when unregistering non-existent plugin', () => {
98
- expect(() => registry.unregister('non-existent')).toThrow("Plugin 'non-existent' is not registered")
99
- })
100
-
101
- it('should prevent unregistering plugin with dependents', async () => {
102
- const pluginA: Plugin = { name: 'plugin-a' }
103
- const pluginB: Plugin = { name: 'plugin-b', dependencies: ['plugin-a'] }
104
-
105
- await registry.register(pluginA)
106
- await registry.register(pluginB)
107
-
108
- expect(() => registry.unregister('plugin-a')).toThrow(
109
- "Cannot unregister plugin 'plugin-a' because it is required by: plugin-b"
110
- )
111
- })
112
- })
113
-
114
- describe('Plugin Retrieval', () => {
115
- it('should get all registered plugins', async () => {
116
- const plugin1: Plugin = { name: 'plugin-1' }
117
- const plugin2: Plugin = { name: 'plugin-2' }
118
-
119
- await registry.register(plugin1)
120
- await registry.register(plugin2)
121
-
122
- const allPlugins = registry.getAll()
123
- expect(allPlugins).toHaveLength(2)
124
- expect(allPlugins).toContain(plugin1)
125
- expect(allPlugins).toContain(plugin2)
126
- })
127
-
128
- it('should return undefined for non-existent plugin', () => {
129
- expect(registry.get('non-existent')).toBeUndefined()
130
- })
131
-
132
- it('should get plugin dependencies', async () => {
133
- const plugin: Plugin = {
134
- name: 'plugin-with-deps',
135
- dependencies: ['dep1', 'dep2']
136
- }
137
-
138
- await registry.register(plugin)
139
- expect(registry.getDependencies('plugin-with-deps')).toEqual(['dep1', 'dep2'])
140
- })
141
-
142
- it('should get plugin dependents', async () => {
143
- const pluginA: Plugin = { name: 'plugin-a' }
144
- const pluginB: Plugin = { name: 'plugin-b', dependencies: ['plugin-a'] }
145
- const pluginC: Plugin = { name: 'plugin-c', dependencies: ['plugin-a'] }
146
-
147
- await registry.register(pluginA)
148
- await registry.register(pluginB)
149
- await registry.register(pluginC)
150
-
151
- const dependents = registry.getDependents('plugin-a')
152
- expect(dependents).toContain('plugin-b')
153
- expect(dependents).toContain('plugin-c')
154
- })
155
-
156
- it('should get registry statistics', async () => {
157
- const plugin1: Plugin = { name: 'plugin-1' }
158
- const plugin2: Plugin = { name: 'plugin-2' }
159
-
160
- await registry.register(plugin1)
161
- await registry.register(plugin2)
162
-
163
- const stats = registry.getStats()
164
- expect(stats.totalPlugins).toBe(2)
165
- expect(stats.loadOrder).toBe(2)
166
- })
167
- })
168
-
169
- describe('Dependency Management', () => {
170
- it('should validate dependencies successfully', async () => {
171
- const pluginA: Plugin = {
172
- name: 'plugin-a'
173
- }
174
-
175
- const pluginB: Plugin = {
176
- name: 'plugin-b',
177
- dependencies: ['plugin-a']
178
- }
179
-
180
- await registry.register(pluginA)
181
- await registry.register(pluginB)
182
-
183
- expect(() => registry.validateDependencies()).not.toThrow()
184
- })
185
-
186
- it('should throw error for missing dependencies', async () => {
187
- const pluginWithMissingDep: Plugin = {
188
- name: 'plugin-with-missing-dep',
189
- dependencies: ['non-existent-plugin']
190
- }
191
-
192
- await registry.register(pluginWithMissingDep)
193
- expect(() => registry.validateDependencies()).toThrow(
194
- "Plugin dependency validation failed"
195
- )
196
- })
197
-
198
- it('should detect circular dependencies', async () => {
199
- const pluginA: Plugin = {
200
- name: 'plugin-a',
201
- dependencies: ['plugin-b']
202
- }
203
-
204
- const pluginB: Plugin = {
205
- name: 'plugin-b',
206
- dependencies: ['plugin-a']
207
- }
208
-
209
- await registry.register(pluginA)
210
-
211
- await expect(registry.register(pluginB)).rejects.toThrow('Circular dependency detected')
212
- })
213
- })
214
-
215
- describe('Load Order', () => {
216
- it('should determine correct load order based on dependencies', async () => {
217
- const pluginA: Plugin = {
218
- name: 'plugin-a'
219
- }
220
-
221
- const pluginB: Plugin = {
222
- name: 'plugin-b',
223
- dependencies: ['plugin-a']
224
- }
225
-
226
- const pluginC: Plugin = {
227
- name: 'plugin-c',
228
- dependencies: ['plugin-b']
229
- }
230
-
231
- await registry.register(pluginA)
232
- await registry.register(pluginB)
233
- await registry.register(pluginC)
234
-
235
- const loadOrder = registry.getLoadOrder()
236
-
237
- expect(loadOrder.indexOf('plugin-a')).toBeLessThan(loadOrder.indexOf('plugin-b'))
238
- expect(loadOrder.indexOf('plugin-b')).toBeLessThan(loadOrder.indexOf('plugin-c'))
239
- })
240
-
241
- it('should respect plugin priorities', async () => {
242
- const lowPriorityPlugin: Plugin = {
243
- name: 'low-priority',
244
- priority: 1
245
- }
246
-
247
- const highPriorityPlugin: Plugin = {
248
- name: 'high-priority',
249
- priority: 10
250
- }
251
-
252
- await registry.register(lowPriorityPlugin)
253
- await registry.register(highPriorityPlugin)
254
-
255
- const loadOrder = registry.getLoadOrder()
256
-
257
- expect(loadOrder.indexOf('high-priority')).toBeLessThan(loadOrder.indexOf('low-priority'))
258
- })
259
-
260
- it('should handle plugins without priorities', async () => {
261
- const pluginWithoutPriority: Plugin = {
262
- name: 'no-priority'
263
- }
264
-
265
- const pluginWithPriority: Plugin = {
266
- name: 'with-priority',
267
- priority: 5
268
- }
269
-
270
- await registry.register(pluginWithoutPriority)
271
- await registry.register(pluginWithPriority)
272
-
273
- const loadOrder = registry.getLoadOrder()
274
-
275
- expect(loadOrder.indexOf('with-priority')).toBeLessThan(loadOrder.indexOf('no-priority'))
276
- })
277
- })
278
-
279
- describe('Plugin Discovery', () => {
280
- it('should discover plugins from directories', async () => {
281
- // This would require mocking the filesystem
282
- // For now, just test that the method exists and returns an array
283
- const results = await registry.discoverPlugins({
284
- directories: ['non-existent-dir']
285
- })
286
-
287
- expect(Array.isArray(results)).toBe(true)
288
- })
289
-
290
- it('should load plugin from path', async () => {
291
- // This would require mocking the filesystem and import
292
- // For now, just test that the method exists
293
- const result = await registry.loadPlugin('non-existent-path')
294
-
295
- expect(result).toHaveProperty('success')
296
- expect(result.success).toBe(false)
297
- expect(result).toHaveProperty('error')
298
- })
299
- })
300
-
301
- describe('Plugin Configuration', () => {
302
- it('should validate plugin configuration', async () => {
303
- const plugin: Plugin = {
304
- name: 'config-plugin',
305
- configSchema: {
306
- type: 'object',
307
- properties: {
308
- apiKey: { type: 'string' }
309
- },
310
- required: ['apiKey']
311
- }
312
- }
313
-
314
- const config = {
315
- plugins: {
316
- enabled: ['config-plugin'],
317
- disabled: [],
318
- config: {
319
- 'config-plugin': {
320
- apiKey: 'test-key'
321
- }
322
- }
323
- }
324
- }
325
-
326
- const registryWithConfig = new PluginRegistry({
327
- logger: mockLogger,
328
- config: config as any
329
- })
330
-
331
- await registryWithConfig.register(plugin)
332
- expect(registryWithConfig.get('config-plugin')).toBe(plugin)
333
- })
334
- })
335
- })
@@ -1,139 +0,0 @@
1
- /**
2
- * Tests for Error Handling System
3
- */
4
-
5
- import { describe, it, expect } from 'vitest'
6
- import {
7
- FluxStackError,
8
- ValidationError,
9
- NotFoundError,
10
- UnauthorizedError,
11
- ForbiddenError,
12
- ConflictError,
13
- InternalServerError,
14
- ServiceUnavailableError
15
- } from '../errors'
16
-
17
- describe('Error Classes', () => {
18
- describe('FluxStackError', () => {
19
- it('should create error with all properties', () => {
20
- const context = { field: 'email', value: 'invalid' }
21
- const error = new FluxStackError('Test error', 'TEST_ERROR', 400, context)
22
-
23
- expect(error.message).toBe('Test error')
24
- expect(error.code).toBe('TEST_ERROR')
25
- expect(error.statusCode).toBe(400)
26
- expect(error.context).toBe(context)
27
- expect(error.timestamp).toBeInstanceOf(Date)
28
- expect(error.name).toBe('FluxStackError')
29
- })
30
-
31
- it('should default to status code 500', () => {
32
- const error = new FluxStackError('Test error', 'TEST_ERROR')
33
- expect(error.statusCode).toBe(500)
34
- })
35
-
36
- it('should serialize to JSON correctly', () => {
37
- const error = new FluxStackError('Test error', 'TEST_ERROR', 400, { test: true })
38
- const json = error.toJSON()
39
-
40
- expect(json.name).toBe('FluxStackError')
41
- expect(json.message).toBe('Test error')
42
- expect(json.code).toBe('TEST_ERROR')
43
- expect(json.statusCode).toBe(400)
44
- expect(json.context).toEqual({ test: true })
45
- expect(json.timestamp).toBeInstanceOf(Date)
46
- expect(json.stack).toBeDefined()
47
- })
48
- })
49
-
50
- describe('ValidationError', () => {
51
- it('should create validation error with correct defaults', () => {
52
- const error = new ValidationError('Invalid input')
53
-
54
- expect(error.message).toBe('Invalid input')
55
- expect(error.code).toBe('VALIDATION_ERROR')
56
- expect(error.statusCode).toBe(400)
57
- expect(error.name).toBe('ValidationError')
58
- })
59
-
60
- it('should include context', () => {
61
- const context = { field: 'email', rule: 'required' }
62
- const error = new ValidationError('Email is required', context)
63
-
64
- expect(error.context).toBe(context)
65
- })
66
- })
67
-
68
- describe('NotFoundError', () => {
69
- it('should create not found error', () => {
70
- const error = new NotFoundError('User')
71
-
72
- expect(error.message).toBe('User not found')
73
- expect(error.code).toBe('NOT_FOUND')
74
- expect(error.statusCode).toBe(404)
75
- expect(error.name).toBe('NotFoundError')
76
- })
77
- })
78
-
79
- describe('UnauthorizedError', () => {
80
- it('should create unauthorized error with default message', () => {
81
- const error = new UnauthorizedError()
82
-
83
- expect(error.message).toBe('Unauthorized')
84
- expect(error.code).toBe('UNAUTHORIZED')
85
- expect(error.statusCode).toBe(401)
86
- expect(error.name).toBe('UnauthorizedError')
87
- })
88
-
89
- it('should create unauthorized error with custom message', () => {
90
- const error = new UnauthorizedError('Invalid token')
91
-
92
- expect(error.message).toBe('Invalid token')
93
- })
94
- })
95
-
96
- describe('ForbiddenError', () => {
97
- it('should create forbidden error', () => {
98
- const error = new ForbiddenError('Access denied')
99
-
100
- expect(error.message).toBe('Access denied')
101
- expect(error.code).toBe('FORBIDDEN')
102
- expect(error.statusCode).toBe(403)
103
- expect(error.name).toBe('ForbiddenError')
104
- })
105
- })
106
-
107
- describe('ConflictError', () => {
108
- it('should create conflict error', () => {
109
- const error = new ConflictError('Resource already exists')
110
-
111
- expect(error.message).toBe('Resource already exists')
112
- expect(error.code).toBe('CONFLICT')
113
- expect(error.statusCode).toBe(409)
114
- expect(error.name).toBe('ConflictError')
115
- })
116
- })
117
-
118
- describe('InternalServerError', () => {
119
- it('should create internal server error with default message', () => {
120
- const error = new InternalServerError()
121
-
122
- expect(error.message).toBe('Internal server error')
123
- expect(error.code).toBe('INTERNAL_SERVER_ERROR')
124
- expect(error.statusCode).toBe(500)
125
- expect(error.name).toBe('InternalServerError')
126
- })
127
- })
128
-
129
- describe('ServiceUnavailableError', () => {
130
- it('should create service unavailable error', () => {
131
- const error = new ServiceUnavailableError('Database is down')
132
-
133
- expect(error.message).toBe('Database is down')
134
- expect(error.code).toBe('SERVICE_UNAVAILABLE')
135
- expect(error.statusCode).toBe(503)
136
- expect(error.name).toBe('ServiceUnavailableError')
137
- })
138
- })
139
- })