An open API service indexing awesome lists of open source software.

https://github.com/juzzlin/argengine

A simple CLI argument (option) parser library for modern C++ based on lambda callbacks
https://github.com/juzzlin/argengine

argument-parser argument-parsing command-line-parser cplusplus cplusplus-11 cplusplus-17 lambda-functions library linux modern-cpp option-parser windows

Last synced: 10 months ago
JSON representation

A simple CLI argument (option) parser library for modern C++ based on lambda callbacks

Awesome Lists containing this project

README

          

Argengine
=========

Looking for a simple CLI argument parser for your C++ project? `Argengine` might be for you.

# Features

* Automatic help generation
* Based on simple lambda callbacks
* Builds with CMake
* Designed for modern C++
* Extensive unit tests
* Single .hpp/.cpp
* Very easy to use

# Design Choices

* Argument values are passed as strings. The client is responsible for possible data conversions.
* Most of the data is passed by value for a cleaner API. It is assumed that argument parsing is not the performance bottleneck of your application
* The PIMPL idiom is used to maintain API/ABI compatibility

# Installation

## Use from sources

Just add `src/argengine.hpp` and `src/argengine.cpp` to your project and start using it!

## Use as a CMake subproject

Copy contents of `Argengine` under your main project (or clone as a Git submodule).

In your `CMakeLists.txt`:

```
add_subdirectory(Argengine EXCLUDE_FROM_ALL)
include_directories(Argengine/src)
```

Link to the library:

```
target_link_libraries(${YOUR_TARGET_NAME} Argengine_static)
```

In your code:

```
#include "argengine.hpp"
```

## Use as a library

Build and install:

`$ mkdir build && cd build`

`$ cmake ..`

`$ make`

`$ sudo make install`

Link to `libArgengine_static.a` or `libArgengine.so`.

# Usage In A Nutshell

The basic principle is that for each option a lambda callback is added.

A valueless callback:

```
...

juzzlin::Argengine ae(argc, argv);
ae.addOption({"-f", "--foo"}, [] {
// Do something
});
ae.parse();

...
```

A single-value callback:

```
...

juzzlin::Argengine ae(argc, argv);
ae.addOption({"-f", "--foo"}, [] (std::string value) {
// Do something with value
});
ae.parse();

...
```

There can be as many option variants as liked, usually the short and long versions, e.g `-f` and `--foo`.

`Argengine` doesn't care about the naming of the options and they can be anything: `-f`, `a`, `/c`, `foo`, `--foo` ...

Positional arguments (for example a file name for a text editor after other options) can be received with a single callback:

```
...

juzzlin::Argengine ae(argc, argv);
ae.setPositionalArgumentCallback([] (std::vector args) {
// Do something with arguments
});
ae.parse();

...
```

If the callback for positional arguments is set, then no errors about `unknown options` will occur as all additional options will be taken as positional arguments.

# Help

By default, `Argengine` will create a simple help that is shown with `-h` or `--help`.

Without any additional options a possible output will look like this:

```
Usage: ./ex1 [OPTIONS]

Options:

-h, --help Show this help.
```

The help can be manually printed with `Argengine::printHelp()`.

The default help can be disabled by constructing `Argengine` with `Argengine::Argengine(argc, argv, false)`.

## Custom help

A custom help can be added with:

`void Argengine::addHelp(OptionVariants optionVariants, ValuelessCallback callback)`

You'd still very likely want to call `Argengine::printHelp()` in the callback and just add some stuff around it.

## Sorting order of options

The sorting order of options can be selected with:

`void Argengine::setHelpSorting(HelpSorting helpSorting)`

## Custom help text

The text printed before options can be set with:

`void Argengine::setHelpText(std::string helpText)`

# Examples

## Valueless options: The simplest possible example

Valueless options are options without any value, so they are just flags. The lambda callback is of the form `[] {}`.

