@navios/commander 0.5.0

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 (72) hide show
  1. package/LICENSE +7 -0
  2. package/README.md +242 -0
  3. package/dist/src/commander.application.d.mts +29 -0
  4. package/dist/src/commander.application.d.mts.map +1 -0
  5. package/dist/src/commander.factory.d.mts +8 -0
  6. package/dist/src/commander.factory.d.mts.map +1 -0
  7. package/dist/src/decorators/cli-module.decorator.d.mts +7 -0
  8. package/dist/src/decorators/cli-module.decorator.d.mts.map +1 -0
  9. package/dist/src/decorators/command.decorator.d.mts +8 -0
  10. package/dist/src/decorators/command.decorator.d.mts.map +1 -0
  11. package/dist/src/decorators/index.d.mts +3 -0
  12. package/dist/src/decorators/index.d.mts.map +1 -0
  13. package/dist/src/index.d.mts +8 -0
  14. package/dist/src/index.d.mts.map +1 -0
  15. package/dist/src/interfaces/cli-module.interface.d.mts +5 -0
  16. package/dist/src/interfaces/cli-module.interface.d.mts.map +1 -0
  17. package/dist/src/interfaces/command-handler.interface.d.mts +4 -0
  18. package/dist/src/interfaces/command-handler.interface.d.mts.map +1 -0
  19. package/dist/src/interfaces/index.d.mts +3 -0
  20. package/dist/src/interfaces/index.d.mts.map +1 -0
  21. package/dist/src/interfaces/module.interface.d.mts +5 -0
  22. package/dist/src/interfaces/module.interface.d.mts.map +1 -0
  23. package/dist/src/metadata/cli-module.metadata.d.mts +11 -0
  24. package/dist/src/metadata/cli-module.metadata.d.mts.map +1 -0
  25. package/dist/src/metadata/command.metadata.d.mts +12 -0
  26. package/dist/src/metadata/command.metadata.d.mts.map +1 -0
  27. package/dist/src/metadata/index.d.mts +3 -0
  28. package/dist/src/metadata/index.d.mts.map +1 -0
  29. package/dist/src/services/cli-parser.service.d.mts +44 -0
  30. package/dist/src/services/cli-parser.service.d.mts.map +1 -0
  31. package/dist/src/services/index.d.mts +3 -0
  32. package/dist/src/services/index.d.mts.map +1 -0
  33. package/dist/src/services/module-loader.service.d.mts +33 -0
  34. package/dist/src/services/module-loader.service.d.mts.map +1 -0
  35. package/dist/tsconfig.lib.tsbuildinfo +1 -0
  36. package/dist/tsconfig.tsbuildinfo +1 -0
  37. package/dist/tsup.config.d.mts +3 -0
  38. package/dist/tsup.config.d.mts.map +1 -0
  39. package/dist/vitest.config.d.mts +3 -0
  40. package/dist/vitest.config.d.mts.map +1 -0
  41. package/lib/_tsup-dts-rollup.d.mts +456 -0
  42. package/lib/_tsup-dts-rollup.d.ts +456 -0
  43. package/lib/index.d.mts +98 -0
  44. package/lib/index.d.ts +98 -0
  45. package/lib/index.js +541 -0
  46. package/lib/index.js.map +1 -0
  47. package/lib/index.mjs +524 -0
  48. package/lib/index.mjs.map +1 -0
  49. package/package.json +40 -0
  50. package/project.json +66 -0
  51. package/src/__tests__/commander.factory.e2e.spec.mts +965 -0
  52. package/src/commander.application.mts +159 -0
  53. package/src/commander.factory.mts +20 -0
  54. package/src/decorators/cli-module.decorator.mts +39 -0
  55. package/src/decorators/command.decorator.mts +29 -0
  56. package/src/decorators/index.mts +2 -0
  57. package/src/index.mts +7 -0
  58. package/src/interfaces/command-handler.interface.mts +3 -0
  59. package/src/interfaces/index.mts +2 -0
  60. package/src/interfaces/module.interface.mts +4 -0
  61. package/src/metadata/cli-module.metadata.mts +54 -0
  62. package/src/metadata/command.metadata.mts +54 -0
  63. package/src/metadata/index.mts +2 -0
  64. package/src/services/__tests__/cli-parser.service.spec.mts +404 -0
  65. package/src/services/cli-parser.service.mts +231 -0
  66. package/src/services/index.mts +2 -0
  67. package/src/services/module-loader.service.mts +120 -0
  68. package/tsconfig.json +18 -0
  69. package/tsconfig.lib.json +8 -0
  70. package/tsconfig.spec.json +13 -0
  71. package/tsup.config.mts +12 -0
  72. package/vitest.config.mts +9 -0
