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,334 +0,0 @@
|
|
|
1
|
-
import type { CliCommand, CliContext, CliArgument, CliOption } from "../plugins/types"
|
|
2
|
-
import { getConfigSync } from "../config"
|
|
3
|
-
import { logger } from "../utils/logger"
|
|
4
|
-
import { createTimer, formatBytes, isProduction, isDevelopment } from "../utils/helpers"
|
|
5
|
-
|
|
6
|
-
export class CliCommandRegistry {
|
|
7
|
-
private commands = new Map<string, CliCommand>()
|
|
8
|
-
private aliases = new Map<string, string>()
|
|
9
|
-
private context: CliContext
|
|
10
|
-
|
|
11
|
-
constructor() {
|
|
12
|
-
const config = getConfigSync()
|
|
13
|
-
|
|
14
|
-
this.context = {
|
|
15
|
-
config,
|
|
16
|
-
logger: {
|
|
17
|
-
debug: (message: string, meta?: any) => logger.debug(message, meta),
|
|
18
|
-
info: (message: string, meta?: any) => logger.info(message, meta),
|
|
19
|
-
warn: (message: string, meta?: any) => logger.warn(message, meta),
|
|
20
|
-
error: (message: string, meta?: any) => logger.error(message, meta),
|
|
21
|
-
child: (context: any) => (logger as any).child(context),
|
|
22
|
-
time: (label: string) => (logger as any).time(label),
|
|
23
|
-
timeEnd: (label: string) => (logger as any).timeEnd(label),
|
|
24
|
-
request: (method: string, path: string, status?: number, duration?: number) =>
|
|
25
|
-
logger.request(method, path, status, duration)
|
|
26
|
-
},
|
|
27
|
-
utils: {
|
|
28
|
-
createTimer,
|
|
29
|
-
formatBytes,
|
|
30
|
-
isProduction,
|
|
31
|
-
isDevelopment,
|
|
32
|
-
getEnvironment: () => process.env.NODE_ENV || 'development',
|
|
33
|
-
createHash: (data: string) => {
|
|
34
|
-
const crypto = require('crypto')
|
|
35
|
-
return crypto.createHash('sha256').update(data).digest('hex')
|
|
36
|
-
},
|
|
37
|
-
deepMerge: (target: any, source: any) => {
|
|
38
|
-
const result = { ...target }
|
|
39
|
-
for (const key in source) {
|
|
40
|
-
if (source[key] && typeof source[key] === 'object' && !Array.isArray(source[key])) {
|
|
41
|
-
result[key] = this.context.utils.deepMerge(result[key] || {}, source[key])
|
|
42
|
-
} else {
|
|
43
|
-
result[key] = source[key]
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
|
-
return result
|
|
47
|
-
},
|
|
48
|
-
validateSchema: (_data: any, _schema: any) => {
|
|
49
|
-
try {
|
|
50
|
-
return { valid: true, errors: [] }
|
|
51
|
-
} catch (error) {
|
|
52
|
-
return { valid: false, errors: [error instanceof Error ? error.message : 'Validation failed'] }
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
},
|
|
56
|
-
workingDir: process.cwd(),
|
|
57
|
-
packageInfo: {
|
|
58
|
-
name: 'fluxstack',
|
|
59
|
-
version: '1.0.0'
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
register(command: CliCommand): void {
|
|
65
|
-
// Register main command
|
|
66
|
-
this.commands.set(command.name, command)
|
|
67
|
-
|
|
68
|
-
// Register aliases
|
|
69
|
-
if (command.aliases) {
|
|
70
|
-
for (const alias of command.aliases) {
|
|
71
|
-
this.aliases.set(alias, command.name)
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
get(name: string): CliCommand | undefined {
|
|
77
|
-
// Check direct command
|
|
78
|
-
const command = this.commands.get(name)
|
|
79
|
-
if (command) return command
|
|
80
|
-
|
|
81
|
-
// Check alias
|
|
82
|
-
const aliasTarget = this.aliases.get(name)
|
|
83
|
-
if (aliasTarget) {
|
|
84
|
-
return this.commands.get(aliasTarget)
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
return undefined
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
has(name: string): boolean {
|
|
91
|
-
return this.commands.has(name) || this.aliases.has(name)
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
getAll(): CliCommand[] {
|
|
95
|
-
return Array.from(this.commands.values())
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
getAllByCategory(): Map<string, CliCommand[]> {
|
|
99
|
-
const categories = new Map<string, CliCommand[]>()
|
|
100
|
-
|
|
101
|
-
for (const command of this.commands.values()) {
|
|
102
|
-
if (command.hidden) continue
|
|
103
|
-
|
|
104
|
-
const category = command.category || 'General'
|
|
105
|
-
if (!categories.has(category)) {
|
|
106
|
-
categories.set(category, [])
|
|
107
|
-
}
|
|
108
|
-
categories.get(category)!.push(command)
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
return categories
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
async execute(commandName: string, args: string[]): Promise<number> {
|
|
115
|
-
const command = this.get(commandName)
|
|
116
|
-
|
|
117
|
-
if (!command) {
|
|
118
|
-
console.error(`❌ Unknown command: ${commandName}`)
|
|
119
|
-
this.showHelp()
|
|
120
|
-
return 1
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
try {
|
|
124
|
-
// Parse arguments and options
|
|
125
|
-
const { parsedArgs, parsedOptions } = this.parseArgs(command, args)
|
|
126
|
-
|
|
127
|
-
// Validate required arguments
|
|
128
|
-
if (command.arguments) {
|
|
129
|
-
for (let i = 0; i < command.arguments.length; i++) {
|
|
130
|
-
const arg = command.arguments[i]
|
|
131
|
-
if (arg.required && !parsedArgs[i]) {
|
|
132
|
-
console.error(`❌ Missing required argument: ${arg.name}`)
|
|
133
|
-
this.showCommandHelp(command)
|
|
134
|
-
return 1
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
// Validate required options
|
|
140
|
-
if (command.options) {
|
|
141
|
-
for (const option of command.options) {
|
|
142
|
-
if (option.required && !(option.name in parsedOptions)) {
|
|
143
|
-
console.error(`❌ Missing required option: --${option.name}`)
|
|
144
|
-
this.showCommandHelp(command)
|
|
145
|
-
return 1
|
|
146
|
-
}
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
// Execute command
|
|
151
|
-
await command.handler(parsedArgs, parsedOptions, this.context)
|
|
152
|
-
return 0
|
|
153
|
-
|
|
154
|
-
} catch (error) {
|
|
155
|
-
console.error(`❌ Command failed:`, error instanceof Error ? error.message : String(error))
|
|
156
|
-
return 1
|
|
157
|
-
}
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
private parseArgs(command: CliCommand, args: string[]): { parsedArgs: any[], parsedOptions: any } {
|
|
161
|
-
const parsedArgs: any[] = []
|
|
162
|
-
const parsedOptions: any = {}
|
|
163
|
-
|
|
164
|
-
let i = 0
|
|
165
|
-
while (i < args.length) {
|
|
166
|
-
const arg = args[i]
|
|
167
|
-
|
|
168
|
-
// Handle options (--name or -n)
|
|
169
|
-
if (arg.startsWith('--')) {
|
|
170
|
-
const optionName = arg.slice(2)
|
|
171
|
-
const option = command.options?.find(o => o.name === optionName)
|
|
172
|
-
|
|
173
|
-
if (option) {
|
|
174
|
-
if (option.type === 'boolean') {
|
|
175
|
-
parsedOptions[optionName] = true
|
|
176
|
-
} else if (option.type === 'array') {
|
|
177
|
-
parsedOptions[optionName] = parsedOptions[optionName] || []
|
|
178
|
-
if (i + 1 < args.length && !args[i + 1].startsWith('-')) {
|
|
179
|
-
parsedOptions[optionName].push(args[++i])
|
|
180
|
-
}
|
|
181
|
-
} else {
|
|
182
|
-
if (i + 1 < args.length && !args[i + 1].startsWith('-')) {
|
|
183
|
-
parsedOptions[optionName] = this.convertType(args[++i], option.type)
|
|
184
|
-
}
|
|
185
|
-
}
|
|
186
|
-
}
|
|
187
|
-
}
|
|
188
|
-
// Handle short options (-n)
|
|
189
|
-
else if (arg.startsWith('-') && arg.length === 2) {
|
|
190
|
-
const shortName = arg.slice(1)
|
|
191
|
-
const option = command.options?.find(o => o.short === shortName)
|
|
192
|
-
|
|
193
|
-
if (option) {
|
|
194
|
-
if (option.type === 'boolean') {
|
|
195
|
-
parsedOptions[option.name] = true
|
|
196
|
-
} else {
|
|
197
|
-
if (i + 1 < args.length && !args[i + 1].startsWith('-')) {
|
|
198
|
-
parsedOptions[option.name] = this.convertType(args[++i], option.type)
|
|
199
|
-
}
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
// Handle positional arguments
|
|
204
|
-
else {
|
|
205
|
-
const argIndex = parsedArgs.length
|
|
206
|
-
const argDef = command.arguments?.[argIndex]
|
|
207
|
-
|
|
208
|
-
if (argDef) {
|
|
209
|
-
parsedArgs.push(this.convertType(arg, argDef.type))
|
|
210
|
-
} else {
|
|
211
|
-
parsedArgs.push(arg)
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
i++
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
// Apply defaults
|
|
219
|
-
if (command.arguments) {
|
|
220
|
-
for (let i = 0; i < command.arguments.length; i++) {
|
|
221
|
-
if (parsedArgs[i] === undefined && command.arguments[i].default !== undefined) {
|
|
222
|
-
parsedArgs[i] = command.arguments[i].default
|
|
223
|
-
}
|
|
224
|
-
}
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
if (command.options) {
|
|
228
|
-
for (const option of command.options) {
|
|
229
|
-
if (!(option.name in parsedOptions) && option.default !== undefined) {
|
|
230
|
-
parsedOptions[option.name] = option.default
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
return { parsedArgs, parsedOptions }
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
private convertType(value: string, type?: 'string' | 'number' | 'boolean' | 'array'): any {
|
|
239
|
-
if (!type || type === 'string') return value
|
|
240
|
-
if (type === 'number') return Number(value)
|
|
241
|
-
if (type === 'boolean') return value.toLowerCase() === 'true'
|
|
242
|
-
if (type === 'array') return [value] // Convert single value to array
|
|
243
|
-
return value
|
|
244
|
-
}
|
|
245
|
-
|
|
246
|
-
showHelp(): void {
|
|
247
|
-
console.log(`
|
|
248
|
-
⚡ FluxStack Framework CLI
|
|
249
|
-
|
|
250
|
-
Usage:
|
|
251
|
-
flux <command> [options] [arguments]
|
|
252
|
-
fluxstack <command> [options] [arguments]
|
|
253
|
-
|
|
254
|
-
Built-in Commands:`)
|
|
255
|
-
|
|
256
|
-
const categories = this.getAllByCategory()
|
|
257
|
-
|
|
258
|
-
for (const [category, commands] of categories) {
|
|
259
|
-
console.log(`\n${category}:`)
|
|
260
|
-
for (const command of commands) {
|
|
261
|
-
const aliases = command.aliases?.length ? ` (${command.aliases.join(', ')})` : ''
|
|
262
|
-
console.log(` ${command.name}${aliases.padEnd(20)} ${command.description}`)
|
|
263
|
-
}
|
|
264
|
-
}
|
|
265
|
-
|
|
266
|
-
console.log(`
|
|
267
|
-
Examples:
|
|
268
|
-
flux dev # Start development server
|
|
269
|
-
flux build --production # Build for production
|
|
270
|
-
flux create my-app # Create new project
|
|
271
|
-
flux help <command> # Show help for specific command
|
|
272
|
-
|
|
273
|
-
Use "flux help <command>" for more information about a specific command.`)
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
showCommandHelp(command: CliCommand): void {
|
|
277
|
-
console.log(`\n${command.description}`)
|
|
278
|
-
|
|
279
|
-
if (command.usage) {
|
|
280
|
-
console.log(`\nUsage:\n ${command.usage}`)
|
|
281
|
-
} else {
|
|
282
|
-
let usage = `flux ${command.name}`
|
|
283
|
-
|
|
284
|
-
if (command.arguments) {
|
|
285
|
-
for (const arg of command.arguments) {
|
|
286
|
-
if (arg.required) {
|
|
287
|
-
usage += ` <${arg.name}>`
|
|
288
|
-
} else {
|
|
289
|
-
usage += ` [${arg.name}]`
|
|
290
|
-
}
|
|
291
|
-
}
|
|
292
|
-
}
|
|
293
|
-
|
|
294
|
-
if (command.options?.length) {
|
|
295
|
-
usage += ` [options]`
|
|
296
|
-
}
|
|
297
|
-
|
|
298
|
-
console.log(`\nUsage:\n ${usage}`)
|
|
299
|
-
}
|
|
300
|
-
|
|
301
|
-
if (command.arguments?.length) {
|
|
302
|
-
console.log(`\nArguments:`)
|
|
303
|
-
for (const arg of command.arguments) {
|
|
304
|
-
const required = arg.required ? ' (required)' : ''
|
|
305
|
-
const defaultValue = arg.default !== undefined ? ` (default: ${arg.default})` : ''
|
|
306
|
-
console.log(` ${arg.name.padEnd(15)} ${arg.description}${required}${defaultValue}`)
|
|
307
|
-
}
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
if (command.options?.length) {
|
|
311
|
-
console.log(`\nOptions:`)
|
|
312
|
-
for (const option of command.options) {
|
|
313
|
-
const short = option.short ? `-${option.short}, ` : ' '
|
|
314
|
-
const required = option.required ? ' (required)' : ''
|
|
315
|
-
const defaultValue = option.default !== undefined ? ` (default: ${option.default})` : ''
|
|
316
|
-
console.log(` ${short}--${option.name.padEnd(15)} ${option.description}${required}${defaultValue}`)
|
|
317
|
-
}
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
if (command.examples?.length) {
|
|
321
|
-
console.log(`\nExamples:`)
|
|
322
|
-
for (const example of command.examples) {
|
|
323
|
-
console.log(` ${example}`)
|
|
324
|
-
}
|
|
325
|
-
}
|
|
326
|
-
|
|
327
|
-
if (command.aliases?.length) {
|
|
328
|
-
console.log(`\nAliases: ${command.aliases.join(', ')}`)
|
|
329
|
-
}
|
|
330
|
-
}
|
|
331
|
-
}
|
|
332
|
-
|
|
333
|
-
// Global registry instance
|
|
334
|
-
export const cliRegistry = new CliCommandRegistry()
|