```
#include "argengine.hpp"
#include

int main(int argc, char ** argv)
{
juzzlin::Argengine ae(argc, argv);
ae.addOption({"-f", "--foo"}, [] {
std::cout << "'-f' or '--foo' given!" << std::endl;
});
ae.parse();

return 0;
}
```

## Single-value options: The simplest possible example

Single-value options are options that can have only one value.

As an example, for option `-f` The following formats are allowed: `-f 42`, `-f42`, `-f=42`.

Preferably there should be either space or '`=`'. The spaceless format is accepted if not ambiguous.

The lambda callback is of the form `[] (std::string value) {}`.

```
#include "argengine.hpp"
#include

int main(int argc, char ** argv)
{
juzzlin::Argengine ae(argc, argv);
ae.addOption({"-f", "--foo"}, [] (std::string value) {
std::cout << "Value for '-f' or '--foo': " << value << std::endl;
});
ae.parse();

return 0;
}
```

## General: Marking an option **required**

In order to mark an option mandatory, there's an overload that accepts `bool required` right after the callback:

```
...

juzzlin::Argengine ae(argc, argv);
ae.addOption({"-f", "--foo"}, [] {
// Do something
},
true); // Required

...
```

## General: Adding option documentation for the auto-generated help

One can add option documentation for the auto-generated help.

Consider this example code:

```
...

juzzlin::Argengine ae(argc, argv);
ae.addOption(
{ "-p" }, [](std::string value) {
std::cout << value.size() << std::endl;
},
true, "Print length of given text. This option is required.", "TEXT");

...
```

Here we have added a documentation string `Print length of given text. This option is required.` and also a variable name `TEXT` for option `-p`.

Thus, this will generate a help content like this:

```
Options:

-h, --help Show this help.
-p [TEXT] Print length of given text. This option is required.

```

## General: Setting two or more options as conflicting

Sometimes we want to prevent the user from giving a certain set of arguments.

Consider this example code:

```
...

juzzlin::Argengine ae(argc, argv);
ae.addOption(
{ "foo" }, [] {
std::cout << "Foo enabled!" << std::endl;
});

ae.addOption(
{ "bar" }, [] {
std::cout << "Bar enabled!" << std::endl;
});

// Set "foo" and "bar" as conflicting options.
ae.addConflictingOptions({ "foo", "bar" });
...
```

Now, if we give both `foo` and `bar` to the application, we'll get an error like this:

```
Argengine: Conflicting options: 'bar', 'foo'. These options cannot coexist.
```

## General: Adding option groups

Option groups will make the given options depend on each other.

Consider this example code:

```
...

juzzlin::Argengine ae(argc, argv);
ae.addOption(
{ "foo" }, [] {
std::cout << "Foo enabled!" << std::endl;
});

ae.addOption(
{ "bar" }, [] {
std::cout << "Bar enabled!" << std::endl;
});

// Add "foo" and "bar" as an option group.
ae.addOptionGroup({ "foo", "bar" });
...
```

Now, if we give only `foo` to the application, we'll get an error like this:

```
Argengine: These options must coexist: 'bar', 'foo'. Missing options: 'bar'.
```

## General: Error handling

For error handling there are two options: exceptions or error value.

`Argengine::parse()` will throw on error, `Argengine::parse(Error & error)` will set and error.

Example of handling exceptions:

```
...

try {
ae.parse();
} catch(std::runtime_error & e) {
std::cerr << e.what() << std::endl;
ae.printHelp();
return EXIT_FAILURE;
}

...
```

Example of handling error values:

```
...
Argengine::Error error;
ae.parse(error);

if (error.code != Argengine::Error::Code::Ok) {
std::cerr << error.message << std::endl
<< std::endl;
ae.printHelp();
return EXIT_FAILURE;
}

...
```

# Requirements

C++17

# Licence

MIT

# Projects currently using Argengine

* https://github.com/juzzlin/DustRacing2D

* https://github.com/juzzlin/Heimer

* https://github.com/juzzlin/Noteahead