package/lib/index.js ADDED
@@ -0,0 +1,541 @@
1
+ 'use strict';
2
+
3
+ var di = require('@navios/di');
4
+
5
+ var __create = Object.create;
6
+ var __defProp = Object.defineProperty;
7
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
8
+ var __knownSymbol = (name, symbol) => (symbol = Symbol[name]) ? symbol : Symbol.for("Symbol." + name);
9
+ var __typeError = (msg) => {
10
+ throw TypeError(msg);
11
+ };
12
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
13
+ var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
14
+ var __decoratorStart = (base) => [, , , __create(null)];
15
+ var __decoratorStrings = ["class", "method", "getter", "setter", "accessor", "field", "value", "get", "set"];
16
+ var __expectFn = (fn) => fn !== void 0 && typeof fn !== "function" ? __typeError("Function expected") : fn;
17
+ var __decoratorContext = (kind, name, done, metadata, fns) => ({ kind: __decoratorStrings[kind], name, metadata, addInitializer: (fn) => done._ ? __typeError("Already initialized") : fns.push(__expectFn(fn || null)) });
18
+ var __decoratorMetadata = (array, target) => __defNormalProp(target, __knownSymbol("metadata"), array[3]);
19
+ var __runInitializers = (array, flags, self, value) => {
20
+ for (var i = 0, fns = array[flags >> 1], n = fns && fns.length; i < n; i++) fns[i].call(self) ;
21
+ return value;
22
+ };
23
+ var __decorateElement = (array, flags, name, decorators, target, extra) => {
24
+ var it, done, ctx, k = flags & 7, p = false;
25
+ var j = 0;
26
+ var extraInitializers = array[j] || (array[j] = []);
27
+ var desc = k && ((target = target.prototype), k < 5 && (k > 3 || !p) && __getOwnPropDesc(target , name));
28
+ __name(target, name);
29
+ for (var i = decorators.length - 1; i >= 0; i--) {
30
+ ctx = __decoratorContext(k, name, done = {}, array[3], extraInitializers);
31
+ it = (0, decorators[i])(target, ctx), done._ = 1;
32
+ __expectFn(it) && (target = it);
33
+ }
34
+ return __decoratorMetadata(array, target), desc && __defProp(target, name, desc), p ? k ^ 4 ? extra : desc : target;
35
+ };
36
+
37
+ // src/metadata/command.metadata.mts
38
+ var CommandMetadataKey = Symbol("CommandMetadataKey");
39
+ function getCommandMetadata(target, context, path, optionsSchema) {
40
+ if (context.metadata) {
41
+ const metadata = context.metadata[CommandMetadataKey];
42
+ if (metadata) {
43
+ return metadata;
44
+ } else {
45
+ const newMetadata = {
46
+ path,
47
+ optionsSchema,
48
+ customAttributes: /* @__PURE__ */ new Map()
49
+ };
50
+ context.metadata[CommandMetadataKey] = newMetadata;
51
+ target[CommandMetadataKey] = newMetadata;
52
+ return newMetadata;
53
+ }
54
+ }
55
+ throw new Error("[Navios Commander] Wrong environment.");
56
+ }
57
+ function extractCommandMetadata(target) {
58
+ const metadata = target[CommandMetadataKey];
59
+ if (!metadata) {
60
+ throw new Error(
61
+ "[Navios Commander] Command metadata not found. Make sure to use @Command decorator."
62
+ );
63
+ }
64
+ return metadata;
65
+ }
66
+ function hasCommandMetadata(target) {
67
+ const metadata = target[CommandMetadataKey];
68
+ return !!metadata;
69
+ }
70
+
71
+ // src/metadata/cli-module.metadata.mts
72
+ var CliModuleMetadataKey = Symbol("CliModuleMetadataKey");
73
+ function getCliModuleMetadata(target, context) {
74
+ if (context.metadata) {
75
+ const metadata = context.metadata[CliModuleMetadataKey];
76
+ if (metadata) {
77
+ return metadata;
78
+ } else {
79
+ const newMetadata = {
80
+ commands: /* @__PURE__ */ new Set(),
81
+ imports: /* @__PURE__ */ new Set(),
82
+ customAttributes: /* @__PURE__ */ new Map()
83
+ };
84
+ context.metadata[CliModuleMetadataKey] = newMetadata;
85
+ target[CliModuleMetadataKey] = newMetadata;
86
+ return newMetadata;
87
+ }
88
+ }
89
+ throw new Error("[Navios Commander] Wrong environment.");
90
+ }
91
+ function extractCliModuleMetadata(target) {
92
+ const metadata = target[CliModuleMetadataKey];
93
+ if (!metadata) {
94
+ throw new Error(
95
+ `[Navios Commander] Module metadata not found for ${target.name}. Make sure to use @CliModule decorator.`
96
+ );
97
+ }
98
+ return metadata;
99
+ }
100
+ function hasCliModuleMetadata(target) {
101
+ return !!target[CliModuleMetadataKey];
102
+ }
103
+
104
+ // src/services/module-loader.service.mts
105
+ var _ModuleLoaderService_decorators, _init;
106
+ _ModuleLoaderService_decorators = [di.Injectable()];
107
+ exports.ModuleLoaderService = class ModuleLoaderService {
108
+ container = di.inject(di.Container);
109
+ modulesMetadata = /* @__PURE__ */ new Map();
110
+ loadedModules = /* @__PURE__ */ new Map();
111
+ commandsMetadata = /* @__PURE__ */ new Map();
112
+ initialized = false;
113
+ async loadModules(appModule) {
114
+ if (this.initialized) {
115
+ return;
116
+ }
117
+ await this.traverseModules(appModule);
118
+ this.initialized = true;
119
+ }
120
+ async traverseModules(module, parentMetadata) {
121
+ const metadata = extractCliModuleMetadata(module);
122
+ if (parentMetadata) {
123
+ this.mergeMetadata(metadata, parentMetadata);
124
+ }
125
+ const moduleName = module.name;
126
+ if (this.modulesMetadata.has(moduleName)) {
127
+ return;
128
+ }
129
+ this.modulesMetadata.set(moduleName, metadata);
130
+ for (const command of metadata.commands) {
131
+ const commandMetadata = extractCommandMetadata(command);
132
+ this.commandsMetadata.set(commandMetadata.path, {
133
+ class: command,
134
+ metadata: commandMetadata
135
+ });
136
+ }
137
+ const imports = metadata.imports ?? /* @__PURE__ */ new Set();
138
+ const loadingPromises = Array.from(imports).map(
139
+ async (importedModule) => this.traverseModules(importedModule, metadata)
140
+ );
141
+ await Promise.all(loadingPromises);
142
+ const instance = await this.container.get(module);
143
+ if (instance.onModuleInit) {
144
+ await instance.onModuleInit();
145
+ }
146
+ this.loadedModules.set(moduleName, instance);
147
+ }
148
+ mergeMetadata(metadata, parentMetadata) {
149
+ if (parentMetadata.customAttributes) {
150
+ for (const [key, value] of parentMetadata.customAttributes) {
151
+ if (metadata.customAttributes.has(key)) {
152
+ continue;
153
+ }
154
+ metadata.customAttributes.set(key, value);
155
+ }
156
+ }
157
+ }
158
+ getAllModules() {
159
+ return this.modulesMetadata;
160
+ }
161
+ getAllCommands() {
162
+ const commands = /* @__PURE__ */ new Map();
163
+ for (const metadata of this.modulesMetadata.values()) {
164
+ for (const command of metadata.commands) {
165
+ commands.set(command.name, command);
166
+ }
167
+ }
168
+ return commands;
169
+ }
170
+ /**
171
+ * Get all commands with their metadata, indexed by command path.
172
+ * This is populated during loadModules, so path information is available
173
+ * before parsing CLI argv.
174
+ */
175
+ getAllCommandsWithMetadata() {
176
+ return this.commandsMetadata;
177
+ }
178
+ /**
179
+ * Get a command by its path, with metadata already extracted.
180
+ * Returns undefined if command is not found.
181
+ */
182
+ getCommandByPath(path) {
183
+ return this.commandsMetadata.get(path);
184
+ }
185
+ dispose() {
186
+ this.modulesMetadata.clear();
187
+ this.loadedModules.clear();
188
+ this.commandsMetadata.clear();
189
+ this.initialized = false;
190
+ }
191
+ };
192
+ _init = __decoratorStart();
193
+ exports.ModuleLoaderService = __decorateElement(_init, 0, "ModuleLoaderService", _ModuleLoaderService_decorators, exports.ModuleLoaderService);
194
+ __runInitializers(_init, 1, exports.ModuleLoaderService);
195
+ var _CliParserService_decorators, _init2;
196
+ _CliParserService_decorators = [di.Injectable()];
197
+ exports.CliParserService = class CliParserService {
198
+ /**
199
+ * Parses command-line arguments from process.argv
200
+ * Commands can be multi-word (e.g., 'db migrate', 'cache clear')
201
+ * Expected format: node script.js command [subcommand...] --flag value --boolean-flag positional1 positional2
202
+ *
203
+ * @param argv - Array of command-line arguments (typically process.argv)
204
+ * @param optionsSchema - Optional Zod schema to determine boolean flags and option types
205
+ * @returns Parsed command (space-separated if multi-word), options, and positional arguments
206
+ */
207
+ parse(argv, optionsSchema) {
208
+ const args = argv.slice(2);
209
+ if (args.length === 0) {
210
+ throw new Error("[Navios Commander] No command provided");
211
+ }
212
+ const booleanFields = optionsSchema ? this.extractBooleanFields(optionsSchema) : /* @__PURE__ */ new Set();
213
+ const commandParts = [];
214
+ let i = 0;
215
+ while (i < args.length && !args[i].startsWith("-")) {
216
+ commandParts.push(args[i]);
217
+ i++;
218
+ }
219
+ if (commandParts.length === 0) {
220
+ throw new Error("[Navios Commander] No command provided");
221
+ }
222
+ const command = commandParts.join(" ");
223
+ const options = {};
224
+ const positionals = [];
225
+ while (i < args.length) {
226
+ const arg = args[i];
227
+ if (arg.startsWith("--")) {
228
+ const key = arg.slice(2);
229
+ const equalIndex = key.indexOf("=");
230
+ if (equalIndex !== -1) {
231
+ const optionName = key.slice(0, equalIndex);
232
+ const optionValue = key.slice(equalIndex + 1);
233
+ options[this.camelCase(optionName)] = this.parseValue(optionValue);
234
+ i++;
235
+ } else {
236
+ const camelCaseKey = this.camelCase(key);
237
+ const isBoolean = booleanFields.has(camelCaseKey) || booleanFields.has(key);
238
+ const nextArg = args[i + 1];
239
+ if (isBoolean) {
240
+ options[camelCaseKey] = true;
241
+ i++;
242
+ } else if (nextArg && !nextArg.startsWith("-")) {
243
+ options[camelCaseKey] = this.parseValue(nextArg);
244
+ i += 2;
245
+ } else {
246
+ options[camelCaseKey] = true;
247
+ i++;
248
+ }
249
+ }
250
+ } else if (arg.startsWith("-") && arg.length > 1 && arg !== "-") {
251
+ const flags = arg.slice(1);
252
+ if (flags.length === 1) {
253
+ const isBoolean = booleanFields.has(flags);
254
+ const nextArg = args[i + 1];
255
+ if (isBoolean) {
256
+ options[flags] = true;
257
+ i++;
258
+ } else if (nextArg && !nextArg.startsWith("-")) {
259
+ options[flags] = this.parseValue(nextArg);
260
+ i += 2;
261
+ } else {
262
+ options[flags] = true;
263
+ i++;
264
+ }
265
+ } else {
266
+ for (const flag of flags) {
267
+ options[flag] = true;
268
+ }
269
+ i++;
270
+ }
271
+ } else {
272
+ positionals.push(arg);
273
+ i++;
274
+ }
275
+ }
276
+ return {
277
+ command,
278
+ options,
279
+ positionals
280
+ };
281
+ }
282
+ /**
283
+ * Converts kebab-case to camelCase
284
+ */
285
+ camelCase(str) {
286
+ return str.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());
287
+ }
288
+ /**
289
+ * Attempts to parse string values into appropriate types
290
+ */
291
+ parseValue(value) {
292
+ if (value === "true") return true;
293
+ if (value === "false") return false;
294
+ if (value === "null") return null;
295
+ if (value === "undefined") return void 0;
296
+ if (/^-?\d+$/.test(value)) {
297
+ return parseInt(value, 10);
298
+ }
299
+ if (/^-?\d+\.\d+$/.test(value)) {
300
+ return parseFloat(value);
301
+ }
302
+ if (value.startsWith("{") && value.endsWith("}") || value.startsWith("[") && value.endsWith("]")) {
303
+ try {
304
+ return JSON.parse(value);
305
+ } catch {
306
+ return value;
307
+ }
308
+ }
309
+ return value;
310
+ }
311
+ /**
312
+ * Extracts boolean field names from a Zod schema
313
+ * Handles ZodObject, ZodOptional, and ZodDefault wrappers
314
+ */
315
+ extractBooleanFields(schema) {
316
+ const booleanFields = /* @__PURE__ */ new Set();
317
+ try {
318
+ const typeName = schema.def.type;
319
+ if (typeName === "object") {
320
+ const shape = schema.def.shape;
321
+ if (shape && typeof shape === "object") {
322
+ for (const [key, fieldSchema] of Object.entries(shape)) {
323
+ if (this.isSchemaBoolean(fieldSchema)) {
324
+ booleanFields.add(key);
325
+ }
326
+ }
327
+ }
328
+ }
329
+ } catch {
330
+ }
331
+ return booleanFields;
332
+ }
333
+ /**
334
+ * Checks if a Zod schema represents a boolean type
335
+ * Unwraps ZodOptional and ZodDefault
336
+ */
337
+ isSchemaBoolean(schema) {
338
+ try {
339
+ let currentSchema = schema;
340
+ const typeName = currentSchema.def.type;
341
+ if (typeName === "optional" || typeName === "default") {
342
+ currentSchema = currentSchema?._def?.innerType || currentSchema;
343
+ }
344
+ const innerTypeName = currentSchema.def.type;
345
+ return innerTypeName === "boolean";
346
+ } catch {
347
+ return false;
348
+ }
349
+ }
350
+ /**
351
+ * Formats help text for available commands
352
+ */
353
+ formatCommandList(commands) {
354
+ const lines = ["Available commands:", ""];
355
+ for (const { path } of commands) {
356
+ lines.push(` ${path}`);
357
+ }
358
+ return lines.join("\n");
359
+ }
360
+ };
361
+ _init2 = __decoratorStart();
362
+ exports.CliParserService = __decorateElement(_init2, 0, "CliParserService", _CliParserService_decorators, exports.CliParserService);
363
+ __runInitializers(_init2, 1, exports.CliParserService);
364
+
365
+ // src/commander.application.mts
366
+ var _CommanderApplication_decorators, _init3;
367
+ _CommanderApplication_decorators = [di.Injectable()];
368
+ exports.CommanderApplication = class CommanderApplication {
369
+ moduleLoader = di.inject(exports.ModuleLoaderService);
370
+ cliParser = di.inject(exports.CliParserService);
371
+ container = di.inject(di.Container);
372
+ appModule = null;
373
+ options = {};
374
+ isInitialized = false;
375
+ async setup(appModule, options = {}) {
376
+ this.appModule = appModule;
377
+ this.options = options;
378
+ }
379
+ getContainer() {
380
+ return this.container;
381
+ }
382
+ async init() {
383
+ if (!this.appModule) {
384
+ throw new Error(
385
+ "[Navios Commander] App module is not set. Call setup() first."
386
+ );
387
+ }
388
+ await this.moduleLoader.loadModules(this.appModule);
389
+ this.isInitialized = true;
390
+ }
391
+ async executeCommand(commandPath, options = {}) {
392
+ if (!this.isInitialized) {
393
+ throw new Error(
394
+ "[Navios Commander] Application is not initialized. Call init() first."
395
+ );
396
+ }
397
+ const commandWithMetadata = this.moduleLoader.getCommandByPath(commandPath);
398
+ if (!commandWithMetadata) {
399
+ throw new Error(`[Navios Commander] Command not found: ${commandPath}`);
400
+ }
401
+ const { class: commandClass, metadata } = commandWithMetadata;
402
+ let validatedOptions = options;
403
+ if (metadata.optionsSchema) {
404
+ validatedOptions = metadata.optionsSchema.parse(options);
405
+ }
406
+ const commandInstance = await this.container.get(
407
+ commandClass
408
+ );
409
+ if (!commandInstance.execute) {
410
+ throw new Error(
411
+ `[Navios Commander] Command ${commandPath} does not implement execute method`
412
+ );
413
+ }
414
+ await commandInstance.execute(validatedOptions);
415
+ }
416
+ getAllCommands() {
417
+ const commandsMap = this.moduleLoader.getAllCommandsWithMetadata();
418
+ const commandsWithMetadata = [];
419
+ for (const [, { class: cmd, metadata }] of commandsMap) {
420
+ commandsWithMetadata.push({
421
+ path: metadata.path,
422
+ class: cmd
423
+ });
424
+ }
425
+ return commandsWithMetadata;
426
+ }
427
+ /**
428
+ * Runs the CLI application by parsing process.argv and executing the command
429
+ * @param argv - Command-line arguments (defaults to process.argv)
430
+ */
431
+ async run(argv = process.argv) {
432
+ if (!this.isInitialized) {
433
+ throw new Error(
434
+ "[Navios Commander] Application is not initialized. Call init() first."
435
+ );
436
+ }
437
+ try {
438
+ const preliminaryParse = this.cliParser.parse(argv);
439
+ const commandWithMetadata = this.moduleLoader.getCommandByPath(
440
+ preliminaryParse.command
441
+ );
442
+ const parsed = commandWithMetadata?.metadata.optionsSchema ? this.cliParser.parse(argv, commandWithMetadata.metadata.optionsSchema) : preliminaryParse;
443
+ if (parsed.command === "help" || parsed.options.help || parsed.options.h) {
444
+ const commands = this.getAllCommands();
445
+ console.log(this.cliParser.formatCommandList(commands));
446
+ return;
447
+ }
448
+ await this.executeCommand(parsed.command, parsed.options);
449
+ } catch (error) {
450
+ if (error instanceof Error) {
451
+ console.error(`Error: ${error.message}`);
452
+ if (error.message.includes("Command not found")) {
453
+ console.log(
454
+ "\n" + this.cliParser.formatCommandList(this.getAllCommands())
455
+ );
456
+ }
457
+ }
458
+ throw error;
459
+ }
460
+ }
461
+ async dispose() {
462
+ if (this.moduleLoader) {
463
+ this.moduleLoader.dispose();
464
+ }
465
+ }
466
+ async close() {
467
+ await this.dispose();
468
+ }
469
+ };
470
+ _init3 = __decoratorStart();
471
+ exports.CommanderApplication = __decorateElement(_init3, 0, "CommanderApplication", _CommanderApplication_decorators, exports.CommanderApplication);
472
+ __runInitializers(_init3, 1, exports.CommanderApplication);
473
+ var CommanderFactory = class {
474
+ static async create(appModule, options = {}) {
475
+ const container = new di.Container();
476
+ const app = await container.get(exports.CommanderApplication);
477
+ await app.setup(appModule, options);
478
+ return app;
479
+ }
480
+ };
481
+ function Command({ path, optionsSchema }) {
482
+ return function(target, context) {
483
+ if (context.kind !== "class") {
484
+ throw new Error(
485
+ "[Navios Commander] @Command decorator can only be used on classes."
486
+ );
487
+ }
488
+ const token = di.InjectionToken.create(target);
489
+ if (context.metadata) {
490
+ getCommandMetadata(target, context, path, optionsSchema);
491
+ }
492
+ return di.Injectable({
493
+ token,
494
+ scope: di.InjectableScope.Singleton
495
+ })(target, context);
496
+ };
497
+ }
498
+ function CliModule({ commands = [], imports = [] } = {
499
+ commands: [],
500
+ imports: []
501
+ }) {
502
+ return (target, context) => {
503
+ if (context.kind !== "class") {
504
+ throw new Error(
505
+ "[Navios Commander] @CliModule decorator can only be used on classes."
506
+ );
507
+ }
508
+ const token = di.InjectionToken.create(target);
509
+ const moduleMetadata = getCliModuleMetadata(target, context);
510
+ for (const command of commands) {
511
+ moduleMetadata.commands.add(command);
512
+ }
513
+ for (const importedModule of imports) {
514
+ moduleMetadata.imports.add(importedModule);
515
+ }
516
+ return di.Injectable({
517
+ token,
518
+ scope: di.InjectableScope.Singleton
519
+ })(target, context);
520
+ };
521
+ }
522
+
523
+ exports.CliModule = CliModule;
524
+ exports.CliModuleMetadataKey = CliModuleMetadataKey;
525
+ exports.Command = Command;
526
+ exports.CommandMetadataKey = CommandMetadataKey;
527
+ exports.CommanderFactory = CommanderFactory;
528
+ exports.extractCliModuleMetadata = extractCliModuleMetadata;
529
+ exports.extractCommandMetadata = extractCommandMetadata;
530
+ exports.getCliModuleMetadata = getCliModuleMetadata;
531
+ exports.getCommandMetadata = getCommandMetadata;
532
+ exports.hasCliModuleMetadata = hasCliModuleMetadata;
533
+ exports.hasCommandMetadata = hasCommandMetadata;
534
+ Object.keys(di).forEach(function (k) {
535
+ if (k !== 'default' && !Object.prototype.hasOwnProperty.call(exports, k)) Object.defineProperty(exports, k, {
536
+ enumerable: true,
537
+ get: function () { return di[k]; }
538
+ });
539
+ });
540
+ //# sourceMappingURL=index.js.map
541
+ //# sourceMappingURL=index.js.map
@@ -0,0 +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"]}