使用boost.program_options创建一个前缀命令

Create a prefix command with Boost.Program_options

本文关键字:一个 前缀 命令 program boost options 创建 使用      更新时间:2023-10-16

我想使用boost.program_options创建一个前缀程序。前缀意味着我的程序位于另一个任意command [args]的前面。因此,我的程序应接受定义的许多关键字参数/标志。第一个位置参数表明命令我在前缀。此命令本身可能是我不知道的任何参数组合,并且可能与程序的参数重叠。因此,第一个位置参数和之后发生的任何事情都应该在std::vector<std::string>中最终出现:

./foo --bar 13 command1                      # Should run fine
./foo command2 positional                    # Should run fine
./foo --bar 13 command3 --unknown argument   # Should run fine
./foo --unknown command4                     # should fail
./foo --bar 13 command5 --bar 42             # Should work but set bar to 13
./foo command6 --bar 42                      # Should not set bar at all
./foo --bar 13 -- command7 --bar 42 --unknown argument     # Should work

正确用法是首先指定任何定义的关键字基本上,以正确的形式,我的定义数量首先应采用任何数量的预定义

我尝试了两种方法:

1)使用allow_unregistered

#include <iostream>
#include <boost/program_options.hpp>
namespace po = boost::program_options;
int main(int argc, const char** argv)
{
    int bar = 0;
    po::options_description desc("Allowed options");
    desc.add_options()
            ("bar", po::value(&bar), "bar");

    po::variables_map vm;
    po::parsed_options parsed =
        po::command_line_parser(argc, argv).options(desc).allow_unregistered().run();
    po::store(parsed, vm);
    po::notify(vm);
    auto command = po::collect_unrecognized(parsed.options, po::include_positional);
    std::cout << "bar: " << bar << ", command:";
    for (const auto& c : command) std::cout << " " << c;
    std::cout << std::endl;
}

这将失败命令4,5,6

2)一个无限制出现的位置选项

std::vector<std::string> command;
desc.add_options()
        ("bar", po::value(&bar), "bar")
        ("command", po::value(&command));
po::positional_options_description p;
p.add("command", -1);
po::variables_map vm;
po::parsed_options parsed =
    po::command_line_parser(argc, argv).options(desc).positional(p).run();
po::store(parsed, vm);
po::notify(vm);

这将失败命令3,5,6。

在boost trac上有一个功能请求,包括补丁:https://svn.boost.org/trac/boost/boost/ticket/6991。机票上没有太多动作,但是该补丁仍然适用于提升(截至1.61.0,最新版本)。

如果您的构建系统允许,则可以将补丁应用于Boost的本地副本;否则,您可以从cmdline.hppdetail/cmdline.hppcmdline.cpp中提取boost::program_options::detail::cmdline,将CC_5提取到您自己的名称空间,修补该组件,并使用修补的组件代替boost::program_options::cmdline

另一种选择是使用其strace1参数破解boost::program_options::detail::cmdline的行为:

po::detail::cmdline cmdline(argc, argv);
cmdline.set_options_description(desc);
cmdline.set_positional_options(p);
std::vector<po::detail::cmdline::style_parser> style_parsers{
    [&](auto& args) { return cmdline.parse_long_option(args); },
    [&](auto& args) { return cmdline.parse_short_option(args); }};
cmdline.extra_style_parser([&](std::vector<std::string>& args) {
    auto const current_size = args.size();
    std::vector<po::option> result;
    for (auto const& parser : style_parsers) {
        auto const next = parser(args);
        result.insert(result.end(), next.begin(), next.end());
        if (args.size() != current_size)
            return result;
    }
    if (args.size() && args[0] != "--") args.insert(args.begin(), "--");
    auto const next = cmdline.parse_terminator(args);
    result.insert(result.end(), next.begin(), next.end());
    return result;
});
po::parsed_options parsed{&desc};
parsed.options = cmdline.run();
po::store(parsed, vm);
po::notify(vm);

示例警告:这是用无证件的图书馆内部设置黑客入侵,并且可以随时破裂。