2018-10-24 04:39:04 +02:00
|
|
|
import initState from './initState';
|
2018-10-25 06:42:39 +02:00
|
|
|
import IOption from './interfaces/IOption';
|
2018-10-24 18:19:36 +02:00
|
|
|
import parseArgument from './parseArgument';
|
2018-10-24 04:39:04 +02:00
|
|
|
import runPreParseHook from './runPreParseHook';
|
2018-10-23 23:17:03 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks the command line arguments against the given specification, and returns the arguments as a JavaScript object, as well as a failure function that prints the usage instructions before terminating.
|
|
|
|
* If an error is found, outputs an error message and terminates with a non-zero exit code.
|
|
|
|
* @param spec A specification of what arguments are expected.
|
|
|
|
* @param preParseHook Optionally, a function to modify the command line arguments before parsing them.
|
|
|
|
*/
|
|
|
|
export default function parseArguments(
|
|
|
|
spec: { [key: string]: IOption },
|
2018-10-25 06:13:24 +02:00
|
|
|
preParseHook?: (args: string[]) => (string[] | undefined),
|
2018-10-25 04:01:00 +02:00
|
|
|
): { fail: (errorMessage: string) => never, args: { [key: string]: string } } {
|
2018-10-24 04:39:04 +02:00
|
|
|
//Initialize state
|
2018-10-24 04:44:49 +02:00
|
|
|
const { outputArgs, shortToLongLookup, requiredOptions, fail } = initState(spec);
|
2018-10-25 06:13:24 +02:00
|
|
|
try {
|
|
|
|
const args = runPreParseHook({ args: process.argv.slice(2), preParseHook });
|
2018-10-23 23:17:03 +02:00
|
|
|
|
2018-10-25 06:13:24 +02:00
|
|
|
//Iterate through all command line arguments. When we have read both name and value, verify the argument for correctness.
|
|
|
|
//Show error if a name has no value afterwards, or a value has no name in front of it.
|
|
|
|
const lastOption = args.reduce(parseArgument.bind(null, { spec, outputArgs, requiredOptions, shortToLongLookup }));
|
2018-10-23 23:17:03 +02:00
|
|
|
|
2018-10-25 06:13:24 +02:00
|
|
|
// argumentName must be cleared to '' after the value is read, so if it is not an empty string, the value was missing
|
|
|
|
if (lastOption !== '') {
|
|
|
|
throw new Error(`Option "${lastOption}" was not followed by a value.`);
|
|
|
|
}
|
2018-10-23 23:17:03 +02:00
|
|
|
|
2018-10-25 06:13:24 +02:00
|
|
|
//check if any entry in requiredArguments was not set
|
2018-10-25 06:42:39 +02:00
|
|
|
Object.entries(requiredOptions).filter(([, wasFound]) => wasFound === false).forEach(([option]) => {
|
|
|
|
throw new Error(`Missing option "${option}" even though it is required.`);
|
|
|
|
});
|
2018-10-25 06:13:24 +02:00
|
|
|
} catch (error) {
|
|
|
|
fail((error as Error).message);
|
2018-10-23 23:17:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
2018-10-24 04:44:49 +02:00
|
|
|
fail,
|
2018-10-23 23:17:03 +02:00
|
|
|
args: outputArgs,
|
|
|
|
};
|
|
|
|
}
|