create-fluxstack 1.0.0 → 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.
Files changed (101) hide show
  1. package/create-fluxstack.ts +32 -17
  2. package/package-template.json +51 -0
  3. package/package.json +2 -1
  4. package/.env +0 -30
  5. package/LICENSE +0 -21
  6. package/app/client/README.md +0 -69
  7. package/app/client/frontend-only.ts +0 -12
  8. package/app/client/index.html +0 -13
  9. package/app/client/public/vite.svg +0 -1
  10. package/app/client/src/App.css +0 -883
  11. package/app/client/src/App.tsx +0 -669
  12. package/app/client/src/assets/react.svg +0 -1
  13. package/app/client/src/components/TestPage.tsx +0 -453
  14. package/app/client/src/index.css +0 -51
  15. package/app/client/src/lib/eden-api.ts +0 -110
  16. package/app/client/src/main.tsx +0 -10
  17. package/app/client/src/vite-env.d.ts +0 -1
  18. package/app/client/tsconfig.app.json +0 -43
  19. package/app/client/tsconfig.json +0 -7
  20. package/app/client/tsconfig.node.json +0 -25
  21. package/app/server/app.ts +0 -10
  22. package/app/server/backend-only.ts +0 -15
  23. package/app/server/controllers/users.controller.ts +0 -69
  24. package/app/server/index.ts +0 -104
  25. package/app/server/routes/index.ts +0 -25
  26. package/app/server/routes/users.routes.ts +0 -121
  27. package/app/server/types/index.ts +0 -1
  28. package/app/shared/types/index.ts +0 -18
  29. package/bun.lock +0 -1053
  30. package/core/__tests__/integration.test.ts +0 -227
  31. package/core/build/index.ts +0 -186
  32. package/core/cli/command-registry.ts +0 -334
  33. package/core/cli/index.ts +0 -394
  34. package/core/cli/plugin-discovery.ts +0 -200
  35. package/core/client/standalone.ts +0 -57
  36. package/core/config/__tests__/config-loader.test.ts +0 -591
  37. package/core/config/__tests__/config-merger.test.ts +0 -657
  38. package/core/config/__tests__/env-converter.test.ts +0 -372
  39. package/core/config/__tests__/env-processor.test.ts +0 -431
  40. package/core/config/__tests__/env.test.ts +0 -452
  41. package/core/config/__tests__/integration.test.ts +0 -418
  42. package/core/config/__tests__/loader.test.ts +0 -331
  43. package/core/config/__tests__/schema.test.ts +0 -129
  44. package/core/config/__tests__/validator.test.ts +0 -318
  45. package/core/config/env-dynamic.ts +0 -326
  46. package/core/config/env.ts +0 -597
  47. package/core/config/index.ts +0 -317
  48. package/core/config/loader.ts +0 -546
  49. package/core/config/runtime-config.ts +0 -322
  50. package/core/config/schema.ts +0 -694
  51. package/core/config/validator.ts +0 -540
  52. package/core/framework/__tests__/server.test.ts +0 -233
  53. package/core/framework/client.ts +0 -132
  54. package/core/framework/index.ts +0 -8
  55. package/core/framework/server.ts +0 -501
  56. package/core/framework/types.ts +0 -63
  57. package/core/plugins/__tests__/built-in.test.ts.disabled +0 -366
  58. package/core/plugins/__tests__/manager.test.ts +0 -398
  59. package/core/plugins/__tests__/monitoring.test.ts +0 -401
  60. package/core/plugins/__tests__/registry.test.ts +0 -335
  61. package/core/plugins/built-in/index.ts +0 -142
  62. package/core/plugins/built-in/logger/index.ts +0 -180
  63. package/core/plugins/built-in/monitoring/README.md +0 -193
  64. package/core/plugins/built-in/monitoring/index.ts +0 -912
  65. package/core/plugins/built-in/static/index.ts +0 -289
  66. package/core/plugins/built-in/swagger/index.ts +0 -229
  67. package/core/plugins/built-in/vite/index.ts +0 -316
  68. package/core/plugins/config.ts +0 -348
  69. package/core/plugins/discovery.ts +0 -350
  70. package/core/plugins/executor.ts +0 -351
  71. package/core/plugins/index.ts +0 -195
  72. package/core/plugins/manager.ts +0 -583
  73. package/core/plugins/registry.ts +0 -424
  74. package/core/plugins/types.ts +0 -254
  75. package/core/server/framework.ts +0 -123
  76. package/core/server/index.ts +0 -8
  77. package/core/server/plugins/database.ts +0 -182
  78. package/core/server/plugins/logger.ts +0 -47
  79. package/core/server/plugins/swagger.ts +0 -34
  80. package/core/server/standalone.ts +0 -91
  81. package/core/templates/create-project.ts +0 -455
  82. package/core/types/api.ts +0 -169
  83. package/core/types/build.ts +0 -174
  84. package/core/types/config.ts +0 -68
  85. package/core/types/index.ts +0 -127
  86. package/core/types/plugin.ts +0 -94
  87. package/core/utils/__tests__/errors.test.ts +0 -139
  88. package/core/utils/__tests__/helpers.test.ts +0 -297
  89. package/core/utils/__tests__/logger.test.ts +0 -141
  90. package/core/utils/env-runtime-v2.ts +0 -232
  91. package/core/utils/env-runtime.ts +0 -252
  92. package/core/utils/errors/codes.ts +0 -115
  93. package/core/utils/errors/handlers.ts +0 -63
  94. package/core/utils/errors/index.ts +0 -81
  95. package/core/utils/helpers.ts +0 -180
  96. package/core/utils/index.ts +0 -18
  97. package/core/utils/logger/index.ts +0 -161
  98. package/core/utils/logger.ts +0 -106
  99. package/core/utils/monitoring/index.ts +0 -212
  100. package/tsconfig.json +0 -51
  101. 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