Subcommands and the App

Subcommands are keyword that invoke a new set of options and features. For example, the git command has a long series of subcommands, like add and commit. Each can have its own options and implementations. This chapter will focus on implementations that are contained in the same C++ application, though the system git uses to extend the main command by calling other commands in separate executables is supported too; that's called "Prefix commands" and is included at the end of this chapter.

The parent App

We'll start by discussing the parent App. You've already used it quite a bit, to create options and set option defaults. There are several other things you can do with an App, however.

You are given a lot of control the help output. You can set a footer with app.footer("My Footer"). You can replace the default help print when a ParseError is thrown with app.failure_message(CLI::FailureMessage::help). The default is CLI:::FailureMessage::simple, and you can easily define a new one. Just make a (lambda) function that takes an App pointer and a reference to an error code (even if you don't use them), and returns a string.

Adding a subcommand

Subcommands can be added just like an option:

CLI::App* sub = app.add_subcommand("sub", "This is a subcommand");

The subcommand should have a name as the first argument, and a little description for the second argument. A pointer to the internally stored subcommand is provided; you usually will be capturing that pointer and using it later (though you can use callbacks if you prefer). As always, feel free to use auto sub = ... instead of naming the type.

You can check to see if the subcommand was received on the command line several ways:

if(*sub) ...
if(sub->parsed()) ...
if(app.got_subcommand(sub)) ...
if(app.got_subcommand("sub")) ...

You can also get a list of subcommands with get_subcommands(), and they will be in parsing order.

There are a lot of options that you can set on a subcommand; in fact, subcommands have exactly the same options as your main app, since they are actually the same class of object (as you may have guessed from the type above). This has the pleasant side affect of making subcommands infinitely nestable.

Required subcommands

Each App has controls to set the number of subcommands you expect. This is controlled by:

app.require_subcommand(/* min */ 0, /* max */ 1);

If you set the max to 0, CLI11 will allow an unlimited number of subcommands. After the (non-unlimited) maximum is reached, CLI11 will stop trying to match subcommands. So the if you pass "one two" to a command, and both one and two are subcommands, it will depend on the maximum number as to whether the "two" is a subcommand or an argument to the "one" subcommand.

As a shortcut, you can also call the require_subcommand method with one argument; that will be the fixed number of subcommands if positive, it will be the maximum number if negative. Calling it without an argument will set the required subcommands to 1 or more.

The maximum number of subcommands is inherited by subcommands. This allows you to set the maximum to 1 once at the beginning on the parent app if you only want single subcommands throughout your app. You should keep this in mind, if you are dealing with lots of nested subcommands.

Using callbacks

You've already seen how to check to see what subcommands were given. It's often much easier, however, to just define the code you want to run when you are making your parser, and not run a bunch of code after CLI11_PARSE to analyse the state (Procedural! Yuck!). You can do that with lambda functions. A std::function<void()> callback .callback() is provided, and CLI11 ensures that all options are prepared and usable by reference capture before entering the callback. An example is shown below in the geet program.

Inheritance of defaults

The following values are inherited when you add a new subcommand. This happens at the point the subcommand is created:

  • The name and description for the help flag
  • The footer
  • The failure message printer function
  • Option defaults
  • Allow extras
  • Prefix command
  • Ignore case
  • Ignore underscore
  • Allow Windows style options
  • Fallthrough
  • Group name
  • Max required subcommands
  • validate positional arguments
  • validate optional arguments

Special modes

There are several special modes for Apps and Subcommands.

Allow extras

Normally CLI11 throws an error if you don't match all items given on the command line. However, you can enable allow_extras() to instead store the extra values in .remaining(). You can get all remaining options including those in contained subcommands recursively in the original order with .remaining(true). .remaining_size() is also provided; this counts the size but ignores the -- special separator if present.

Fallthrough

Fallthrough allows an option that does not match in a subcommand to "fall through" to the parent command; if that parent allows that option, it matches there instead. This was added to allow CLI11 to represent models:

gitbook:code $ ./my_program my_model_1 --model_flag --shared_flag

Here, --shared_flag was set on the main app, and on the command line it "falls through" my_model_1 to match on the main app. This is set through ->fallthrough() on a subcommand.

Subcommand fallthrough

Subcommand fallthrough allows additional subcommands to be triggered after the first subcommand. By default subcommand fallthrough is enabled, but it can be turned off through ->subcommand_fallthrough(false) on a subcommand. This will prevent additional subcommands at the same inheritance level from triggering, the strings would then be treated as positional values. As a technical note if fallthrough is enabled but subcommand fallthrough disabled (this is not the default in both cases), then subcommands on grandparents can still be triggered from the grandchild subcommand, unless subcommand fallthrough is also disabled on the parent. This is an unusual circumstance but may arise in some very particular situations.

Prefix command

This is a special mode that allows "prefix" commands, where the parsing completely stops when it gets to an unknown option. Further unknown options are ignored, even if they could match. Git is the traditional example for prefix commands; if you run git with an unknown subcommand, like "git thing", it then calls another command called "git-thing" with the remaining options intact.

Silent subcommands

Subcommands can be modified by using the silent option. This will prevent the subcommand from showing up in the get_subcommands list. This can be used to make subcommands into modifiers. For example, a help subcommand might look like

    auto sub1 = app.add_subcommand("help")->silent();
    sub1->parse_complete_callback([]() { throw CLI::CallForHelp(); });

This would allow calling help such as:

./app help
./app help sub1

Positional Validation

Some arguments supplied on the command line may be legitamately applied to more than 1 positional argument. In this context enabling positional_validation on the application or subcommand will check any validators before applying the command line argument to the positional option. It is not an error to fail validation in this context, positional arguments not matching any validators will go into the extra_args field which may generate an error depending on settings.

Optional Argument Validation

Similar to positional validation, there are occasional contexts in which case it might be ambiguous whether an argument should be applied to an option or a positional option.

    std::vector<std::string> vec;
    std::vector<int> ivec;
    app.add_option("pos", vec);
    app.add_option("--args", ivec)->check(CLI::Number);
    app.validate_optional_arguments();

In this case a sequence of integers is expected for the argument and remaining strings go to the positional string vector. Without the validate_optional_arguments() active it would be impossible get any later arguments into the positional if the --args option is used. The validator in this context is used to make sure the optional arguments match with what the argument is expecting and if not the -args option is closed, and remaining arguments fall into the positional.

results matching ""

    No results matching ""