create-fluxstack 1.0.1 → 1.0.2
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.
- package/create-fluxstack.ts +2 -3
- package/package.json +1 -1
- package/.env +0 -30
- package/LICENSE +0 -21
- package/app/client/README.md +0 -69
- package/app/client/frontend-only.ts +0 -12
- package/app/client/index.html +0 -13
- package/app/client/public/vite.svg +0 -1
- package/app/client/src/App.css +0 -883
- package/app/client/src/App.tsx +0 -669
- package/app/client/src/assets/react.svg +0 -1
- package/app/client/src/components/TestPage.tsx +0 -453
- package/app/client/src/index.css +0 -51
- package/app/client/src/lib/eden-api.ts +0 -110
- package/app/client/src/main.tsx +0 -10
- package/app/client/src/vite-env.d.ts +0 -1
- package/app/client/tsconfig.app.json +0 -43
- package/app/client/tsconfig.json +0 -7
- package/app/client/tsconfig.node.json +0 -25
- package/app/server/app.ts +0 -10
- package/app/server/backend-only.ts +0 -15
- package/app/server/controllers/users.controller.ts +0 -69
- package/app/server/index.ts +0 -104
- package/app/server/routes/index.ts +0 -25
- package/app/server/routes/users.routes.ts +0 -121
- package/app/server/types/index.ts +0 -1
- package/app/shared/types/index.ts +0 -18
- package/bun.lock +0 -1053
- package/core/__tests__/integration.test.ts +0 -227
- package/core/build/index.ts +0 -186
- package/core/cli/command-registry.ts +0 -334
- package/core/cli/index.ts +0 -394
- package/core/cli/plugin-discovery.ts +0 -200
- package/core/client/standalone.ts +0 -57
- package/core/config/__tests__/config-loader.test.ts +0 -591
- package/core/config/__tests__/config-merger.test.ts +0 -657
- package/core/config/__tests__/env-converter.test.ts +0 -372
- package/core/config/__tests__/env-processor.test.ts +0 -431
- package/core/config/__tests__/env.test.ts +0 -452
- package/core/config/__tests__/integration.test.ts +0 -418
- package/core/config/__tests__/loader.test.ts +0 -331
- package/core/config/__tests__/schema.test.ts +0 -129
- package/core/config/__tests__/validator.test.ts +0 -318
- package/core/config/env-dynamic.ts +0 -326
- package/core/config/env.ts +0 -597
- package/core/config/index.ts +0 -317
- package/core/config/loader.ts +0 -546
- package/core/config/runtime-config.ts +0 -322
- package/core/config/schema.ts +0 -694
- package/core/config/validator.ts +0 -540
- package/core/framework/__tests__/server.test.ts +0 -233
- package/core/framework/client.ts +0 -132
- package/core/framework/index.ts +0 -8
- package/core/framework/server.ts +0 -501
- package/core/framework/types.ts +0 -63
- package/core/plugins/__tests__/built-in.test.ts.disabled +0 -366
- package/core/plugins/__tests__/manager.test.ts +0 -398
- package/core/plugins/__tests__/monitoring.test.ts +0 -401
- package/core/plugins/__tests__/registry.test.ts +0 -335
- package/core/plugins/built-in/index.ts +0 -142
- package/core/plugins/built-in/logger/index.ts +0 -180
- package/core/plugins/built-in/monitoring/README.md +0 -193
- package/core/plugins/built-in/monitoring/index.ts +0 -912
- package/core/plugins/built-in/static/index.ts +0 -289
- package/core/plugins/built-in/swagger/index.ts +0 -229
- package/core/plugins/built-in/vite/index.ts +0 -316
- package/core/plugins/config.ts +0 -348
- package/core/plugins/discovery.ts +0 -350
- package/core/plugins/executor.ts +0 -351
- package/core/plugins/index.ts +0 -195
- package/core/plugins/manager.ts +0 -583
- package/core/plugins/registry.ts +0 -424
- package/core/plugins/types.ts +0 -254
- package/core/server/framework.ts +0 -123
- package/core/server/index.ts +0 -8
- package/core/server/plugins/database.ts +0 -182
- package/core/server/plugins/logger.ts +0 -47
- package/core/server/plugins/swagger.ts +0 -34
- package/core/server/standalone.ts +0 -91
- package/core/templates/create-project.ts +0 -455
- package/core/types/api.ts +0 -169
- package/core/types/build.ts +0 -174
- package/core/types/config.ts +0 -68
- package/core/types/index.ts +0 -127
- package/core/types/plugin.ts +0 -94
- package/core/utils/__tests__/errors.test.ts +0 -139
- package/core/utils/__tests__/helpers.test.ts +0 -297
- package/core/utils/__tests__/logger.test.ts +0 -141
- package/core/utils/env-runtime-v2.ts +0 -232
- package/core/utils/env-runtime.ts +0 -252
- package/core/utils/errors/codes.ts +0 -115
- package/core/utils/errors/handlers.ts +0 -63
- package/core/utils/errors/index.ts +0 -81
- package/core/utils/helpers.ts +0 -180
- package/core/utils/index.ts +0 -18
- package/core/utils/logger/index.ts +0 -161
- package/core/utils/logger.ts +0 -106
- package/core/utils/monitoring/index.ts +0 -212
- package/tsconfig.json +0 -51
- package/vite.config.ts +0 -42
|
@@ -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,142 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Built-in Plugins for FluxStack
|
|
3
|
-
* Core plugins that provide essential functionality
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
// Import all built-in plugins
|
|
7
|
-
import { loggerPlugin } from './logger'
|
|
8
|
-
import { swaggerPlugin } from './swagger'
|
|
9
|
-
import { vitePlugin } from './vite'
|
|
10
|
-
import { staticPlugin } from './static'
|
|
11
|
-
import { monitoringPlugin } from './monitoring'
|
|
12
|
-
|
|
13
|
-
// Export individual plugins
|
|
14
|
-
export { loggerPlugin } from './logger'
|
|
15
|
-
export { swaggerPlugin } from './swagger'
|
|
16
|
-
export { vitePlugin } from './vite'
|
|
17
|
-
export { staticPlugin } from './static'
|
|
18
|
-
export { monitoringPlugin } from './monitoring'
|
|
19
|
-
|
|
20
|
-
// Export as a collection
|
|
21
|
-
export const builtInPlugins = {
|
|
22
|
-
logger: loggerPlugin,
|
|
23
|
-
swagger: swaggerPlugin,
|
|
24
|
-
vite: vitePlugin,
|
|
25
|
-
static: staticPlugin,
|
|
26
|
-
monitoring: monitoringPlugin
|
|
27
|
-
} as const
|
|
28
|
-
|
|
29
|
-
// Export as an array for easy registration
|
|
30
|
-
export const builtInPluginsList = [
|
|
31
|
-
loggerPlugin,
|
|
32
|
-
swaggerPlugin,
|
|
33
|
-
vitePlugin,
|
|
34
|
-
staticPlugin,
|
|
35
|
-
monitoringPlugin
|
|
36
|
-
] as const
|
|
37
|
-
|
|
38
|
-
// Plugin categories
|
|
39
|
-
export const pluginCategories = {
|
|
40
|
-
core: [loggerPlugin, staticPlugin],
|
|
41
|
-
development: [vitePlugin],
|
|
42
|
-
documentation: [swaggerPlugin],
|
|
43
|
-
monitoring: [loggerPlugin, monitoringPlugin]
|
|
44
|
-
} as const
|
|
45
|
-
|
|
46
|
-
// Default plugin configuration
|
|
47
|
-
export const defaultPluginConfig = {
|
|
48
|
-
logger: {
|
|
49
|
-
logRequests: true,
|
|
50
|
-
logResponses: true,
|
|
51
|
-
logErrors: true,
|
|
52
|
-
includeHeaders: false,
|
|
53
|
-
includeBody: false,
|
|
54
|
-
slowRequestThreshold: 1000
|
|
55
|
-
},
|
|
56
|
-
swagger: {
|
|
57
|
-
enabled: true,
|
|
58
|
-
path: '/swagger',
|
|
59
|
-
title: 'FluxStack API',
|
|
60
|
-
description: 'Modern full-stack TypeScript framework with type-safe API endpoints'
|
|
61
|
-
},
|
|
62
|
-
vite: {
|
|
63
|
-
enabled: true,
|
|
64
|
-
port: 5173,
|
|
65
|
-
host: 'localhost',
|
|
66
|
-
checkInterval: 2000,
|
|
67
|
-
maxRetries: 10,
|
|
68
|
-
timeout: 5000
|
|
69
|
-
},
|
|
70
|
-
static: {
|
|
71
|
-
enabled: true,
|
|
72
|
-
publicDir: 'public',
|
|
73
|
-
distDir: 'dist/client',
|
|
74
|
-
indexFile: 'index.html',
|
|
75
|
-
spa: {
|
|
76
|
-
enabled: true,
|
|
77
|
-
fallback: 'index.html'
|
|
78
|
-
}
|
|
79
|
-
},
|
|
80
|
-
monitoring: {
|
|
81
|
-
enabled: false, // Disabled by default to avoid overhead
|
|
82
|
-
httpMetrics: true,
|
|
83
|
-
systemMetrics: true,
|
|
84
|
-
customMetrics: true,
|
|
85
|
-
collectInterval: 5000,
|
|
86
|
-
retentionPeriod: 300000,
|
|
87
|
-
exporters: [
|
|
88
|
-
{
|
|
89
|
-
type: 'console',
|
|
90
|
-
interval: 30000,
|
|
91
|
-
enabled: false
|
|
92
|
-
}
|
|
93
|
-
],
|
|
94
|
-
thresholds: {
|
|
95
|
-
responseTime: 1000,
|
|
96
|
-
errorRate: 0.05,
|
|
97
|
-
memoryUsage: 0.8,
|
|
98
|
-
cpuUsage: 0.8
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
} as const
|
|
102
|
-
|
|
103
|
-
/**
|
|
104
|
-
* Get default plugins for a specific environment
|
|
105
|
-
*/
|
|
106
|
-
export function getDefaultPlugins(environment: 'development' | 'production' | 'test' = 'development') {
|
|
107
|
-
const basePlugins = [loggerPlugin, staticPlugin]
|
|
108
|
-
|
|
109
|
-
switch (environment) {
|
|
110
|
-
case 'development':
|
|
111
|
-
return [...basePlugins, vitePlugin, swaggerPlugin, monitoringPlugin]
|
|
112
|
-
case 'production':
|
|
113
|
-
return [...basePlugins, monitoringPlugin]
|
|
114
|
-
case 'test':
|
|
115
|
-
return [loggerPlugin] // Minimal plugins for testing
|
|
116
|
-
default:
|
|
117
|
-
return basePlugins
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
/**
|
|
122
|
-
* Get plugin by name
|
|
123
|
-
*/
|
|
124
|
-
export function getBuiltInPlugin(name: string) {
|
|
125
|
-
return builtInPlugins[name as keyof typeof builtInPlugins]
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
/**
|
|
129
|
-
* Check if a plugin is built-in
|
|
130
|
-
*/
|
|
131
|
-
export function isBuiltInPlugin(name: string): boolean {
|
|
132
|
-
return name in builtInPlugins
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
/**
|
|
136
|
-
* Get plugins by category
|
|
137
|
-
*/
|
|
138
|
-
export function getPluginsByCategory(category: keyof typeof pluginCategories) {
|
|
139
|
-
return pluginCategories[category] || []
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
export default builtInPlugins
|
|
@@ -1,180 +0,0 @@
|
|
|
1
|
-
import type { Plugin, PluginContext, RequestContext, ResponseContext, ErrorContext } from "../../types"
|
|
2
|
-
|
|
3
|
-
export const loggerPlugin: Plugin = {
|
|
4
|
-
name: "logger",
|
|
5
|
-
version: "1.0.0",
|
|
6
|
-
description: "Enhanced logging plugin for FluxStack with request/response logging",
|
|
7
|
-
author: "FluxStack Team",
|
|
8
|
-
priority: "highest", // Logger should run first
|
|
9
|
-
category: "core",
|
|
10
|
-
tags: ["logging", "monitoring"],
|
|
11
|
-
|
|
12
|
-
configSchema: {
|
|
13
|
-
type: "object",
|
|
14
|
-
properties: {
|
|
15
|
-
logRequests: {
|
|
16
|
-
type: "boolean",
|
|
17
|
-
description: "Enable request logging"
|
|
18
|
-
},
|
|
19
|
-
logResponses: {
|
|
20
|
-
type: "boolean",
|
|
21
|
-
description: "Enable response logging"
|
|
22
|
-
},
|
|
23
|
-
logErrors: {
|
|
24
|
-
type: "boolean",
|
|
25
|
-
description: "Enable error logging"
|
|
26
|
-
},
|
|
27
|
-
includeHeaders: {
|
|
28
|
-
type: "boolean",
|
|
29
|
-
description: "Include headers in request/response logs"
|
|
30
|
-
},
|
|
31
|
-
includeBody: {
|
|
32
|
-
type: "boolean",
|
|
33
|
-
description: "Include body in request/response logs"
|
|
34
|
-
},
|
|
35
|
-
slowRequestThreshold: {
|
|
36
|
-
type: "number",
|
|
37
|
-
minimum: 0,
|
|
38
|
-
description: "Threshold in ms to log slow requests"
|
|
39
|
-
}
|
|
40
|
-
},
|
|
41
|
-
additionalProperties: false
|
|
42
|
-
},
|
|
43
|
-
|
|
44
|
-
defaultConfig: {
|
|
45
|
-
logRequests: process.env.ENABLE_REQUEST_LOGS === 'true',
|
|
46
|
-
logResponses: process.env.ENABLE_REQUEST_LOGS === 'true',
|
|
47
|
-
logErrors: true,
|
|
48
|
-
includeHeaders: false,
|
|
49
|
-
includeBody: false,
|
|
50
|
-
slowRequestThreshold: 1000
|
|
51
|
-
},
|
|
52
|
-
|
|
53
|
-
setup: async (context: PluginContext) => {
|
|
54
|
-
context.logger.info("Enhanced logger plugin initialized", {
|
|
55
|
-
environment: context.config.app?.name || 'fluxstack',
|
|
56
|
-
logLevel: context.config.logging.level,
|
|
57
|
-
format: context.config.logging.format
|
|
58
|
-
})
|
|
59
|
-
},
|
|
60
|
-
|
|
61
|
-
onServerStart: async (context: PluginContext) => {
|
|
62
|
-
context.logger.info("Logger plugin: Server started", {
|
|
63
|
-
port: context.config.server.port,
|
|
64
|
-
host: context.config.server.host,
|
|
65
|
-
apiPrefix: context.config.server.apiPrefix
|
|
66
|
-
})
|
|
67
|
-
},
|
|
68
|
-
|
|
69
|
-
onServerStop: async (context: PluginContext) => {
|
|
70
|
-
context.logger.info("Logger plugin: Server stopped")
|
|
71
|
-
},
|
|
72
|
-
|
|
73
|
-
onRequest: async (context: RequestContext) => {
|
|
74
|
-
const config = getPluginConfig(context)
|
|
75
|
-
|
|
76
|
-
if (!config.logRequests) return
|
|
77
|
-
|
|
78
|
-
const logData: any = {
|
|
79
|
-
method: context.method,
|
|
80
|
-
path: context.path,
|
|
81
|
-
userAgent: context.headers['user-agent'],
|
|
82
|
-
ip: context.headers['x-forwarded-for'] || context.headers['x-real-ip'] || 'unknown'
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
if (config.includeHeaders) {
|
|
86
|
-
logData.headers = context.headers
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
if (config.includeBody && context.body) {
|
|
90
|
-
logData.body = context.body
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
// Use a logger from context if available, otherwise create one
|
|
94
|
-
const logger = (context as any).logger || console
|
|
95
|
-
if (typeof logger.info === 'function') {
|
|
96
|
-
logger.info(`→ ${context.method} ${context.path}`, logData)
|
|
97
|
-
}
|
|
98
|
-
},
|
|
99
|
-
|
|
100
|
-
onResponse: async (context: ResponseContext) => {
|
|
101
|
-
const config = getPluginConfig(context)
|
|
102
|
-
|
|
103
|
-
if (!config.logResponses) return
|
|
104
|
-
|
|
105
|
-
const logData: any = {
|
|
106
|
-
method: context.method,
|
|
107
|
-
path: context.path,
|
|
108
|
-
statusCode: context.statusCode,
|
|
109
|
-
duration: context.duration,
|
|
110
|
-
size: context.size
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
if (config.includeHeaders) {
|
|
114
|
-
const headers: Record<string, string> = {}
|
|
115
|
-
context.response.headers.forEach((value, key) => {
|
|
116
|
-
headers[key] = value
|
|
117
|
-
})
|
|
118
|
-
logData.responseHeaders = headers
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
// Determine log level based on status code and duration
|
|
122
|
-
let logLevel = 'info'
|
|
123
|
-
if (context.statusCode >= 400) {
|
|
124
|
-
logLevel = 'warn'
|
|
125
|
-
}
|
|
126
|
-
if (context.statusCode >= 500) {
|
|
127
|
-
logLevel = 'error'
|
|
128
|
-
}
|
|
129
|
-
if (context.duration > config.slowRequestThreshold) {
|
|
130
|
-
logLevel = 'warn'
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
const logger = (context as any).logger || console
|
|
134
|
-
const logMessage = `← ${context.method} ${context.path} ${context.statusCode} ${context.duration}ms`
|
|
135
|
-
|
|
136
|
-
if (typeof logger[logLevel] === 'function') {
|
|
137
|
-
logger[logLevel](logMessage, logData)
|
|
138
|
-
}
|
|
139
|
-
},
|
|
140
|
-
|
|
141
|
-
onError: async (context: ErrorContext) => {
|
|
142
|
-
const config = getPluginConfig(context)
|
|
143
|
-
|
|
144
|
-
if (!config.logErrors) return
|
|
145
|
-
|
|
146
|
-
// Skip logging for NOT_FOUND errors unless explicitly enabled
|
|
147
|
-
if (context.error.message === 'NOT_FOUND' && !process.env.ENABLE_NOT_FOUND_LOGS) {
|
|
148
|
-
return
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
const logData: any = {
|
|
152
|
-
method: context.method,
|
|
153
|
-
path: context.path,
|
|
154
|
-
duration: context.duration,
|
|
155
|
-
error: {
|
|
156
|
-
name: context.error.name,
|
|
157
|
-
message: context.error.message,
|
|
158
|
-
stack: context.error.stack
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
if (config.includeHeaders) {
|
|
163
|
-
logData.headers = context.headers
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
const logger = (context as any).logger || console
|
|
167
|
-
if (typeof logger.error === 'function') {
|
|
168
|
-
logger.error(`✗ ${context.method} ${context.path} - ${context.error.message}`, logData)
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
// Helper function to get plugin config from context
|
|
174
|
-
function getPluginConfig(_context: any) {
|
|
175
|
-
// In a real implementation, this would get the config from the plugin context
|
|
176
|
-
// For now, return default config
|
|
177
|
-
return loggerPlugin.defaultConfig || {}
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
export default loggerPlugin
|