parse-my-command 0.1.11 → 0.2.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/README.md +13 -9
- package/dist/index.cjs +33 -4
- package/dist/index.d.cts +4 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.js +33 -4
- package/package.json +3 -3
package/README.md
CHANGED
|
@@ -11,7 +11,7 @@ Parse `argv` with **Commander.js** without executing the command
|
|
|
11
11
|
<hr />
|
|
12
12
|
|
|
13
13
|
[Commander.js](https://github.com/tj/commander.js) doesn't support parsing
|
|
14
|
-
`argv` without executing the command. This
|
|
14
|
+
`argv` without executing the command. This module provides a workaround for that.
|
|
15
15
|
|
|
16
16
|
## Installation
|
|
17
17
|
|
|
@@ -41,14 +41,17 @@ const childCommand = rootCommand
|
|
|
41
41
|
|
|
42
42
|
const argv = ["node", "index.mjs", "-a", "value1", "child", "-b", "value2"];
|
|
43
43
|
|
|
44
|
-
const {
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
44
|
+
const {
|
|
45
|
+
matchedCommand,
|
|
46
|
+
providedOptions,
|
|
47
|
+
missingOptions,
|
|
48
|
+
providedOptionsSources,
|
|
49
|
+
} = partialParse(rootCommand, argv);
|
|
48
50
|
|
|
49
51
|
console.log(matchedCommand.name()); // child
|
|
50
52
|
console.log(providedOptions.get(childCommand)); // { optionB: 'value2' }
|
|
51
53
|
console.log(missingOptions.get(childCommand)); // Set(1) { 'optionC' }
|
|
54
|
+
console.log(providedOptionsSources.get(childCommand)); // Map(1) { 'optionB' => 'cli' }
|
|
52
55
|
```
|
|
53
56
|
|
|
54
57
|
More examples can be found in the [examples](/examples/) directory.
|
|
@@ -62,7 +65,8 @@ More examples can be found in the [examples](/examples/) directory.
|
|
|
62
65
|
would throw an error (e.g. when displaying help)
|
|
63
66
|
2. In all cases where your command would throw an error before an action is
|
|
64
67
|
executed except for when a required `Option` (not to be confused with an
|
|
65
|
-
`Argument`) is missing (missing options are returned in the result
|
|
68
|
+
`Argument`) is missing (missing options are returned in the result object
|
|
69
|
+
instead).
|
|
66
70
|
|
|
67
71
|
## How It Works and Limitations
|
|
68
72
|
|
|
@@ -72,6 +76,6 @@ subcommands, setting the actions to no-op functions, and then parsing the
|
|
|
72
76
|
|
|
73
77
|
1. The implementation might break if Commander.js changes its internals
|
|
74
78
|
2. Custom argument and option processors are assumed to be pure functions
|
|
75
|
-
3.
|
|
76
|
-
4. Some edge cases might not be handled correctly (
|
|
77
|
-
issue/PR
|
|
79
|
+
3. Hook listeners attached to the commands will be ignored
|
|
80
|
+
4. Some edge cases might not be handled correctly (please feel free to open an
|
|
81
|
+
issue/PR in case you find any)
|
package/dist/index.cjs
CHANGED
|
@@ -107,8 +107,17 @@ var cloneOption = (option) => {
|
|
|
107
107
|
};
|
|
108
108
|
var partialParse = (command, argv, options) => {
|
|
109
109
|
const providedOptions = /* @__PURE__ */ new Map();
|
|
110
|
+
const providedOptionsSources = /* @__PURE__ */ new Map();
|
|
110
111
|
const commandsMap = /* @__PURE__ */ new Map();
|
|
111
112
|
let matchedCommand;
|
|
113
|
+
const setProvidedOptionSource = (command2, optionKey, source) => {
|
|
114
|
+
if (!source) {
|
|
115
|
+
return;
|
|
116
|
+
}
|
|
117
|
+
const sourcesMap = providedOptionsSources.get(command2) ?? /* @__PURE__ */ new Map();
|
|
118
|
+
sourcesMap.set(optionKey, source);
|
|
119
|
+
providedOptionsSources.set(command2, sourcesMap);
|
|
120
|
+
};
|
|
112
121
|
const createParserCommand = (parserCommand2, command2) => {
|
|
113
122
|
commandsMap.set(parserCommand2, command2);
|
|
114
123
|
copyCommandSettings(command2, parserCommand2);
|
|
@@ -118,14 +127,29 @@ var partialParse = (command, argv, options) => {
|
|
|
118
127
|
parserCommand2.addOption(cloneOption(option));
|
|
119
128
|
}
|
|
120
129
|
parserCommand2.hook("preSubcommand", (thisCommand, actionCommand) => {
|
|
121
|
-
|
|
122
|
-
|
|
130
|
+
for (const cmd of [thisCommand, actionCommand]) {
|
|
131
|
+
providedOptions.set(commandsMap.get(cmd), cmd.opts());
|
|
132
|
+
for (const optionKey of Object.keys(cmd.opts())) {
|
|
133
|
+
setProvidedOptionSource(
|
|
134
|
+
cmd,
|
|
135
|
+
optionKey,
|
|
136
|
+
cmd.getOptionValueSource(optionKey)
|
|
137
|
+
);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
123
140
|
});
|
|
124
141
|
parserCommand2.action(() => {
|
|
125
142
|
providedOptions.set(
|
|
126
143
|
commandsMap.get(parserCommand2),
|
|
127
144
|
parserCommand2.opts()
|
|
128
145
|
);
|
|
146
|
+
for (const optionKey of Object.keys(parserCommand2.opts())) {
|
|
147
|
+
setProvidedOptionSource(
|
|
148
|
+
commandsMap.get(parserCommand2),
|
|
149
|
+
optionKey,
|
|
150
|
+
parserCommand2.getOptionValueSource(optionKey)
|
|
151
|
+
);
|
|
152
|
+
}
|
|
129
153
|
matchedCommand = command2;
|
|
130
154
|
});
|
|
131
155
|
for (const subcommand of command2.commands) {
|
|
@@ -137,11 +161,16 @@ var partialParse = (command, argv, options) => {
|
|
|
137
161
|
const parserCommand = createParserCommand(new import_commander.Command(), command);
|
|
138
162
|
parserCommand.parse(argv, options);
|
|
139
163
|
const missingOptions = matchedCommand ? findMissingOptions(matchedCommand, providedOptions) : /* @__PURE__ */ new Map();
|
|
140
|
-
return {
|
|
164
|
+
return {
|
|
165
|
+
matchedCommand,
|
|
166
|
+
missingOptions,
|
|
167
|
+
providedOptions,
|
|
168
|
+
providedOptionsSources
|
|
169
|
+
};
|
|
141
170
|
};
|
|
142
171
|
// Annotate the CommonJS export names for ESM import in node:
|
|
143
172
|
0 && (module.exports = {
|
|
144
173
|
findMissingOptions,
|
|
145
174
|
partialParse
|
|
146
175
|
});
|
|
147
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/noop.ts"],
  "sourcesContent": ["import { noop } from \"./noop.ts\";\nimport camelCase from \"camelcase\";\nimport {\n  type OptionValues,\n  Command,\n  type ParseOptions,\n  Option,\n} from \"commander\";\n\nexport const findMissingOptions = (\n  command: Command,\n  providedOptionsByCommand: Map<Command, OptionValues>,\n) => {\n  const missingOptionsByCommand = new Map<Command, Set<string>>();\n\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  let currentCommand: Command | undefined | null = command;\n\n  while (currentCommand) {\n    const missingOptions = new Set<string>();\n    const providedOptions = providedOptionsByCommand.get(currentCommand);\n    for (const option of currentCommand.options) {\n      const key = option.negate\n        ? camelCase(option.long!.replace(/^--no-/, \"\"))\n        : camelCase((option.long ?? option.short)!);\n\n      if (providedOptions?.[key] !== undefined) {\n        continue;\n      }\n\n      missingOptions.add(key);\n    }\n\n    missingOptionsByCommand.set(currentCommand, missingOptions);\n    currentCommand = currentCommand.parent;\n  }\n\n  return missingOptionsByCommand;\n};\n\nexport type PartialParseResult = {\n  /**\n   * The command whose action will be executed\n   */\n  matchedCommand: Command | undefined;\n  /**\n   * A map of commands to a set of missing options for that command\n   */\n  missingOptions: Map<Command, Set<string>>;\n  /**\n   * A map of commands to the options provided for that command\n   */\n  providedOptions: Map<Command, OptionValues>;\n};\n\nconst copyCommandSettings = (source: Command, target: Command) => {\n  for (const keysToCopy of [\n    \"_allowExcessArguments\",\n    \"_allowUnknownOption\",\n    \"_args\",\n    \"_combineFlagAndOptionalValue\",\n    \"_defaultCommandName\",\n    \"_enablePositionalOptions\",\n    \"_passThroughOptions\",\n  ] as const) {\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n    (target as any)[keysToCopy] = (source as any)[keysToCopy];\n  }\n\n  target.name(source.name());\n  target.aliases(source.aliases());\n  target.version(\n    (source as Command & { version: () => string }).version(),\n    source.options.find(\n      (option) =>\n        option.attributeName() ===\n        (source as Command & { _versionOptionName: string })._versionOptionName,\n    )?.flags,\n  );\n};\n\nconst disableCommandOutput = (command: Command) => {\n  command.configureOutput({\n    writeOut: noop,\n    writeErr: noop,\n    outputError: noop,\n  });\n};\n\nconst cloneOption = (option: Option) => {\n  const newOption = new Option(option.flags, option.description);\n  newOption.makeOptionMandatory(false);\n  newOption.default(option.defaultValue, option.defaultValueDescription);\n\n  newOption.preset((option as Option & { presetArg: unknown }).presetArg);\n  newOption.env((option as Option & { envVar: string }).envVar);\n  if (option.parseArg) {\n    newOption.argParser(option.parseArg);\n  }\n\n  if (option.argChoices) {\n    newOption.choices(option.argChoices);\n  }\n\n  newOption.conflicts(\n    (option as Option & { conflictsWith: string[] }).conflictsWith,\n  );\n\n  (newOption as Option & { implied: unknown }).implied = (\n    option as Option & { implied: unknown }\n  ).implied;\n\n  return newOption;\n};\n\n/**\n * Partially parse argv for a command without executing the action. @see {@link Command.parse}\n *\n * @returns An object containing the matched command, the provided options, and the missing options.\n */\nexport const partialParse = (\n  command: Command,\n  argv: readonly string[],\n  options?: ParseOptions,\n): PartialParseResult => {\n  const providedOptions = new Map<Command, OptionValues>();\n  const commandsMap = new Map<Command, Command>();\n  let matchedCommand: Command | undefined;\n\n  const createParserCommand = (parserCommand: Command, command: Command) => {\n    commandsMap.set(parserCommand, command);\n\n    copyCommandSettings(command, parserCommand);\n    disableCommandOutput(parserCommand);\n    parserCommand.exitOverride();\n\n    for (const option of command.options) {\n      parserCommand.addOption(cloneOption(option));\n    }\n\n    parserCommand.hook(\"preSubcommand\", (thisCommand, actionCommand) => {\n      providedOptions.set(commandsMap.get(parserCommand)!, thisCommand.opts());\n      providedOptions.set(commandsMap.get(actionCommand)!, thisCommand.opts());\n    });\n\n    parserCommand.action(() => {\n      providedOptions.set(\n        commandsMap.get(parserCommand)!,\n        parserCommand.opts(),\n      );\n      matchedCommand = command;\n    });\n\n    for (const subcommand of command.commands as Command[]) {\n      const parserSubcommand = parserCommand.command(subcommand.name());\n      createParserCommand(parserSubcommand, subcommand);\n    }\n\n    return parserCommand;\n  };\n\n  const parserCommand = createParserCommand(new Command(), command);\n  parserCommand.parse(argv, options);\n\n  const missingOptions = matchedCommand\n    ? findMissingOptions(matchedCommand, providedOptions)\n    : new Map<Command, Set<string>>();\n\n  return { matchedCommand, missingOptions, providedOptions };\n};\n", "// eslint-disable-next-line @typescript-eslint/no-empty-function\nexport const noop = () => {};\n"],
  "mappings": "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",
  "names": ["camelCase", "parserCommand", "command"]
}

|
|
176
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/noop.ts"],
  "sourcesContent": ["import { noop } from \"./noop.ts\";\nimport camelCase from \"camelcase\";\nimport {\n  type OptionValues,\n  Command,\n  type ParseOptions,\n  Option,\n  type OptionValueSource,\n} from \"commander\";\n\nexport const findMissingOptions = (\n  command: Command,\n  providedOptionsByCommand: Map<Command, OptionValues>,\n) => {\n  const missingOptionsByCommand = new Map<Command, Set<string>>();\n\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  let currentCommand: Command | undefined | null = command;\n\n  while (currentCommand) {\n    const missingOptions = new Set<string>();\n    const providedOptions = providedOptionsByCommand.get(currentCommand);\n    for (const option of currentCommand.options) {\n      const key = option.negate\n        ? camelCase(option.long!.replace(/^--no-/, \"\"))\n        : camelCase((option.long ?? option.short)!);\n\n      if (providedOptions?.[key] !== undefined) {\n        continue;\n      }\n\n      missingOptions.add(key);\n    }\n\n    missingOptionsByCommand.set(currentCommand, missingOptions);\n    currentCommand = currentCommand.parent;\n  }\n\n  return missingOptionsByCommand;\n};\n\nexport type PartialParseResult = {\n  /**\n   * The command whose action will be executed\n   */\n  matchedCommand: Command | undefined;\n  /**\n   * A map of commands to a set of missing options for that command\n   */\n  missingOptions: Map<Command, Set<string>>;\n  /**\n   * A map of commands to the options provided for that command\n   */\n  providedOptions: Map<Command, OptionValues>;\n  /**\n   * A map of commands to a map of option keys to the source of the option value\n   */\n  providedOptionsSources: Map<Command, Map<string, string | undefined>>;\n};\n\nconst copyCommandSettings = (source: Command, target: Command) => {\n  for (const keysToCopy of [\n    \"_allowExcessArguments\",\n    \"_allowUnknownOption\",\n    \"_args\",\n    \"_combineFlagAndOptionalValue\",\n    \"_defaultCommandName\",\n    \"_enablePositionalOptions\",\n    \"_passThroughOptions\",\n  ] as const) {\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n    (target as any)[keysToCopy] = (source as any)[keysToCopy];\n  }\n\n  target.name(source.name());\n  target.aliases(source.aliases());\n  target.version(\n    (source as Command & { version: () => string }).version(),\n    source.options.find(\n      (option) =>\n        option.attributeName() ===\n        (source as Command & { _versionOptionName: string })._versionOptionName,\n    )?.flags,\n  );\n};\n\nconst disableCommandOutput = (command: Command) => {\n  command.configureOutput({\n    writeOut: noop,\n    writeErr: noop,\n    outputError: noop,\n  });\n};\n\nconst cloneOption = (option: Option) => {\n  const newOption = new Option(option.flags, option.description);\n  newOption.makeOptionMandatory(false);\n  newOption.default(option.defaultValue, option.defaultValueDescription);\n\n  newOption.preset((option as Option & { presetArg: unknown }).presetArg);\n  newOption.env((option as Option & { envVar: string }).envVar);\n  if (option.parseArg) {\n    newOption.argParser(option.parseArg);\n  }\n\n  if (option.argChoices) {\n    newOption.choices(option.argChoices);\n  }\n\n  newOption.conflicts(\n    (option as Option & { conflictsWith: string[] }).conflictsWith,\n  );\n\n  (newOption as Option & { implied: unknown }).implied = (\n    option as Option & { implied: unknown }\n  ).implied;\n\n  return newOption;\n};\n\n/**\n * Partially parse argv for a command without executing the action. @see {@link Command.parse}\n *\n * @returns An object containing the matched command, the provided options, and the missing options.\n */\nexport const partialParse = (\n  command: Command,\n  argv: readonly string[],\n  options?: ParseOptions,\n): PartialParseResult => {\n  const providedOptions = new Map<Command, OptionValues>();\n  const providedOptionsSources = new Map<\n    Command,\n    Map<string, string | undefined>\n  >();\n  const commandsMap = new Map<Command, Command>();\n  let matchedCommand: Command | undefined;\n\n  const setProvidedOptionSource = (\n    command: Command,\n    optionKey: string,\n    source: OptionValueSource | undefined,\n  ) => {\n    if (!source) {\n      return;\n    }\n\n    const sourcesMap =\n      providedOptionsSources.get(command) ?? new Map<string, string>();\n    sourcesMap.set(optionKey, source);\n    providedOptionsSources.set(command, sourcesMap);\n  };\n\n  const createParserCommand = (parserCommand: Command, command: Command) => {\n    commandsMap.set(parserCommand, command);\n\n    copyCommandSettings(command, parserCommand);\n    disableCommandOutput(parserCommand);\n    parserCommand.exitOverride();\n\n    for (const option of command.options) {\n      parserCommand.addOption(cloneOption(option));\n    }\n\n    parserCommand.hook(\"preSubcommand\", (thisCommand, actionCommand) => {\n      for (const cmd of [thisCommand, actionCommand]) {\n        providedOptions.set(commandsMap.get(cmd)!, cmd.opts());\n\n        for (const optionKey of Object.keys(cmd.opts())) {\n          setProvidedOptionSource(\n            cmd,\n            optionKey,\n            cmd.getOptionValueSource(optionKey),\n          );\n        }\n      }\n    });\n\n    parserCommand.action(() => {\n      providedOptions.set(\n        commandsMap.get(parserCommand)!,\n        parserCommand.opts(),\n      );\n\n      for (const optionKey of Object.keys(parserCommand.opts())) {\n        setProvidedOptionSource(\n          commandsMap.get(parserCommand)!,\n          optionKey,\n          parserCommand.getOptionValueSource(optionKey),\n        );\n      }\n\n      matchedCommand = command;\n    });\n\n    for (const subcommand of command.commands as Command[]) {\n      const parserSubcommand = parserCommand.command(subcommand.name());\n      createParserCommand(parserSubcommand, subcommand);\n    }\n\n    return parserCommand;\n  };\n\n  const parserCommand = createParserCommand(new Command(), command);\n  parserCommand.parse(argv, options);\n\n  const missingOptions = matchedCommand\n    ? findMissingOptions(matchedCommand, providedOptions)\n    : new Map<Command, Set<string>>();\n\n  return {\n    matchedCommand,\n    missingOptions,\n    providedOptions,\n    providedOptionsSources,\n  };\n};\n", "// eslint-disable-next-line @typescript-eslint/no-empty-function\nexport const noop = () => {};\n"],
  "mappings": "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",
  "names": ["camelCase", "command", "parserCommand"]
}

|
package/dist/index.d.cts
CHANGED
|
@@ -14,6 +14,10 @@ type PartialParseResult = {
|
|
|
14
14
|
* A map of commands to the options provided for that command
|
|
15
15
|
*/
|
|
16
16
|
providedOptions: Map<Command, OptionValues>;
|
|
17
|
+
/**
|
|
18
|
+
* A map of commands to a map of option keys to the source of the option value
|
|
19
|
+
*/
|
|
20
|
+
providedOptionsSources: Map<Command, Map<string, string | undefined>>;
|
|
17
21
|
};
|
|
18
22
|
/**
|
|
19
23
|
* Partially parse argv for a command without executing the action. @see {@link Command.parse}
|
package/dist/index.d.ts
CHANGED
|
@@ -14,6 +14,10 @@ type PartialParseResult = {
|
|
|
14
14
|
* A map of commands to the options provided for that command
|
|
15
15
|
*/
|
|
16
16
|
providedOptions: Map<Command, OptionValues>;
|
|
17
|
+
/**
|
|
18
|
+
* A map of commands to a map of option keys to the source of the option value
|
|
19
|
+
*/
|
|
20
|
+
providedOptionsSources: Map<Command, Map<string, string | undefined>>;
|
|
17
21
|
};
|
|
18
22
|
/**
|
|
19
23
|
* Partially parse argv for a command without executing the action. @see {@link Command.parse}
|
package/dist/index.js
CHANGED
|
@@ -74,8 +74,17 @@ var cloneOption = (option) => {
|
|
|
74
74
|
};
|
|
75
75
|
var partialParse = (command, argv, options) => {
|
|
76
76
|
const providedOptions = /* @__PURE__ */ new Map();
|
|
77
|
+
const providedOptionsSources = /* @__PURE__ */ new Map();
|
|
77
78
|
const commandsMap = /* @__PURE__ */ new Map();
|
|
78
79
|
let matchedCommand;
|
|
80
|
+
const setProvidedOptionSource = (command2, optionKey, source) => {
|
|
81
|
+
if (!source) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
const sourcesMap = providedOptionsSources.get(command2) ?? /* @__PURE__ */ new Map();
|
|
85
|
+
sourcesMap.set(optionKey, source);
|
|
86
|
+
providedOptionsSources.set(command2, sourcesMap);
|
|
87
|
+
};
|
|
79
88
|
const createParserCommand = (parserCommand2, command2) => {
|
|
80
89
|
commandsMap.set(parserCommand2, command2);
|
|
81
90
|
copyCommandSettings(command2, parserCommand2);
|
|
@@ -85,14 +94,29 @@ var partialParse = (command, argv, options) => {
|
|
|
85
94
|
parserCommand2.addOption(cloneOption(option));
|
|
86
95
|
}
|
|
87
96
|
parserCommand2.hook("preSubcommand", (thisCommand, actionCommand) => {
|
|
88
|
-
|
|
89
|
-
|
|
97
|
+
for (const cmd of [thisCommand, actionCommand]) {
|
|
98
|
+
providedOptions.set(commandsMap.get(cmd), cmd.opts());
|
|
99
|
+
for (const optionKey of Object.keys(cmd.opts())) {
|
|
100
|
+
setProvidedOptionSource(
|
|
101
|
+
cmd,
|
|
102
|
+
optionKey,
|
|
103
|
+
cmd.getOptionValueSource(optionKey)
|
|
104
|
+
);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
90
107
|
});
|
|
91
108
|
parserCommand2.action(() => {
|
|
92
109
|
providedOptions.set(
|
|
93
110
|
commandsMap.get(parserCommand2),
|
|
94
111
|
parserCommand2.opts()
|
|
95
112
|
);
|
|
113
|
+
for (const optionKey of Object.keys(parserCommand2.opts())) {
|
|
114
|
+
setProvidedOptionSource(
|
|
115
|
+
commandsMap.get(parserCommand2),
|
|
116
|
+
optionKey,
|
|
117
|
+
parserCommand2.getOptionValueSource(optionKey)
|
|
118
|
+
);
|
|
119
|
+
}
|
|
96
120
|
matchedCommand = command2;
|
|
97
121
|
});
|
|
98
122
|
for (const subcommand of command2.commands) {
|
|
@@ -104,10 +128,15 @@ var partialParse = (command, argv, options) => {
|
|
|
104
128
|
const parserCommand = createParserCommand(new Command(), command);
|
|
105
129
|
parserCommand.parse(argv, options);
|
|
106
130
|
const missingOptions = matchedCommand ? findMissingOptions(matchedCommand, providedOptions) : /* @__PURE__ */ new Map();
|
|
107
|
-
return {
|
|
131
|
+
return {
|
|
132
|
+
matchedCommand,
|
|
133
|
+
missingOptions,
|
|
134
|
+
providedOptions,
|
|
135
|
+
providedOptionsSources
|
|
136
|
+
};
|
|
108
137
|
};
|
|
109
138
|
export {
|
|
110
139
|
findMissingOptions,
|
|
111
140
|
partialParse
|
|
112
141
|
};
|
|
113
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/noop.ts", "../src/index.ts"],
  "sourcesContent": ["// eslint-disable-next-line @typescript-eslint/no-empty-function\nexport const noop = () => {};\n", "import { noop } from \"./noop.ts\";\nimport camelCase from \"camelcase\";\nimport {\n  type OptionValues,\n  Command,\n  type ParseOptions,\n  Option,\n} from \"commander\";\n\nexport const findMissingOptions = (\n  command: Command,\n  providedOptionsByCommand: Map<Command, OptionValues>,\n) => {\n  const missingOptionsByCommand = new Map<Command, Set<string>>();\n\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  let currentCommand: Command | undefined | null = command;\n\n  while (currentCommand) {\n    const missingOptions = new Set<string>();\n    const providedOptions = providedOptionsByCommand.get(currentCommand);\n    for (const option of currentCommand.options) {\n      const key = option.negate\n        ? camelCase(option.long!.replace(/^--no-/, \"\"))\n        : camelCase((option.long ?? option.short)!);\n\n      if (providedOptions?.[key] !== undefined) {\n        continue;\n      }\n\n      missingOptions.add(key);\n    }\n\n    missingOptionsByCommand.set(currentCommand, missingOptions);\n    currentCommand = currentCommand.parent;\n  }\n\n  return missingOptionsByCommand;\n};\n\nexport type PartialParseResult = {\n  /**\n   * The command whose action will be executed\n   */\n  matchedCommand: Command | undefined;\n  /**\n   * A map of commands to a set of missing options for that command\n   */\n  missingOptions: Map<Command, Set<string>>;\n  /**\n   * A map of commands to the options provided for that command\n   */\n  providedOptions: Map<Command, OptionValues>;\n};\n\nconst copyCommandSettings = (source: Command, target: Command) => {\n  for (const keysToCopy of [\n    \"_allowExcessArguments\",\n    \"_allowUnknownOption\",\n    \"_args\",\n    \"_combineFlagAndOptionalValue\",\n    \"_defaultCommandName\",\n    \"_enablePositionalOptions\",\n    \"_passThroughOptions\",\n  ] as const) {\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n    (target as any)[keysToCopy] = (source as any)[keysToCopy];\n  }\n\n  target.name(source.name());\n  target.aliases(source.aliases());\n  target.version(\n    (source as Command & { version: () => string }).version(),\n    source.options.find(\n      (option) =>\n        option.attributeName() ===\n        (source as Command & { _versionOptionName: string })._versionOptionName,\n    )?.flags,\n  );\n};\n\nconst disableCommandOutput = (command: Command) => {\n  command.configureOutput({\n    writeOut: noop,\n    writeErr: noop,\n    outputError: noop,\n  });\n};\n\nconst cloneOption = (option: Option) => {\n  const newOption = new Option(option.flags, option.description);\n  newOption.makeOptionMandatory(false);\n  newOption.default(option.defaultValue, option.defaultValueDescription);\n\n  newOption.preset((option as Option & { presetArg: unknown }).presetArg);\n  newOption.env((option as Option & { envVar: string }).envVar);\n  if (option.parseArg) {\n    newOption.argParser(option.parseArg);\n  }\n\n  if (option.argChoices) {\n    newOption.choices(option.argChoices);\n  }\n\n  newOption.conflicts(\n    (option as Option & { conflictsWith: string[] }).conflictsWith,\n  );\n\n  (newOption as Option & { implied: unknown }).implied = (\n    option as Option & { implied: unknown }\n  ).implied;\n\n  return newOption;\n};\n\n/**\n * Partially parse argv for a command without executing the action. @see {@link Command.parse}\n *\n * @returns An object containing the matched command, the provided options, and the missing options.\n */\nexport const partialParse = (\n  command: Command,\n  argv: readonly string[],\n  options?: ParseOptions,\n): PartialParseResult => {\n  const providedOptions = new Map<Command, OptionValues>();\n  const commandsMap = new Map<Command, Command>();\n  let matchedCommand: Command | undefined;\n\n  const createParserCommand = (parserCommand: Command, command: Command) => {\n    commandsMap.set(parserCommand, command);\n\n    copyCommandSettings(command, parserCommand);\n    disableCommandOutput(parserCommand);\n    parserCommand.exitOverride();\n\n    for (const option of command.options) {\n      parserCommand.addOption(cloneOption(option));\n    }\n\n    parserCommand.hook(\"preSubcommand\", (thisCommand, actionCommand) => {\n      providedOptions.set(commandsMap.get(parserCommand)!, thisCommand.opts());\n      providedOptions.set(commandsMap.get(actionCommand)!, thisCommand.opts());\n    });\n\n    parserCommand.action(() => {\n      providedOptions.set(\n        commandsMap.get(parserCommand)!,\n        parserCommand.opts(),\n      );\n      matchedCommand = command;\n    });\n\n    for (const subcommand of command.commands as Command[]) {\n      const parserSubcommand = parserCommand.command(subcommand.name());\n      createParserCommand(parserSubcommand, subcommand);\n    }\n\n    return parserCommand;\n  };\n\n  const parserCommand = createParserCommand(new Command(), command);\n  parserCommand.parse(argv, options);\n\n  const missingOptions = matchedCommand\n    ? findMissingOptions(matchedCommand, providedOptions)\n    : new Map<Command, Set<string>>();\n\n  return { matchedCommand, missingOptions, providedOptions };\n};\n"],
  "mappings": "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",
  "names": ["parserCommand", "command"]
}

|
|
142
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/noop.ts", "../src/index.ts"],
  "sourcesContent": ["// eslint-disable-next-line @typescript-eslint/no-empty-function\nexport const noop = () => {};\n", "import { noop } from \"./noop.ts\";\nimport camelCase from \"camelcase\";\nimport {\n  type OptionValues,\n  Command,\n  type ParseOptions,\n  Option,\n  type OptionValueSource,\n} from \"commander\";\n\nexport const findMissingOptions = (\n  command: Command,\n  providedOptionsByCommand: Map<Command, OptionValues>,\n) => {\n  const missingOptionsByCommand = new Map<Command, Set<string>>();\n\n  // eslint-disable-next-line @typescript-eslint/ban-types\n  let currentCommand: Command | undefined | null = command;\n\n  while (currentCommand) {\n    const missingOptions = new Set<string>();\n    const providedOptions = providedOptionsByCommand.get(currentCommand);\n    for (const option of currentCommand.options) {\n      const key = option.negate\n        ? camelCase(option.long!.replace(/^--no-/, \"\"))\n        : camelCase((option.long ?? option.short)!);\n\n      if (providedOptions?.[key] !== undefined) {\n        continue;\n      }\n\n      missingOptions.add(key);\n    }\n\n    missingOptionsByCommand.set(currentCommand, missingOptions);\n    currentCommand = currentCommand.parent;\n  }\n\n  return missingOptionsByCommand;\n};\n\nexport type PartialParseResult = {\n  /**\n   * The command whose action will be executed\n   */\n  matchedCommand: Command | undefined;\n  /**\n   * A map of commands to a set of missing options for that command\n   */\n  missingOptions: Map<Command, Set<string>>;\n  /**\n   * A map of commands to the options provided for that command\n   */\n  providedOptions: Map<Command, OptionValues>;\n  /**\n   * A map of commands to a map of option keys to the source of the option value\n   */\n  providedOptionsSources: Map<Command, Map<string, string | undefined>>;\n};\n\nconst copyCommandSettings = (source: Command, target: Command) => {\n  for (const keysToCopy of [\n    \"_allowExcessArguments\",\n    \"_allowUnknownOption\",\n    \"_args\",\n    \"_combineFlagAndOptionalValue\",\n    \"_defaultCommandName\",\n    \"_enablePositionalOptions\",\n    \"_passThroughOptions\",\n  ] as const) {\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment\n    (target as any)[keysToCopy] = (source as any)[keysToCopy];\n  }\n\n  target.name(source.name());\n  target.aliases(source.aliases());\n  target.version(\n    (source as Command & { version: () => string }).version(),\n    source.options.find(\n      (option) =>\n        option.attributeName() ===\n        (source as Command & { _versionOptionName: string })._versionOptionName,\n    )?.flags,\n  );\n};\n\nconst disableCommandOutput = (command: Command) => {\n  command.configureOutput({\n    writeOut: noop,\n    writeErr: noop,\n    outputError: noop,\n  });\n};\n\nconst cloneOption = (option: Option) => {\n  const newOption = new Option(option.flags, option.description);\n  newOption.makeOptionMandatory(false);\n  newOption.default(option.defaultValue, option.defaultValueDescription);\n\n  newOption.preset((option as Option & { presetArg: unknown }).presetArg);\n  newOption.env((option as Option & { envVar: string }).envVar);\n  if (option.parseArg) {\n    newOption.argParser(option.parseArg);\n  }\n\n  if (option.argChoices) {\n    newOption.choices(option.argChoices);\n  }\n\n  newOption.conflicts(\n    (option as Option & { conflictsWith: string[] }).conflictsWith,\n  );\n\n  (newOption as Option & { implied: unknown }).implied = (\n    option as Option & { implied: unknown }\n  ).implied;\n\n  return newOption;\n};\n\n/**\n * Partially parse argv for a command without executing the action. @see {@link Command.parse}\n *\n * @returns An object containing the matched command, the provided options, and the missing options.\n */\nexport const partialParse = (\n  command: Command,\n  argv: readonly string[],\n  options?: ParseOptions,\n): PartialParseResult => {\n  const providedOptions = new Map<Command, OptionValues>();\n  const providedOptionsSources = new Map<\n    Command,\n    Map<string, string | undefined>\n  >();\n  const commandsMap = new Map<Command, Command>();\n  let matchedCommand: Command | undefined;\n\n  const setProvidedOptionSource = (\n    command: Command,\n    optionKey: string,\n    source: OptionValueSource | undefined,\n  ) => {\n    if (!source) {\n      return;\n    }\n\n    const sourcesMap =\n      providedOptionsSources.get(command) ?? new Map<string, string>();\n    sourcesMap.set(optionKey, source);\n    providedOptionsSources.set(command, sourcesMap);\n  };\n\n  const createParserCommand = (parserCommand: Command, command: Command) => {\n    commandsMap.set(parserCommand, command);\n\n    copyCommandSettings(command, parserCommand);\n    disableCommandOutput(parserCommand);\n    parserCommand.exitOverride();\n\n    for (const option of command.options) {\n      parserCommand.addOption(cloneOption(option));\n    }\n\n    parserCommand.hook(\"preSubcommand\", (thisCommand, actionCommand) => {\n      for (const cmd of [thisCommand, actionCommand]) {\n        providedOptions.set(commandsMap.get(cmd)!, cmd.opts());\n\n        for (const optionKey of Object.keys(cmd.opts())) {\n          setProvidedOptionSource(\n            cmd,\n            optionKey,\n            cmd.getOptionValueSource(optionKey),\n          );\n        }\n      }\n    });\n\n    parserCommand.action(() => {\n      providedOptions.set(\n        commandsMap.get(parserCommand)!,\n        parserCommand.opts(),\n      );\n\n      for (const optionKey of Object.keys(parserCommand.opts())) {\n        setProvidedOptionSource(\n          commandsMap.get(parserCommand)!,\n          optionKey,\n          parserCommand.getOptionValueSource(optionKey),\n        );\n      }\n\n      matchedCommand = command;\n    });\n\n    for (const subcommand of command.commands as Command[]) {\n      const parserSubcommand = parserCommand.command(subcommand.name());\n      createParserCommand(parserSubcommand, subcommand);\n    }\n\n    return parserCommand;\n  };\n\n  const parserCommand = createParserCommand(new Command(), command);\n  parserCommand.parse(argv, options);\n\n  const missingOptions = matchedCommand\n    ? findMissingOptions(matchedCommand, providedOptions)\n    : new Map<Command, Set<string>>();\n\n  return {\n    matchedCommand,\n    missingOptions,\n    providedOptions,\n    providedOptionsSources,\n  };\n};\n"],
  "mappings": "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",
  "names": ["command", "parserCommand"]
}

|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "parse-my-command",
|
|
3
|
-
"version": "0.1
|
|
3
|
+
"version": "0.2.1",
|
|
4
4
|
"description": "Parse argv with Commander.js without executing the command",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"parser",
|
|
@@ -60,8 +60,8 @@
|
|
|
60
60
|
"devDependencies": {
|
|
61
61
|
"@trivago/prettier-plugin-sort-imports": "^4.2.0",
|
|
62
62
|
"@types/node": "^20.5.9",
|
|
63
|
-
"@typescript-eslint/eslint-plugin": "^6.
|
|
64
|
-
"@typescript-eslint/parser": "^6.
|
|
63
|
+
"@typescript-eslint/eslint-plugin": "^6.6.0",
|
|
64
|
+
"@typescript-eslint/parser": "^6.6.0",
|
|
65
65
|
"c8": "^8.0.1",
|
|
66
66
|
"eslint": "^8.48.0",
|
|
67
67
|
"eslint-config-prettier": "^9.0.0",
|