Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/stypox/arg-parser
Argument parser for modern C++: no runtime overhead; auto-generated help screen.
https://github.com/stypox/arg-parser
argparse argparser argument-parser argument-parsing arguments command-line commandline compile-time compiler-optimization cpp-library cpp17 cpp20 local localstorage overhead reference-data
Last synced: 9 days ago
JSON representation
Argument parser for modern C++: no runtime overhead; auto-generated help screen.
- Host: GitHub
- URL: https://github.com/stypox/arg-parser
- Owner: Stypox
- License: mit
- Created: 2019-01-30T19:27:08.000Z (almost 6 years ago)
- Default Branch: master
- Last Pushed: 2021-08-26T16:30:41.000Z (about 3 years ago)
- Last Synced: 2023-03-08T18:53:35.475Z (over 1 year ago)
- Topics: argparse, argparser, argument-parser, argument-parsing, arguments, command-line, commandline, compile-time, compiler-optimization, cpp-library, cpp17, cpp20, local, localstorage, overhead, reference-data
- Language: C++
- Homepage:
- Size: 105 KB
- Stars: 7
- Watchers: 2
- Forks: 2
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Argument parser
A **C++** library that **parses command line arguments** and automatically generates a **help screen**. The initialization of the parser should not impose **any run-time overhead**.
# Features
## **Options**
There are three different options that can be used:
- **Switch option**: they are not followed by any value (e.g. `--help`). When used they set the underlying reference to a provided value.
- **Option**: they except an integer, a decimal number or some text (e.g. `--size=50 --gravity=9.8 --say=Hello!`).
- The type `T` of the underlying reference must meet one of these requirements:
- `std::is_integer`: excepts an integer that does not overflow/underflow `T` limits.
Represented by `I` in the help screen.
- `std::is_floating_point`: excepts a decimal number that does not overflow/underflow `T` limits.
Represented by `D` in the help screen.
- `std::is_convertible`: excepts some text.
Represented by `T` in the help screen.
- excepts a valid value, checked using a **user-defined validation function** (if present).
- **Manual option**: they except an arbitrary string which is manipulated in a non-standard way. (e.g. `--html=Hello!
`)
- Requires a **user-defined function** to convert the string to the type of the underlying reference.
- Represented by `S` in the help screen.
Every option has these attributes:
- ***name***: used for error reporting;
- ***reference*** to a variable: the output in which to save the inserted value;
- one or more ***arguments***: what the user has to pass in the console to set the value of the option;
- ***description***: used in the help screen;
- whether it is ***required*** or not; represented by `*` in the help screen when set to `true`;## **Error checking and reporting**
During the parsing process every argument has to meet these requirements:
- every argument must have a **corresponding option** (this does not apply if positional arguments are valid);
- that option should **not have already been encountered**;
- if the option excepts a **value**, the argument must contain one (but empty texts/strings are ok);
- the value has to be **convertible** to the underlying variable type (either normally or via the user-defined function);
- for *normal options*, the value must **not overflow** (this only applies to integers and decimal numbers)During the validation process every computed option has to meet these requirements:
- if the option is required, it must **have been encountered**;
- for *normal options*, the value must **pass the validity check**, represented by the user-defined function (that passes by default);The parsing process and the validation process are **separate**, so that even if an option is invalid no error is generated until the validation starts. This is useful, for example, to display the help screen when `--help` is provided, even if other options are invalid. Every error contains an **thorough description** about what caused it.
## **Help screen**
See [below](#output) for an example help screen.
- Titles and lines of description can be added to the help screen by providing **help sections**.
- The **indentation** of the description of sections can be changed. When the indentation is not enough a newline is added between the arguments and the description
- The first argument is considered, by default, the **executable path**, but this can be manually changed. The executable path is used for the help screen.# Installation
Just **download** the header file `argparser.hpp` and **`#include`** it into your project! If you want to `#include` it as `` you need to add `-IPATH/TO/arg-parser/include` to your compiler options.
Note: it requires C++17, so add to your compiler options `-std=c++17`. C++20 is also supported, along with `requires` clauses.# Documentation
All types are defined in `namespace stypox`. To simplify reading, `std::` is omitted before `tuple`, `string`, `string_view`, `vector` and `array`; also `const Type&` is written just `Type`. Read the code for more precise details.
## **ArgParser**
`ArgParser` is the class that does the job of parsing arguments.### ArgParser::ArgParser()
`(tuple options, string_view programName, size_t descriptionIndentation = 25)`
Constructs the ArgParser object. `Options...` must be made only of `SwitchOption`, `Option`, `ManualOption` or `HelpSection`. The `tuple` can be instantiated using `std::make_tuple(Options...)`.### ArgParser::parse()
(1) `void (Iter first, Iter last, bool firstArgumentIsExecutablePath)`
(2) `void (int argc, char const* argv[], bool firstArgumentIsExecutablePath = true)`
Parses all the arguments in range [first, last) (1) / [argv, argv+argc) (2), reports parsing errors (by throwing `std::runtime_error`) as described [above](#options), saves the new values for options. Throws `std::out_of_range` if `firstArgumentIsExecutablePath` is set to `true` but the list of arguments is empty.### ArgParser::parsePositional()
(1) `vector (Iter first, Iter last, bool firstArgumentIsExecutablePath)`
(2) `vector (int argc, char const* argv[], bool firstArgumentIsExecutablePath = true)`
Parses all the arguments in range [first, last) (1) / [argv, argv+argc) (2), reports parsing errors (by throwing `std::runtime_error`) as described [above](#options), saves the new values for options. Returns the arguments that didn't match any option. Throws `std::out_of_range` if `firstArgumentIsExecutablePath` is set to `true` but the list of arguments is empty.### ArgParser::validate()
`void ()`
Reports logical errors (by throwing `std::runtime_error`) as described [above](#error-checking-and-reporting).### ArgParser::reset()
`void ()`
Every argument is set as if it had never been encountered.### ArgParser::usage()
`string ()`
Returns the usage screen. See the [output of the code below](#output) for an example.### ArgParser::help()
`string ()`
Returns the help screen. The indentation of the description of options can be set in the constructor. See the [output of the code below](#output) for an example.## SwitchOption, Option, ManualOption, HelpSection
`HelpSection`
`SwitchOption`, `Option` and `ManualOption` are the classes that keep information about every option. The difference between them is explained [above](#options). The array of possible arguments (of size `N`) can be initialized using `stypox::args()`.
`HelpSection` is a class that holds a string of text to be printed in the help screen.### args()
`array args(Args... list)`
Builds an array of possible arguments using the provided `list` (needed for the constructors of options).### SwitchOption::SwitchOption()
(when T is bool) `(string_view name, T& output, array arguments, string_view help, T valueWhenSet = true, required = false)`
(when T is not bool) `(string_view name, T& output, array arguments, string_view help, T valueWhenSet, required = false)`
`SwitchOption`'s constructor. When parsing, the value will be saved in `output`.### Option::Option()
`(string_view name, T& output, array arguments, string_view help, required = false, F validityChecker = [](){ return true; })`
`Option`'s constructor. When parsing, the value will be saved in `output`. When validating `validityChecker` is called with `(output)` (it must return `bool`). See [above](#options) to read about the valid types `T`.### ManualOption::ManualOption()
`(string_view name, T& output, array arguments, string_view help, F assignerFunctor, required = false)`
`ManualOption`'s constructor. When parsing, the value, converted to `T` by calling `assignerFunctor(string_view)`, will be saved in `output`.### HelpSection::HelpSection()
`(string_view title)`
`HelpSection`'s constructor. When generating the help screen `title` is appended to it followed by `\n`.# Example
```cpp
#include
#include "arg-parser/include/stypox/argparser.hpp"int main(int argc, char const* argv[]) {
// initialize variables with the default values
bool help = false;
bool usage = false;
int exitCode = 0;
float cake = 0.5f;// a default value would be useless, since the corresponding option is required
std::pair person;stypox::ArgParser p{
std::make_tuple(
stypox::HelpSection{"\nHelp options:"},
stypox::SwitchOption{"help", help, stypox::args("-?", "-h", "--help"), "show help screen"},
stypox::SwitchOption{"usage", usage, stypox::args("-u", "--usage"), "show usage screen"},stypox::HelpSection{"\nOther important options:"},
stypox::SwitchOption{"exitWithError", exitCode, stypox::args("--exit-with-error"), "set exit code to 1",
1 /* value to set to variable @exitCode when the option is used*/},
stypox::Option{"cake", cake, stypox::args("-c=", "--cake="), "how much cake will you take? (0-1; default=0.5)",
false, // option not required
[](float value){ // validation function
return value >= 0.0f && value <= 1.0f;
}},
stypox::ManualOption{"person", person, stypox::args("-p=", "--person="), "a person (format: `name;age`)",
[](const std::string_view& str) { // conversion function
try {
size_t semicolonIndex = str.find_first_of(";");
if (semicolonIndex == std::string::npos)
throw std::runtime_error{""};std::string name{str.substr(0, semicolonIndex)},
age{str.substr(semicolonIndex + 1, str.size() - semicolonIndex - 1)};
// the type of the returned value matches the type of variable @person
return std::pair{name, std::stoi(age)};
} catch (...) {
throw std::runtime_error{"Invalid person " + std::string{str}};
}
},
true /* option required */}
),
"Example program by Stypox"
};p.parse(argc, argv);
// check for help before validating
if (help) {
std::cout << p.help();
return 0;
}
if (usage) {
std::cout << p.usage();
return 0;
}p.validate();
std::cout << "Here is a piece of cake of size " << cake << "!\n"
<< person.first << " is now my best friend: he is " << person.second << " years old :-D\n";return exitCode;
}
```
### Output
Output for different commands (`; echo 'Exit code: '$?;` is only there to print the exit code):
```sh
$ ./executable; echo 'Exit code: '$?;
terminate called after throwing an instance of 'std::runtime_error'
what(): Option person is required
Aborted
Exit code: 134$ ./executable --help; echo 'Exit code: '$?;
Example program by Stypox
Legend: I=integer; D=decimal; T=text; S=custom string; *=required;
Usage: ./executable [-?] [-u] [--exit-with-error] [-c=D] -p=SHelp options:
-? -h --help show help screen
-u --usage show usage screenOther important options:
--exit-with-error set exit code to 1
-c=D --cake=D how much cake will you take? (0-1; default=0.5)
-p=S --person=S *a person (format: `name;age`)Exit code: 0
$./executable --usage; echo 'Exit code: '$?;
Example program by Stypox
Legend: I=integer; D=decimal; T=text; S=custom string; *=required;
Usage: ./executable [-?] [-u] [--exit-with-error] [-c=D] -p=S
Exit code: 0$./executable --cake=0.7245 --person="John Smith;17" --exit-with-error; echo 'Exit code: '$?;
Here is a piece of cake of size 0.7245!
John Smith is now my best friend: he is 17 years old :-D
Exit code: 1$ ./executable --cak; echo 'Exit code: '$?;
terminate called after throwing an instance of 'std::runtime_error'
what(): Unknown argument: --cak
Aborted
Exit code: 134$ ./executable --cake=1.2; echo 'Exit code: '$?;
terminate called after throwing an instance of 'std::runtime_error'
what(): Option cake: value 1.200000 is not allowed
Aborted
Exit code: 134$ ./executable --cake=1.2e10000; echo 'Exit code: '$?;
terminate called after throwing an instance of 'std::runtime_error'
what(): Option cake: out of range decimal "1.2e10000" (must be between 0.000000 and 340282346638528859811704183484516925440.000000): --cake=1.2e10000
Aborted
Exit code: 134
```