@navios/commander 0.5.0 → 0.5.1
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/dist/src/services/cli-parser.service.d.mts +10 -0
- package/dist/src/services/cli-parser.service.d.mts.map +1 -1
- package/dist/tsconfig.lib.tsbuildinfo +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/lib/_tsup-dts-rollup.d.mts +10 -0
- package/lib/_tsup-dts-rollup.d.ts +10 -0
- package/lib/index.js +64 -1
- package/lib/index.js.map +1 -1
- package/lib/index.mjs +64 -1
- package/lib/index.mjs.map +1 -1
- package/package.json +1 -1
- package/src/services/__tests__/cli-parser.service.spec.mts +251 -0
- package/src/services/cli-parser.service.mts +81 -2
|
@@ -152,11 +152,21 @@ declare class CliParserService {
|
|
|
152
152
|
* Handles ZodObject, ZodOptional, and ZodDefault wrappers
|
|
153
153
|
*/
|
|
154
154
|
private extractBooleanFields;
|
|
155
|
+
/**
|
|
156
|
+
* Extracts array field names from a Zod schema
|
|
157
|
+
* Handles ZodObject, ZodOptional, and ZodDefault wrappers
|
|
158
|
+
*/
|
|
159
|
+
private extractArrayFields;
|
|
155
160
|
/**
|
|
156
161
|
* Checks if a Zod schema represents a boolean type
|
|
157
162
|
* Unwraps ZodOptional and ZodDefault
|
|
158
163
|
*/
|
|
159
164
|
private isSchemaBoolean;
|
|
165
|
+
/**
|
|
166
|
+
* Checks if a Zod schema represents an array type
|
|
167
|
+
* Unwraps ZodOptional and ZodDefault
|
|
168
|
+
*/
|
|
169
|
+
private isSchemaArray;
|
|
160
170
|
/**
|
|
161
171
|
* Formats help text for available commands
|
|
162
172
|
*/
|
|
@@ -152,11 +152,21 @@ declare class CliParserService {
|
|
|
152
152
|
* Handles ZodObject, ZodOptional, and ZodDefault wrappers
|
|
153
153
|
*/
|
|
154
154
|
private extractBooleanFields;
|
|
155
|
+
/**
|
|
156
|
+
* Extracts array field names from a Zod schema
|
|
157
|
+
* Handles ZodObject, ZodOptional, and ZodDefault wrappers
|
|
158
|
+
*/
|
|
159
|
+
private extractArrayFields;
|
|
155
160
|
/**
|
|
156
161
|
* Checks if a Zod schema represents a boolean type
|
|
157
162
|
* Unwraps ZodOptional and ZodDefault
|
|
158
163
|
*/
|
|
159
164
|
private isSchemaBoolean;
|
|
165
|
+
/**
|
|
166
|
+
* Checks if a Zod schema represents an array type
|
|
167
|
+
* Unwraps ZodOptional and ZodDefault
|
|
168
|
+
*/
|
|
169
|
+
private isSchemaArray;
|
|
160
170
|
/**
|
|
161
171
|
* Formats help text for available commands
|
|
162
172
|
*/
|
package/lib/index.js
CHANGED
|
@@ -210,6 +210,7 @@ exports.CliParserService = class CliParserService {
|
|
|
210
210
|
throw new Error("[Navios Commander] No command provided");
|
|
211
211
|
}
|
|
212
212
|
const booleanFields = optionsSchema ? this.extractBooleanFields(optionsSchema) : /* @__PURE__ */ new Set();
|
|
213
|
+
const arrayFields = optionsSchema ? this.extractArrayFields(optionsSchema) : /* @__PURE__ */ new Set();
|
|
213
214
|
const commandParts = [];
|
|
214
215
|
let i = 0;
|
|
215
216
|
while (i < args.length && !args[i].startsWith("-")) {
|
|
@@ -230,15 +231,31 @@ exports.CliParserService = class CliParserService {
|
|
|
230
231
|
if (equalIndex !== -1) {
|
|
231
232
|
const optionName = key.slice(0, equalIndex);
|
|
232
233
|
const optionValue = key.slice(equalIndex + 1);
|
|
233
|
-
|
|
234
|
+
const camelCaseKey = this.camelCase(optionName);
|
|
235
|
+
const isArray = arrayFields.has(camelCaseKey) || arrayFields.has(optionName);
|
|
236
|
+
if (isArray) {
|
|
237
|
+
if (!options[camelCaseKey]) {
|
|
238
|
+
options[camelCaseKey] = [];
|
|
239
|
+
}
|
|
240
|
+
options[camelCaseKey].push(this.parseValue(optionValue));
|
|
241
|
+
} else {
|
|
242
|
+
options[camelCaseKey] = this.parseValue(optionValue);
|
|
243
|
+
}
|
|
234
244
|
i++;
|
|
235
245
|
} else {
|
|
236
246
|
const camelCaseKey = this.camelCase(key);
|
|
237
247
|
const isBoolean = booleanFields.has(camelCaseKey) || booleanFields.has(key);
|
|
248
|
+
const isArray = arrayFields.has(camelCaseKey) || arrayFields.has(key);
|
|
238
249
|
const nextArg = args[i + 1];
|
|
239
250
|
if (isBoolean) {
|
|
240
251
|
options[camelCaseKey] = true;
|
|
241
252
|
i++;
|
|
253
|
+
} else if (isArray && nextArg && !nextArg.startsWith("-")) {
|
|
254
|
+
if (!options[camelCaseKey]) {
|
|
255
|
+
options[camelCaseKey] = [];
|
|
256
|
+
}
|
|
257
|
+
options[camelCaseKey].push(this.parseValue(nextArg));
|
|
258
|
+
i += 2;
|
|
242
259
|
} else if (nextArg && !nextArg.startsWith("-")) {
|
|
243
260
|
options[camelCaseKey] = this.parseValue(nextArg);
|
|
244
261
|
i += 2;
|
|
@@ -251,10 +268,17 @@ exports.CliParserService = class CliParserService {
|
|
|
251
268
|
const flags = arg.slice(1);
|
|
252
269
|
if (flags.length === 1) {
|
|
253
270
|
const isBoolean = booleanFields.has(flags);
|
|
271
|
+
const isArray = arrayFields.has(flags);
|
|
254
272
|
const nextArg = args[i + 1];
|
|
255
273
|
if (isBoolean) {
|
|
256
274
|
options[flags] = true;
|
|
257
275
|
i++;
|
|
276
|
+
} else if (isArray && nextArg && !nextArg.startsWith("-")) {
|
|
277
|
+
if (!options[flags]) {
|
|
278
|
+
options[flags] = [];
|
|
279
|
+
}
|
|
280
|
+
options[flags].push(this.parseValue(nextArg));
|
|
281
|
+
i += 2;
|
|
258
282
|
} else if (nextArg && !nextArg.startsWith("-")) {
|
|
259
283
|
options[flags] = this.parseValue(nextArg);
|
|
260
284
|
i += 2;
|
|
@@ -330,6 +354,28 @@ exports.CliParserService = class CliParserService {
|
|
|
330
354
|
}
|
|
331
355
|
return booleanFields;
|
|
332
356
|
}
|
|
357
|
+
/**
|
|
358
|
+
* Extracts array field names from a Zod schema
|
|
359
|
+
* Handles ZodObject, ZodOptional, and ZodDefault wrappers
|
|
360
|
+
*/
|
|
361
|
+
extractArrayFields(schema) {
|
|
362
|
+
const arrayFields = /* @__PURE__ */ new Set();
|
|
363
|
+
try {
|
|
364
|
+
const typeName = schema.def.type;
|
|
365
|
+
if (typeName === "object") {
|
|
366
|
+
const shape = schema.def.shape;
|
|
367
|
+
if (shape && typeof shape === "object") {
|
|
368
|
+
for (const [key, fieldSchema] of Object.entries(shape)) {
|
|
369
|
+
if (this.isSchemaArray(fieldSchema)) {
|
|
370
|
+
arrayFields.add(key);
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
} catch {
|
|
376
|
+
}
|
|
377
|
+
return arrayFields;
|
|
378
|
+
}
|
|
333
379
|
/**
|
|
334
380
|
* Checks if a Zod schema represents a boolean type
|
|
335
381
|
* Unwraps ZodOptional and ZodDefault
|
|
@@ -347,6 +393,23 @@ exports.CliParserService = class CliParserService {
|
|
|
347
393
|
return false;
|
|
348
394
|
}
|
|
349
395
|
}
|
|
396
|
+
/**
|
|
397
|
+
* Checks if a Zod schema represents an array type
|
|
398
|
+
* Unwraps ZodOptional and ZodDefault
|
|
399
|
+
*/
|
|
400
|
+
isSchemaArray(schema) {
|
|
401
|
+
try {
|
|
402
|
+
let currentSchema = schema;
|
|
403
|
+
const typeName = currentSchema.def.type;
|
|
404
|
+
if (typeName === "optional" || typeName === "default") {
|
|
405
|
+
currentSchema = currentSchema?._def?.innerType || currentSchema;
|
|
406
|
+
}
|
|
407
|
+
const innerTypeName = currentSchema.def.type;
|
|
408
|
+
return innerTypeName === "array";
|
|
409
|
+
} catch {
|
|
410
|
+
return false;
|
|
411
|
+
}
|
|
412
|
+
}
|
|
350
413
|
/**
|
|
351
414
|
* Formats help text for available commands
|
|
352
415
|
*/
|
package/lib/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/metadata/command.metadata.mts","../src/metadata/cli-module.metadata.mts","../src/services/module-loader.service.mts","../src/services/cli-parser.service.mts","../src/commander.application.mts","../src/commander.factory.mts","../src/decorators/command.decorator.mts","../src/decorators/cli-module.decorator.mts"],"names":["Injectable","ModuleLoaderService","inject","Container","_init","CliParserService","CommanderApplication","InjectionToken","InjectableScope"],"mappings":"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","file":"index.js","sourcesContent":["import type { ClassType } from '@navios/di'\nimport type { ZodObject } from 'zod'\n\nexport const CommandMetadataKey = Symbol('CommandMetadataKey')\n\nexport interface CommandMetadata {\n path: string\n optionsSchema?: ZodObject\n customAttributes: Map<string | symbol, any>\n}\n\nexport function getCommandMetadata(\n target: ClassType,\n context: ClassDecoratorContext,\n path: string,\n optionsSchema?: ZodObject,\n): CommandMetadata {\n if (context.metadata) {\n const metadata = context.metadata[CommandMetadataKey] as\n | CommandMetadata\n | undefined\n if (metadata) {\n return metadata\n } else {\n const newMetadata: CommandMetadata = {\n path,\n optionsSchema,\n customAttributes: new Map<string | symbol, any>(),\n }\n context.metadata[CommandMetadataKey] = newMetadata\n // @ts-expect-error We add a custom metadata key to the target\n target[CommandMetadataKey] = newMetadata\n return newMetadata\n }\n }\n throw new Error('[Navios Commander] Wrong environment.')\n}\n\nexport function extractCommandMetadata(target: ClassType): CommandMetadata {\n // @ts-expect-error We add a custom metadata key to the target\n const metadata = target[CommandMetadataKey] as CommandMetadata | undefined\n if (!metadata) {\n throw new Error(\n '[Navios Commander] Command metadata not found. Make sure to use @Command decorator.',\n )\n }\n return metadata\n}\n\nexport function hasCommandMetadata(target: ClassType): boolean {\n // @ts-expect-error We add a custom metadata key to the target\n const metadata = target[CommandMetadataKey] as CommandMetadata | undefined\n return !!metadata\n}\n","import type { ClassType } from '@navios/di'\n\nexport const CliModuleMetadataKey = Symbol('CliModuleMetadataKey')\n\nexport interface CliModuleMetadata {\n commands: Set<ClassType>\n imports: Set<ClassType>\n customAttributes: Map<string | symbol, any>\n}\n\nexport function getCliModuleMetadata(\n target: ClassType,\n context: ClassDecoratorContext,\n): CliModuleMetadata {\n if (context.metadata) {\n const metadata = context.metadata[CliModuleMetadataKey] as\n | CliModuleMetadata\n | undefined\n if (metadata) {\n return metadata\n } else {\n const newMetadata: CliModuleMetadata = {\n commands: new Set<ClassType>(),\n imports: new Set<ClassType>(),\n customAttributes: new Map<string | symbol, any>(),\n }\n context.metadata[CliModuleMetadataKey] = newMetadata\n // @ts-expect-error We add a custom metadata key to the target\n target[CliModuleMetadataKey] = newMetadata\n return newMetadata\n }\n }\n throw new Error('[Navios Commander] Wrong environment.')\n}\n\nexport function extractCliModuleMetadata(\n target: ClassType,\n): CliModuleMetadata {\n // @ts-expect-error We add a custom metadata key to the target\n const metadata = target[CliModuleMetadataKey] as\n | CliModuleMetadata\n | undefined\n if (!metadata) {\n throw new Error(\n `[Navios Commander] Module metadata not found for ${target.name}. Make sure to use @CliModule decorator.`,\n )\n }\n return metadata\n}\n\nexport function hasCliModuleMetadata(target: ClassType): boolean {\n // @ts-expect-error We add a custom metadata key to the target\n return !!target[CliModuleMetadataKey]\n}\n","import type { ClassTypeWithInstance } from '@navios/di'\n\nimport { Container, inject, Injectable } from '@navios/di'\n\nimport type { CommandHandler, Module } from '../interfaces/index.mjs'\nimport type { CliModuleMetadata, CommandMetadata } from '../metadata/index.mjs'\n\nimport {\n extractCliModuleMetadata,\n extractCommandMetadata,\n} from '../metadata/index.mjs'\n\nexport interface CommandWithMetadata {\n class: ClassTypeWithInstance<CommandHandler>\n metadata: CommandMetadata\n}\n\n@Injectable()\nexport class ModuleLoaderService {\n protected container = inject(Container)\n private modulesMetadata: Map<string, CliModuleMetadata> = new Map()\n private loadedModules: Map<string, any> = new Map()\n private commandsMetadata: Map<string, CommandWithMetadata> = new Map()\n private initialized = false\n\n async loadModules(appModule: ClassTypeWithInstance<Module>) {\n if (this.initialized) {\n return\n }\n await this.traverseModules(appModule)\n this.initialized = true\n }\n\n private async traverseModules(\n module: ClassTypeWithInstance<Module>,\n parentMetadata?: CliModuleMetadata,\n ) {\n const metadata = extractCliModuleMetadata(module)\n if (parentMetadata) {\n this.mergeMetadata(metadata, parentMetadata)\n }\n const moduleName = module.name\n if (this.modulesMetadata.has(moduleName)) {\n return\n }\n this.modulesMetadata.set(moduleName, metadata)\n\n // Collect command metadata during module loading\n for (const command of metadata.commands) {\n const commandMetadata = extractCommandMetadata(command)\n this.commandsMetadata.set(commandMetadata.path, {\n class: command,\n metadata: commandMetadata,\n })\n }\n\n const imports = metadata.imports ?? new Set()\n const loadingPromises = Array.from(imports).map(async (importedModule) =>\n this.traverseModules(importedModule, metadata),\n )\n await Promise.all(loadingPromises)\n const instance = await this.container.get(module)\n if (instance.onModuleInit) {\n await instance.onModuleInit()\n }\n this.loadedModules.set(moduleName, instance)\n }\n\n private mergeMetadata(\n metadata: CliModuleMetadata,\n parentMetadata: CliModuleMetadata,\n ): void {\n if (parentMetadata.customAttributes) {\n for (const [key, value] of parentMetadata.customAttributes) {\n if (metadata.customAttributes.has(key)) {\n continue\n }\n metadata.customAttributes.set(key, value)\n }\n }\n }\n\n getAllModules(): Map<string, CliModuleMetadata> {\n return this.modulesMetadata\n }\n\n getAllCommands(): Map<string, ClassTypeWithInstance<any>> {\n const commands = new Map<string, ClassTypeWithInstance<any>>()\n for (const metadata of this.modulesMetadata.values()) {\n for (const command of metadata.commands) {\n commands.set(command.name, command)\n }\n }\n return commands\n }\n\n /**\n * Get all commands with their metadata, indexed by command path.\n * This is populated during loadModules, so path information is available\n * before parsing CLI argv.\n */\n getAllCommandsWithMetadata(): Map<string, CommandWithMetadata> {\n return this.commandsMetadata\n }\n\n /**\n * Get a command by its path, with metadata already extracted.\n * Returns undefined if command is not found.\n */\n getCommandByPath(path: string): CommandWithMetadata | undefined {\n return this.commandsMetadata.get(path)\n }\n\n dispose() {\n this.modulesMetadata.clear()\n this.loadedModules.clear()\n this.commandsMetadata.clear()\n this.initialized = false\n }\n}\n","import type { ZodObject, ZodType } from 'zod'\n\nimport { Injectable } from '@navios/di'\n\nexport interface ParsedCliArgs {\n command: string\n options: Record<string, any>\n positionals: string[]\n}\n\n@Injectable()\nexport class CliParserService {\n /**\n * Parses command-line arguments from process.argv\n * Commands can be multi-word (e.g., 'db migrate', 'cache clear')\n * Expected format: node script.js command [subcommand...] --flag value --boolean-flag positional1 positional2\n *\n * @param argv - Array of command-line arguments (typically process.argv)\n * @param optionsSchema - Optional Zod schema to determine boolean flags and option types\n * @returns Parsed command (space-separated if multi-word), options, and positional arguments\n */\n parse(argv: string[], optionsSchema?: ZodObject): ParsedCliArgs {\n // Skip first two args (node and script path)\n const args = argv.slice(2)\n\n if (args.length === 0) {\n throw new Error('[Navios Commander] No command provided')\n }\n\n // Extract boolean field names from schema for accurate parsing\n const booleanFields = optionsSchema\n ? this.extractBooleanFields(optionsSchema)\n : new Set<string>()\n\n // Collect command words until we hit an argument that starts with '-' or '--'\n const commandParts: string[] = []\n let i = 0\n while (i < args.length && !args[i].startsWith('-')) {\n commandParts.push(args[i])\n i++\n }\n\n if (commandParts.length === 0) {\n throw new Error('[Navios Commander] No command provided')\n }\n\n const command = commandParts.join(' ')\n const options: Record<string, any> = {}\n const positionals: string[] = []\n while (i < args.length) {\n const arg = args[i]\n\n if (arg.startsWith('--')) {\n // Long option format: --key=value or --key value\n const key = arg.slice(2)\n const equalIndex = key.indexOf('=')\n\n if (equalIndex !== -1) {\n // Format: --key=value\n const optionName = key.slice(0, equalIndex)\n const optionValue = key.slice(equalIndex + 1)\n options[this.camelCase(optionName)] = this.parseValue(optionValue)\n i++\n } else {\n // Format: --key value or --boolean-flag\n const camelCaseKey = this.camelCase(key)\n const isBoolean =\n booleanFields.has(camelCaseKey) || booleanFields.has(key)\n const nextArg = args[i + 1]\n\n if (isBoolean) {\n // Known boolean flag from schema\n options[camelCaseKey] = true\n i++\n } else if (nextArg && !nextArg.startsWith('-')) {\n // Has a value\n options[camelCaseKey] = this.parseValue(nextArg)\n i += 2\n } else {\n // Assume boolean flag\n options[camelCaseKey] = true\n i++\n }\n }\n } else if (arg.startsWith('-') && arg.length > 1 && arg !== '-') {\n // Short option format: -k value or -abc (multiple flags)\n const flags = arg.slice(1)\n\n if (flags.length === 1) {\n // Single short flag: -k value or -k\n const isBoolean = booleanFields.has(flags)\n const nextArg = args[i + 1]\n\n if (isBoolean) {\n // Known boolean flag from schema\n options[flags] = true\n i++\n } else if (nextArg && !nextArg.startsWith('-')) {\n options[flags] = this.parseValue(nextArg)\n i += 2\n } else {\n options[flags] = true\n i++\n }\n } else {\n // Multiple short flags: -abc -> {a: true, b: true, c: true}\n for (const flag of flags) {\n options[flag] = true\n }\n i++\n }\n } else {\n // Positional argument\n positionals.push(arg)\n i++\n }\n }\n\n return {\n command,\n options,\n positionals,\n }\n }\n\n /**\n * Converts kebab-case to camelCase\n */\n private camelCase(str: string): string {\n return str.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase())\n }\n\n /**\n * Attempts to parse string values into appropriate types\n */\n private parseValue(value: string): any {\n // Check for boolean\n if (value === 'true') return true\n if (value === 'false') return false\n\n // Check for null/undefined\n if (value === 'null') return null\n if (value === 'undefined') return undefined\n\n // Check for number\n if (/^-?\\d+$/.test(value)) {\n return parseInt(value, 10)\n }\n if (/^-?\\d+\\.\\d+$/.test(value)) {\n return parseFloat(value)\n }\n\n // Check for JSON\n if (\n (value.startsWith('{') && value.endsWith('}')) ||\n (value.startsWith('[') && value.endsWith(']'))\n ) {\n try {\n return JSON.parse(value)\n } catch {\n // If parsing fails, return as string\n return value\n }\n }\n\n // Return as string\n return value\n }\n\n /**\n * Extracts boolean field names from a Zod schema\n * Handles ZodObject, ZodOptional, and ZodDefault wrappers\n */\n private extractBooleanFields(schema: ZodObject): Set<string> {\n const booleanFields = new Set<string>()\n\n try {\n // Check if schema has _def.typeName (Zod schema structure)\n const typeName = schema.def.type\n\n if (typeName === 'object') {\n // Extract shape from ZodObject\n const shape = schema.def.shape\n\n if (shape && typeof shape === 'object') {\n for (const [key, fieldSchema] of Object.entries(shape)) {\n if (this.isSchemaBoolean(fieldSchema as any)) {\n booleanFields.add(key)\n }\n }\n }\n }\n } catch {\n // Silently fail if schema introspection fails\n }\n\n return booleanFields\n }\n\n /**\n * Checks if a Zod schema represents a boolean type\n * Unwraps ZodOptional and ZodDefault\n */\n private isSchemaBoolean(schema: ZodType): boolean {\n try {\n let currentSchema = schema\n const typeName = currentSchema.def.type\n\n // Unwrap ZodOptional and ZodDefault\n if (typeName === 'optional' || typeName === 'default') {\n currentSchema = (currentSchema as any)?._def?.innerType || currentSchema\n }\n\n const innerTypeName = currentSchema.def.type\n return innerTypeName === 'boolean'\n } catch {\n return false\n }\n }\n\n /**\n * Formats help text for available commands\n */\n formatCommandList(commands: Array<{ path: string; class: any }>): string {\n const lines = ['Available commands:', '']\n for (const { path } of commands) {\n lines.push(` ${path}`)\n }\n return lines.join('\\n')\n }\n}\n","import type { ClassTypeWithInstance, InjectionToken } from '@navios/di'\n\nimport { Container, inject, Injectable } from '@navios/di'\n\nimport type { CommandHandler, Module } from './interfaces/index.mjs'\n\nimport { CliParserService, ModuleLoaderService } from './services/index.mjs'\n\nexport interface CommanderApplicationOptions {}\n\n@Injectable()\nexport class CommanderApplication {\n private moduleLoader = inject(ModuleLoaderService)\n private cliParser = inject(CliParserService)\n protected container = inject(Container)\n\n private appModule: ClassTypeWithInstance<Module> | null = null\n private options: CommanderApplicationOptions = {}\n\n isInitialized = false\n\n async setup(\n appModule: ClassTypeWithInstance<Module>,\n options: CommanderApplicationOptions = {},\n ) {\n this.appModule = appModule\n this.options = options\n }\n\n getContainer() {\n return this.container\n }\n\n async init() {\n if (!this.appModule) {\n throw new Error(\n '[Navios Commander] App module is not set. Call setup() first.',\n )\n }\n await this.moduleLoader.loadModules(this.appModule)\n this.isInitialized = true\n }\n\n async executeCommand(commandPath: string, options: any = {}) {\n if (!this.isInitialized) {\n throw new Error(\n '[Navios Commander] Application is not initialized. Call init() first.',\n )\n }\n\n // Use pre-collected command metadata from module loading\n const commandWithMetadata = this.moduleLoader.getCommandByPath(commandPath)\n\n if (!commandWithMetadata) {\n throw new Error(`[Navios Commander] Command not found: ${commandPath}`)\n }\n\n const { class: commandClass, metadata } = commandWithMetadata\n\n // Validate options with zod schema if provided\n let validatedOptions = options\n if (metadata.optionsSchema) {\n validatedOptions = metadata.optionsSchema.parse(options)\n }\n\n // Get command instance and execute\n const commandInstance = await this.container.get<CommandHandler>(\n commandClass as unknown as InjectionToken<CommandHandler>,\n )\n\n if (!commandInstance.execute) {\n throw new Error(\n `[Navios Commander] Command ${commandPath} does not implement execute method`,\n )\n }\n\n await commandInstance.execute(validatedOptions)\n }\n\n getAllCommands() {\n // Use pre-collected command metadata from module loading\n const commandsMap = this.moduleLoader.getAllCommandsWithMetadata()\n const commandsWithMetadata: Array<{\n path: string\n class: ClassTypeWithInstance<any>\n }> = []\n\n for (const [, { class: cmd, metadata }] of commandsMap) {\n commandsWithMetadata.push({\n path: metadata.path,\n class: cmd,\n })\n }\n\n return commandsWithMetadata\n }\n\n /**\n * Runs the CLI application by parsing process.argv and executing the command\n * @param argv - Command-line arguments (defaults to process.argv)\n */\n async run(argv: string[] = process.argv) {\n if (!this.isInitialized) {\n throw new Error(\n '[Navios Commander] Application is not initialized. Call init() first.',\n )\n }\n\n try {\n // First, try to extract the command path to get its schema\n // We need to do a preliminary parse to find the command\n const preliminaryParse = this.cliParser.parse(argv)\n const commandWithMetadata = this.moduleLoader.getCommandByPath(\n preliminaryParse.command,\n )\n\n // Re-parse with schema if available\n const parsed = commandWithMetadata?.metadata.optionsSchema\n ? this.cliParser.parse(argv, commandWithMetadata.metadata.optionsSchema)\n : preliminaryParse\n\n // Handle special commands\n if (\n parsed.command === 'help' ||\n parsed.options.help ||\n parsed.options.h\n ) {\n const commands = this.getAllCommands()\n console.log(this.cliParser.formatCommandList(commands))\n return\n }\n\n // Execute the command\n await this.executeCommand(parsed.command, parsed.options)\n } catch (error) {\n if (error instanceof Error) {\n console.error(`Error: ${error.message}`)\n\n // Show available commands on error\n if (error.message.includes('Command not found')) {\n console.log(\n '\\n' + this.cliParser.formatCommandList(this.getAllCommands()),\n )\n }\n }\n throw error\n }\n }\n\n async dispose() {\n if (this.moduleLoader) {\n this.moduleLoader.dispose()\n }\n }\n\n async close() {\n await this.dispose()\n }\n}\n","import type { ClassTypeWithInstance } from '@navios/di'\n\nimport { Container } from '@navios/di'\n\nimport type { CommanderApplicationOptions } from './commander.application.mjs'\nimport type { Module } from './interfaces/index.mjs'\n\nimport { CommanderApplication } from './commander.application.mjs'\n\nexport class CommanderFactory {\n static async create(\n appModule: ClassTypeWithInstance<Module>,\n options: CommanderApplicationOptions = {},\n ) {\n const container = new Container()\n const app = await container.get(CommanderApplication)\n await app.setup(appModule, options)\n return app\n }\n}\n","import type { ClassType } from '@navios/di'\nimport type { ZodObject } from 'zod'\n\nimport { Injectable, InjectableScope, InjectionToken } from '@navios/di'\n\nimport { getCommandMetadata } from '../metadata/index.mjs'\n\nexport interface CommandOptions {\n path: string\n optionsSchema?: ZodObject\n}\n\nexport function Command({ path, optionsSchema }: CommandOptions) {\n return function (target: ClassType, context: ClassDecoratorContext) {\n if (context.kind !== 'class') {\n throw new Error(\n '[Navios Commander] @Command decorator can only be used on classes.',\n )\n }\n const token = InjectionToken.create(target)\n if (context.metadata) {\n getCommandMetadata(target, context, path, optionsSchema)\n }\n return Injectable({\n token,\n scope: InjectableScope.Singleton,\n })(target, context)\n }\n}\n","import type { ClassType } from '@navios/di'\n\nimport { Injectable, InjectableScope, InjectionToken } from '@navios/di'\n\nimport { getCliModuleMetadata } from '../metadata/index.mjs'\n\nexport interface CliModuleOptions {\n commands?: ClassType[] | Set<ClassType>\n imports?: ClassType[] | Set<ClassType>\n}\n\nexport function CliModule(\n { commands = [], imports = [] }: CliModuleOptions = {\n commands: [],\n imports: [],\n },\n) {\n return (target: ClassType, context: ClassDecoratorContext) => {\n if (context.kind !== 'class') {\n throw new Error(\n '[Navios Commander] @CliModule decorator can only be used on classes.',\n )\n }\n // Register the module in the service locator\n const token = InjectionToken.create(target)\n const moduleMetadata = getCliModuleMetadata(target, context)\n for (const command of commands) {\n moduleMetadata.commands.add(command)\n }\n for (const importedModule of imports) {\n moduleMetadata.imports.add(importedModule)\n }\n\n return Injectable({\n token,\n scope: InjectableScope.Singleton,\n })(target, context)\n }\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/metadata/command.metadata.mts","../src/metadata/cli-module.metadata.mts","../src/services/module-loader.service.mts","../src/services/cli-parser.service.mts","../src/commander.application.mts","../src/commander.factory.mts","../src/decorators/command.decorator.mts","../src/decorators/cli-module.decorator.mts"],"names":["Injectable","ModuleLoaderService","inject","Container","_init","CliParserService","CommanderApplication","InjectionToken","InjectableScope"],"mappings":"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","file":"index.js","sourcesContent":["import type { ClassType } from '@navios/di'\nimport type { ZodObject } from 'zod'\n\nexport const CommandMetadataKey = Symbol('CommandMetadataKey')\n\nexport interface CommandMetadata {\n path: string\n optionsSchema?: ZodObject\n customAttributes: Map<string | symbol, any>\n}\n\nexport function getCommandMetadata(\n target: ClassType,\n context: ClassDecoratorContext,\n path: string,\n optionsSchema?: ZodObject,\n): CommandMetadata {\n if (context.metadata) {\n const metadata = context.metadata[CommandMetadataKey] as\n | CommandMetadata\n | undefined\n if (metadata) {\n return metadata\n } else {\n const newMetadata: CommandMetadata = {\n path,\n optionsSchema,\n customAttributes: new Map<string | symbol, any>(),\n }\n context.metadata[CommandMetadataKey] = newMetadata\n // @ts-expect-error We add a custom metadata key to the target\n target[CommandMetadataKey] = newMetadata\n return newMetadata\n }\n }\n throw new Error('[Navios Commander] Wrong environment.')\n}\n\nexport function extractCommandMetadata(target: ClassType): CommandMetadata {\n // @ts-expect-error We add a custom metadata key to the target\n const metadata = target[CommandMetadataKey] as CommandMetadata | undefined\n if (!metadata) {\n throw new Error(\n '[Navios Commander] Command metadata not found. Make sure to use @Command decorator.',\n )\n }\n return metadata\n}\n\nexport function hasCommandMetadata(target: ClassType): boolean {\n // @ts-expect-error We add a custom metadata key to the target\n const metadata = target[CommandMetadataKey] as CommandMetadata | undefined\n return !!metadata\n}\n","import type { ClassType } from '@navios/di'\n\nexport const CliModuleMetadataKey = Symbol('CliModuleMetadataKey')\n\nexport interface CliModuleMetadata {\n commands: Set<ClassType>\n imports: Set<ClassType>\n customAttributes: Map<string | symbol, any>\n}\n\nexport function getCliModuleMetadata(\n target: ClassType,\n context: ClassDecoratorContext,\n): CliModuleMetadata {\n if (context.metadata) {\n const metadata = context.metadata[CliModuleMetadataKey] as\n | CliModuleMetadata\n | undefined\n if (metadata) {\n return metadata\n } else {\n const newMetadata: CliModuleMetadata = {\n commands: new Set<ClassType>(),\n imports: new Set<ClassType>(),\n customAttributes: new Map<string | symbol, any>(),\n }\n context.metadata[CliModuleMetadataKey] = newMetadata\n // @ts-expect-error We add a custom metadata key to the target\n target[CliModuleMetadataKey] = newMetadata\n return newMetadata\n }\n }\n throw new Error('[Navios Commander] Wrong environment.')\n}\n\nexport function extractCliModuleMetadata(\n target: ClassType,\n): CliModuleMetadata {\n // @ts-expect-error We add a custom metadata key to the target\n const metadata = target[CliModuleMetadataKey] as\n | CliModuleMetadata\n | undefined\n if (!metadata) {\n throw new Error(\n `[Navios Commander] Module metadata not found for ${target.name}. Make sure to use @CliModule decorator.`,\n )\n }\n return metadata\n}\n\nexport function hasCliModuleMetadata(target: ClassType): boolean {\n // @ts-expect-error We add a custom metadata key to the target\n return !!target[CliModuleMetadataKey]\n}\n","import type { ClassTypeWithInstance } from '@navios/di'\n\nimport { Container, inject, Injectable } from '@navios/di'\n\nimport type { CommandHandler, Module } from '../interfaces/index.mjs'\nimport type { CliModuleMetadata, CommandMetadata } from '../metadata/index.mjs'\n\nimport {\n extractCliModuleMetadata,\n extractCommandMetadata,\n} from '../metadata/index.mjs'\n\nexport interface CommandWithMetadata {\n class: ClassTypeWithInstance<CommandHandler>\n metadata: CommandMetadata\n}\n\n@Injectable()\nexport class ModuleLoaderService {\n protected container = inject(Container)\n private modulesMetadata: Map<string, CliModuleMetadata> = new Map()\n private loadedModules: Map<string, any> = new Map()\n private commandsMetadata: Map<string, CommandWithMetadata> = new Map()\n private initialized = false\n\n async loadModules(appModule: ClassTypeWithInstance<Module>) {\n if (this.initialized) {\n return\n }\n await this.traverseModules(appModule)\n this.initialized = true\n }\n\n private async traverseModules(\n module: ClassTypeWithInstance<Module>,\n parentMetadata?: CliModuleMetadata,\n ) {\n const metadata = extractCliModuleMetadata(module)\n if (parentMetadata) {\n this.mergeMetadata(metadata, parentMetadata)\n }\n const moduleName = module.name\n if (this.modulesMetadata.has(moduleName)) {\n return\n }\n this.modulesMetadata.set(moduleName, metadata)\n\n // Collect command metadata during module loading\n for (const command of metadata.commands) {\n const commandMetadata = extractCommandMetadata(command)\n this.commandsMetadata.set(commandMetadata.path, {\n class: command,\n metadata: commandMetadata,\n })\n }\n\n const imports = metadata.imports ?? new Set()\n const loadingPromises = Array.from(imports).map(async (importedModule) =>\n this.traverseModules(importedModule, metadata),\n )\n await Promise.all(loadingPromises)\n const instance = await this.container.get(module)\n if (instance.onModuleInit) {\n await instance.onModuleInit()\n }\n this.loadedModules.set(moduleName, instance)\n }\n\n private mergeMetadata(\n metadata: CliModuleMetadata,\n parentMetadata: CliModuleMetadata,\n ): void {\n if (parentMetadata.customAttributes) {\n for (const [key, value] of parentMetadata.customAttributes) {\n if (metadata.customAttributes.has(key)) {\n continue\n }\n metadata.customAttributes.set(key, value)\n }\n }\n }\n\n getAllModules(): Map<string, CliModuleMetadata> {\n return this.modulesMetadata\n }\n\n getAllCommands(): Map<string, ClassTypeWithInstance<any>> {\n const commands = new Map<string, ClassTypeWithInstance<any>>()\n for (const metadata of this.modulesMetadata.values()) {\n for (const command of metadata.commands) {\n commands.set(command.name, command)\n }\n }\n return commands\n }\n\n /**\n * Get all commands with their metadata, indexed by command path.\n * This is populated during loadModules, so path information is available\n * before parsing CLI argv.\n */\n getAllCommandsWithMetadata(): Map<string, CommandWithMetadata> {\n return this.commandsMetadata\n }\n\n /**\n * Get a command by its path, with metadata already extracted.\n * Returns undefined if command is not found.\n */\n getCommandByPath(path: string): CommandWithMetadata | undefined {\n return this.commandsMetadata.get(path)\n }\n\n dispose() {\n this.modulesMetadata.clear()\n this.loadedModules.clear()\n this.commandsMetadata.clear()\n this.initialized = false\n }\n}\n","import type { ZodObject, ZodType } from 'zod'\n\nimport { Injectable } from '@navios/di'\n\nexport interface ParsedCliArgs {\n command: string\n options: Record<string, any>\n positionals: string[]\n}\n\n@Injectable()\nexport class CliParserService {\n /**\n * Parses command-line arguments from process.argv\n * Commands can be multi-word (e.g., 'db migrate', 'cache clear')\n * Expected format: node script.js command [subcommand...] --flag value --boolean-flag positional1 positional2\n *\n * @param argv - Array of command-line arguments (typically process.argv)\n * @param optionsSchema - Optional Zod schema to determine boolean flags and option types\n * @returns Parsed command (space-separated if multi-word), options, and positional arguments\n */\n parse(argv: string[], optionsSchema?: ZodObject): ParsedCliArgs {\n // Skip first two args (node and script path)\n const args = argv.slice(2)\n\n if (args.length === 0) {\n throw new Error('[Navios Commander] No command provided')\n }\n\n // Extract boolean and array field names from schema for accurate parsing\n const booleanFields = optionsSchema\n ? this.extractBooleanFields(optionsSchema)\n : new Set<string>()\n const arrayFields = optionsSchema\n ? this.extractArrayFields(optionsSchema)\n : new Set<string>()\n\n // Collect command words until we hit an argument that starts with '-' or '--'\n const commandParts: string[] = []\n let i = 0\n while (i < args.length && !args[i].startsWith('-')) {\n commandParts.push(args[i])\n i++\n }\n\n if (commandParts.length === 0) {\n throw new Error('[Navios Commander] No command provided')\n }\n\n const command = commandParts.join(' ')\n const options: Record<string, any> = {}\n const positionals: string[] = []\n while (i < args.length) {\n const arg = args[i]\n\n if (arg.startsWith('--')) {\n // Long option format: --key=value or --key value\n const key = arg.slice(2)\n const equalIndex = key.indexOf('=')\n\n if (equalIndex !== -1) {\n // Format: --key=value\n const optionName = key.slice(0, equalIndex)\n const optionValue = key.slice(equalIndex + 1)\n const camelCaseKey = this.camelCase(optionName)\n const isArray = arrayFields.has(camelCaseKey) || arrayFields.has(optionName)\n\n if (isArray) {\n // For array fields, accumulate values\n if (!options[camelCaseKey]) {\n options[camelCaseKey] = []\n }\n options[camelCaseKey].push(this.parseValue(optionValue))\n } else {\n options[camelCaseKey] = this.parseValue(optionValue)\n }\n i++\n } else {\n // Format: --key value or --boolean-flag\n const camelCaseKey = this.camelCase(key)\n const isBoolean =\n booleanFields.has(camelCaseKey) || booleanFields.has(key)\n const isArray = arrayFields.has(camelCaseKey) || arrayFields.has(key)\n const nextArg = args[i + 1]\n\n if (isBoolean) {\n // Known boolean flag from schema\n options[camelCaseKey] = true\n i++\n } else if (isArray && nextArg && !nextArg.startsWith('-')) {\n // Known array field from schema - accumulate values\n if (!options[camelCaseKey]) {\n options[camelCaseKey] = []\n }\n options[camelCaseKey].push(this.parseValue(nextArg))\n i += 2\n } else if (nextArg && !nextArg.startsWith('-')) {\n // Has a value\n options[camelCaseKey] = this.parseValue(nextArg)\n i += 2\n } else {\n // Assume boolean flag\n options[camelCaseKey] = true\n i++\n }\n }\n } else if (arg.startsWith('-') && arg.length > 1 && arg !== '-') {\n // Short option format: -k value or -abc (multiple flags)\n const flags = arg.slice(1)\n\n if (flags.length === 1) {\n // Single short flag: -k value or -k\n const isBoolean = booleanFields.has(flags)\n const isArray = arrayFields.has(flags)\n const nextArg = args[i + 1]\n\n if (isBoolean) {\n // Known boolean flag from schema\n options[flags] = true\n i++\n } else if (isArray && nextArg && !nextArg.startsWith('-')) {\n // Known array field from schema - accumulate values\n if (!options[flags]) {\n options[flags] = []\n }\n options[flags].push(this.parseValue(nextArg))\n i += 2\n } else if (nextArg && !nextArg.startsWith('-')) {\n options[flags] = this.parseValue(nextArg)\n i += 2\n } else {\n options[flags] = true\n i++\n }\n } else {\n // Multiple short flags: -abc -> {a: true, b: true, c: true}\n for (const flag of flags) {\n options[flag] = true\n }\n i++\n }\n } else {\n // Positional argument\n positionals.push(arg)\n i++\n }\n }\n\n return {\n command,\n options,\n positionals,\n }\n }\n\n /**\n * Converts kebab-case to camelCase\n */\n private camelCase(str: string): string {\n return str.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase())\n }\n\n /**\n * Attempts to parse string values into appropriate types\n */\n private parseValue(value: string): any {\n // Check for boolean\n if (value === 'true') return true\n if (value === 'false') return false\n\n // Check for null/undefined\n if (value === 'null') return null\n if (value === 'undefined') return undefined\n\n // Check for number\n if (/^-?\\d+$/.test(value)) {\n return parseInt(value, 10)\n }\n if (/^-?\\d+\\.\\d+$/.test(value)) {\n return parseFloat(value)\n }\n\n // Check for JSON\n if (\n (value.startsWith('{') && value.endsWith('}')) ||\n (value.startsWith('[') && value.endsWith(']'))\n ) {\n try {\n return JSON.parse(value)\n } catch {\n // If parsing fails, return as string\n return value\n }\n }\n\n // Return as string\n return value\n }\n\n /**\n * Extracts boolean field names from a Zod schema\n * Handles ZodObject, ZodOptional, and ZodDefault wrappers\n */\n private extractBooleanFields(schema: ZodObject): Set<string> {\n const booleanFields = new Set<string>()\n\n try {\n // Check if schema has _def.typeName (Zod schema structure)\n const typeName = schema.def.type\n\n if (typeName === 'object') {\n // Extract shape from ZodObject\n const shape = schema.def.shape\n\n if (shape && typeof shape === 'object') {\n for (const [key, fieldSchema] of Object.entries(shape)) {\n if (this.isSchemaBoolean(fieldSchema as any)) {\n booleanFields.add(key)\n }\n }\n }\n }\n } catch {\n // Silently fail if schema introspection fails\n }\n\n return booleanFields\n }\n\n /**\n * Extracts array field names from a Zod schema\n * Handles ZodObject, ZodOptional, and ZodDefault wrappers\n */\n private extractArrayFields(schema: ZodObject): Set<string> {\n const arrayFields = new Set<string>()\n\n try {\n const typeName = schema.def.type\n\n if (typeName === 'object') {\n const shape = schema.def.shape\n\n if (shape && typeof shape === 'object') {\n for (const [key, fieldSchema] of Object.entries(shape)) {\n if (this.isSchemaArray(fieldSchema as any)) {\n arrayFields.add(key)\n }\n }\n }\n }\n } catch {\n // Silently fail if schema introspection fails\n }\n\n return arrayFields\n }\n\n /**\n * Checks if a Zod schema represents a boolean type\n * Unwraps ZodOptional and ZodDefault\n */\n private isSchemaBoolean(schema: ZodType): boolean {\n try {\n let currentSchema = schema\n const typeName = currentSchema.def.type\n\n // Unwrap ZodOptional and ZodDefault\n if (typeName === 'optional' || typeName === 'default') {\n currentSchema = (currentSchema as any)?._def?.innerType || currentSchema\n }\n\n const innerTypeName = currentSchema.def.type\n return innerTypeName === 'boolean'\n } catch {\n return false\n }\n }\n\n /**\n * Checks if a Zod schema represents an array type\n * Unwraps ZodOptional and ZodDefault\n */\n private isSchemaArray(schema: ZodType): boolean {\n try {\n let currentSchema = schema\n const typeName = currentSchema.def.type\n\n // Unwrap ZodOptional and ZodDefault\n if (typeName === 'optional' || typeName === 'default') {\n currentSchema = (currentSchema as any)?._def?.innerType || currentSchema\n }\n\n const innerTypeName = currentSchema.def.type\n return innerTypeName === 'array'\n } catch {\n return false\n }\n }\n\n /**\n * Formats help text for available commands\n */\n formatCommandList(commands: Array<{ path: string; class: any }>): string {\n const lines = ['Available commands:', '']\n for (const { path } of commands) {\n lines.push(` ${path}`)\n }\n return lines.join('\\n')\n }\n}\n","import type { ClassTypeWithInstance, InjectionToken } from '@navios/di'\n\nimport { Container, inject, Injectable } from '@navios/di'\n\nimport type { CommandHandler, Module } from './interfaces/index.mjs'\n\nimport { CliParserService, ModuleLoaderService } from './services/index.mjs'\n\nexport interface CommanderApplicationOptions {}\n\n@Injectable()\nexport class CommanderApplication {\n private moduleLoader = inject(ModuleLoaderService)\n private cliParser = inject(CliParserService)\n protected container = inject(Container)\n\n private appModule: ClassTypeWithInstance<Module> | null = null\n private options: CommanderApplicationOptions = {}\n\n isInitialized = false\n\n async setup(\n appModule: ClassTypeWithInstance<Module>,\n options: CommanderApplicationOptions = {},\n ) {\n this.appModule = appModule\n this.options = options\n }\n\n getContainer() {\n return this.container\n }\n\n async init() {\n if (!this.appModule) {\n throw new Error(\n '[Navios Commander] App module is not set. Call setup() first.',\n )\n }\n await this.moduleLoader.loadModules(this.appModule)\n this.isInitialized = true\n }\n\n async executeCommand(commandPath: string, options: any = {}) {\n if (!this.isInitialized) {\n throw new Error(\n '[Navios Commander] Application is not initialized. Call init() first.',\n )\n }\n\n // Use pre-collected command metadata from module loading\n const commandWithMetadata = this.moduleLoader.getCommandByPath(commandPath)\n\n if (!commandWithMetadata) {\n throw new Error(`[Navios Commander] Command not found: ${commandPath}`)\n }\n\n const { class: commandClass, metadata } = commandWithMetadata\n\n // Validate options with zod schema if provided\n let validatedOptions = options\n if (metadata.optionsSchema) {\n validatedOptions = metadata.optionsSchema.parse(options)\n }\n\n // Get command instance and execute\n const commandInstance = await this.container.get<CommandHandler>(\n commandClass as unknown as InjectionToken<CommandHandler>,\n )\n\n if (!commandInstance.execute) {\n throw new Error(\n `[Navios Commander] Command ${commandPath} does not implement execute method`,\n )\n }\n\n await commandInstance.execute(validatedOptions)\n }\n\n getAllCommands() {\n // Use pre-collected command metadata from module loading\n const commandsMap = this.moduleLoader.getAllCommandsWithMetadata()\n const commandsWithMetadata: Array<{\n path: string\n class: ClassTypeWithInstance<any>\n }> = []\n\n for (const [, { class: cmd, metadata }] of commandsMap) {\n commandsWithMetadata.push({\n path: metadata.path,\n class: cmd,\n })\n }\n\n return commandsWithMetadata\n }\n\n /**\n * Runs the CLI application by parsing process.argv and executing the command\n * @param argv - Command-line arguments (defaults to process.argv)\n */\n async run(argv: string[] = process.argv) {\n if (!this.isInitialized) {\n throw new Error(\n '[Navios Commander] Application is not initialized. Call init() first.',\n )\n }\n\n try {\n // First, try to extract the command path to get its schema\n // We need to do a preliminary parse to find the command\n const preliminaryParse = this.cliParser.parse(argv)\n const commandWithMetadata = this.moduleLoader.getCommandByPath(\n preliminaryParse.command,\n )\n\n // Re-parse with schema if available\n const parsed = commandWithMetadata?.metadata.optionsSchema\n ? this.cliParser.parse(argv, commandWithMetadata.metadata.optionsSchema)\n : preliminaryParse\n\n // Handle special commands\n if (\n parsed.command === 'help' ||\n parsed.options.help ||\n parsed.options.h\n ) {\n const commands = this.getAllCommands()\n console.log(this.cliParser.formatCommandList(commands))\n return\n }\n\n // Execute the command\n await this.executeCommand(parsed.command, parsed.options)\n } catch (error) {\n if (error instanceof Error) {\n console.error(`Error: ${error.message}`)\n\n // Show available commands on error\n if (error.message.includes('Command not found')) {\n console.log(\n '\\n' + this.cliParser.formatCommandList(this.getAllCommands()),\n )\n }\n }\n throw error\n }\n }\n\n async dispose() {\n if (this.moduleLoader) {\n this.moduleLoader.dispose()\n }\n }\n\n async close() {\n await this.dispose()\n }\n}\n","import type { ClassTypeWithInstance } from '@navios/di'\n\nimport { Container } from '@navios/di'\n\nimport type { CommanderApplicationOptions } from './commander.application.mjs'\nimport type { Module } from './interfaces/index.mjs'\n\nimport { CommanderApplication } from './commander.application.mjs'\n\nexport class CommanderFactory {\n static async create(\n appModule: ClassTypeWithInstance<Module>,\n options: CommanderApplicationOptions = {},\n ) {\n const container = new Container()\n const app = await container.get(CommanderApplication)\n await app.setup(appModule, options)\n return app\n }\n}\n","import type { ClassType } from '@navios/di'\nimport type { ZodObject } from 'zod'\n\nimport { Injectable, InjectableScope, InjectionToken } from '@navios/di'\n\nimport { getCommandMetadata } from '../metadata/index.mjs'\n\nexport interface CommandOptions {\n path: string\n optionsSchema?: ZodObject\n}\n\nexport function Command({ path, optionsSchema }: CommandOptions) {\n return function (target: ClassType, context: ClassDecoratorContext) {\n if (context.kind !== 'class') {\n throw new Error(\n '[Navios Commander] @Command decorator can only be used on classes.',\n )\n }\n const token = InjectionToken.create(target)\n if (context.metadata) {\n getCommandMetadata(target, context, path, optionsSchema)\n }\n return Injectable({\n token,\n scope: InjectableScope.Singleton,\n })(target, context)\n }\n}\n","import type { ClassType } from '@navios/di'\n\nimport { Injectable, InjectableScope, InjectionToken } from '@navios/di'\n\nimport { getCliModuleMetadata } from '../metadata/index.mjs'\n\nexport interface CliModuleOptions {\n commands?: ClassType[] | Set<ClassType>\n imports?: ClassType[] | Set<ClassType>\n}\n\nexport function CliModule(\n { commands = [], imports = [] }: CliModuleOptions = {\n commands: [],\n imports: [],\n },\n) {\n return (target: ClassType, context: ClassDecoratorContext) => {\n if (context.kind !== 'class') {\n throw new Error(\n '[Navios Commander] @CliModule decorator can only be used on classes.',\n )\n }\n // Register the module in the service locator\n const token = InjectionToken.create(target)\n const moduleMetadata = getCliModuleMetadata(target, context)\n for (const command of commands) {\n moduleMetadata.commands.add(command)\n }\n for (const importedModule of imports) {\n moduleMetadata.imports.add(importedModule)\n }\n\n return Injectable({\n token,\n scope: InjectableScope.Singleton,\n })(target, context)\n }\n}\n"]}
|
package/lib/index.mjs
CHANGED
|
@@ -209,6 +209,7 @@ var CliParserService = class {
|
|
|
209
209
|
throw new Error("[Navios Commander] No command provided");
|
|
210
210
|
}
|
|
211
211
|
const booleanFields = optionsSchema ? this.extractBooleanFields(optionsSchema) : /* @__PURE__ */ new Set();
|
|
212
|
+
const arrayFields = optionsSchema ? this.extractArrayFields(optionsSchema) : /* @__PURE__ */ new Set();
|
|
212
213
|
const commandParts = [];
|
|
213
214
|
let i = 0;
|
|
214
215
|
while (i < args.length && !args[i].startsWith("-")) {
|
|
@@ -229,15 +230,31 @@ var CliParserService = class {
|
|
|
229
230
|
if (equalIndex !== -1) {
|
|
230
231
|
const optionName = key.slice(0, equalIndex);
|
|
231
232
|
const optionValue = key.slice(equalIndex + 1);
|
|
232
|
-
|
|
233
|
+
const camelCaseKey = this.camelCase(optionName);
|
|
234
|
+
const isArray = arrayFields.has(camelCaseKey) || arrayFields.has(optionName);
|
|
235
|
+
if (isArray) {
|
|
236
|
+
if (!options[camelCaseKey]) {
|
|
237
|
+
options[camelCaseKey] = [];
|
|
238
|
+
}
|
|
239
|
+
options[camelCaseKey].push(this.parseValue(optionValue));
|
|
240
|
+
} else {
|
|
241
|
+
options[camelCaseKey] = this.parseValue(optionValue);
|
|
242
|
+
}
|
|
233
243
|
i++;
|
|
234
244
|
} else {
|
|
235
245
|
const camelCaseKey = this.camelCase(key);
|
|
236
246
|
const isBoolean = booleanFields.has(camelCaseKey) || booleanFields.has(key);
|
|
247
|
+
const isArray = arrayFields.has(camelCaseKey) || arrayFields.has(key);
|
|
237
248
|
const nextArg = args[i + 1];
|
|
238
249
|
if (isBoolean) {
|
|
239
250
|
options[camelCaseKey] = true;
|
|
240
251
|
i++;
|
|
252
|
+
} else if (isArray && nextArg && !nextArg.startsWith("-")) {
|
|
253
|
+
if (!options[camelCaseKey]) {
|
|
254
|
+
options[camelCaseKey] = [];
|
|
255
|
+
}
|
|
256
|
+
options[camelCaseKey].push(this.parseValue(nextArg));
|
|
257
|
+
i += 2;
|
|
241
258
|
} else if (nextArg && !nextArg.startsWith("-")) {
|
|
242
259
|
options[camelCaseKey] = this.parseValue(nextArg);
|
|
243
260
|
i += 2;
|
|
@@ -250,10 +267,17 @@ var CliParserService = class {
|
|
|
250
267
|
const flags = arg.slice(1);
|
|
251
268
|
if (flags.length === 1) {
|
|
252
269
|
const isBoolean = booleanFields.has(flags);
|
|
270
|
+
const isArray = arrayFields.has(flags);
|
|
253
271
|
const nextArg = args[i + 1];
|
|
254
272
|
if (isBoolean) {
|
|
255
273
|
options[flags] = true;
|
|
256
274
|
i++;
|
|
275
|
+
} else if (isArray && nextArg && !nextArg.startsWith("-")) {
|
|
276
|
+
if (!options[flags]) {
|
|
277
|
+
options[flags] = [];
|
|
278
|
+
}
|
|
279
|
+
options[flags].push(this.parseValue(nextArg));
|
|
280
|
+
i += 2;
|
|
257
281
|
} else if (nextArg && !nextArg.startsWith("-")) {
|
|
258
282
|
options[flags] = this.parseValue(nextArg);
|
|
259
283
|
i += 2;
|
|
@@ -329,6 +353,28 @@ var CliParserService = class {
|
|
|
329
353
|
}
|
|
330
354
|
return booleanFields;
|
|
331
355
|
}
|
|
356
|
+
/**
|
|
357
|
+
* Extracts array field names from a Zod schema
|
|
358
|
+
* Handles ZodObject, ZodOptional, and ZodDefault wrappers
|
|
359
|
+
*/
|
|
360
|
+
extractArrayFields(schema) {
|
|
361
|
+
const arrayFields = /* @__PURE__ */ new Set();
|
|
362
|
+
try {
|
|
363
|
+
const typeName = schema.def.type;
|
|
364
|
+
if (typeName === "object") {
|
|
365
|
+
const shape = schema.def.shape;
|
|
366
|
+
if (shape && typeof shape === "object") {
|
|
367
|
+
for (const [key, fieldSchema] of Object.entries(shape)) {
|
|
368
|
+
if (this.isSchemaArray(fieldSchema)) {
|
|
369
|
+
arrayFields.add(key);
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
} catch {
|
|
375
|
+
}
|
|
376
|
+
return arrayFields;
|
|
377
|
+
}
|
|
332
378
|
/**
|
|
333
379
|
* Checks if a Zod schema represents a boolean type
|
|
334
380
|
* Unwraps ZodOptional and ZodDefault
|
|
@@ -346,6 +392,23 @@ var CliParserService = class {
|
|
|
346
392
|
return false;
|
|
347
393
|
}
|
|
348
394
|
}
|
|
395
|
+
/**
|
|
396
|
+
* Checks if a Zod schema represents an array type
|
|
397
|
+
* Unwraps ZodOptional and ZodDefault
|
|
398
|
+
*/
|
|
399
|
+
isSchemaArray(schema) {
|
|
400
|
+
try {
|
|
401
|
+
let currentSchema = schema;
|
|
402
|
+
const typeName = currentSchema.def.type;
|
|
403
|
+
if (typeName === "optional" || typeName === "default") {
|
|
404
|
+
currentSchema = currentSchema?._def?.innerType || currentSchema;
|
|
405
|
+
}
|
|
406
|
+
const innerTypeName = currentSchema.def.type;
|
|
407
|
+
return innerTypeName === "array";
|
|
408
|
+
} catch {
|
|
409
|
+
return false;
|
|
410
|
+
}
|
|
411
|
+
}
|
|
349
412
|
/**
|
|
350
413
|
* Formats help text for available commands
|
|
351
414
|
*/
|