/**
* argparse.h is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License v3 as published by
* the Free Software Foundation.
*
* argparse.h is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with argparse.h. If not, see .
*
* Author: Jesse Laning
*/
#ifndef ARGPARSE_H
#define ARGPARSE_H
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
class ArgumentParser {
private:
struct Argument;
public:
class ArgumentNotFound : public std::runtime_error {
public:
ArgumentNotFound(ArgumentParser::Argument &arg) noexcept
: std::runtime_error(
("Required argument not found: " + arg._name).c_str()) {}
};
ArgumentParser(const std::string &desc) : _desc(desc), _help(false) {}
ArgumentParser(const std::string desc, int argc, char *argv[])
: ArgumentParser(desc) {
parse(argc, argv);
}
void add_argument(const std::string &name, const std::string &long_name,
const std::string &desc, const bool required = false) {
_arguments.push_back({name, desc, required});
_pairs[name] = long_name;
}
void add_argument(const std::string &name, const std::string &desc,
const bool required = false) {
_arguments.push_back({name, desc, required});
}
bool is_help() { return _help; }
void print_help() {
std::cout << "Usage: " << _bin << " [options]" << std::endl;
std::cout << "Options:" << std::endl;
for (auto &a : _arguments) {
std::string name = a._name;
auto i = _pairs.find(name);
if (i != _pairs.end()) name.append(", " + i->second);
std::cout << " " << std::setw(23) << std::left << name << std::setw(23)
<< a._desc + (a._required ? " (Required)" : "") << std::endl;
}
}
void parse(int argc, char *argv[]) {
_bin = argv[0];
if (argc > 1) {
std::string name;
std::vector arg_parts;
std::vector free_args;
auto push_arg = [&name, &arg_parts, this]() {
if (!name.empty()) {
if (name[0] == '-') {
_add_variable(name, arg_parts);
} else {
for (char c : name) {
_add_variable(std::string(1, c), arg_parts);
}
}
arg_parts.clear();
}
};
for (int i = 1; i < argc; i++) {
size_t slen = std::strlen(argv[i]);
if (slen == 0) {
continue;
} else if (slen >= 2 && argv[i][0] == '-' && !_is_number(argv[i])) {
push_arg();
if (i == argc - 1) {
name = &(argv[i][1]);
push_arg();
} else {
name = &(argv[i][1]);
}
} else if (name.empty()) {
free_args.push_back(argv[i]);
} else {
arg_parts.push_back(argv[i]);
if (i == argc - 1) {
push_arg();
}
}
}
_add_variable("", free_args);
}
if (!_help) {
for (auto &a : _arguments) {
if (a._required) {
if (_variables.find(a._name) == _variables.end()) {
throw ArgumentNotFound(a);
}
}
}
}
}
bool exists(const std::string &name) {
std::string t = _delimit(name);
if (_pairs.find(t) != _pairs.end()) t = _pairs[t];
return _variables.find(t) != _variables.end();
}
template
std::vector getv(const std::string &name) {
std::vector argstr = getv(name);
std::vector v;
for (auto &s : argstr) {
std::istringstream in(s);
T t;
in >> t;
v.push_back(t);
}
return v;
}
template
T get(const std::string &name) {
std::istringstream in(get(name));
T t;
in >> t >> std::ws;
return t;
}
private:
friend class ArgumentNotFound;
struct Argument {
public:
Argument(const std::string &name, const std::string &desc,
bool required = false)
: _name(name), _desc(desc), _required(required) {}
std::string _name;
std::string _desc;
bool _required;
};
inline void _add_variable(std::string name,
std::vector &arg_parts) {
if (name == "h" || name == "-help") {
_help = true;
print_help();
}
_ltrim(name, [](int c) { return c != (int)'-'; });
name = _delimit(name);
if (_pairs.find(name) != _pairs.end()) name = _pairs[name];
_variables[name] = arg_parts;
}
static std::string _delimit(const std::string &name) {
return std::string(std::min(name.size(), (size_t)2), '-').append(name);
}
static std::string _strip(const std::string &name) {
size_t begin = 0;
begin += name.size() > 0 ? name[0] == '-' : 0;
begin += name.size() > 3 ? name[1] == '-' : 0;
return name.substr(begin);
}
static std::string _upper(const std::string &in) {
std::string out(in);
std::transform(out.begin(), out.end(), out.begin(), ::toupper);
return out;
}
static std::string _escape(const std::string &in) {
std::string out(in);
if (in.find(' ') != std::string::npos)
out = std::string("\"").append(out).append("\"");
return out;
}
static bool _not_space(int ch) { return !std::isspace(ch); }
static inline void _ltrim(std::string &s, bool (*f)(int) = _not_space) {
s.erase(s.begin(), std::find_if(s.begin(), s.end(), f));
}
static inline void _rtrim(std::string &s, bool (*f)(int) = _not_space) {
s.erase(std::find_if(s.rbegin(), s.rend(), f).base(), s.end());
}
static inline void _trim(std::string &s, bool (*f)(int) = _not_space) {
_ltrim(s, f);
_rtrim(s, f);
}
static inline std::string _ltrim_copy(std::string s,
bool (*f)(int) = _not_space) {
_ltrim(s, f);
return s;
}
static inline std::string _rtrim_copy(std::string s,
bool (*f)(int) = _not_space) {
_rtrim(s, f);
return s;
}
static inline std::string _trim_copy(std::string s,
bool (*f)(int) = _not_space) {
_trim(s, f);
return s;
}
template
inline std::string _join(InputIt begin, InputIt end,
const std::string &separator = " ") {
std::ostringstream ss;
if (begin != end) {
ss << *begin++;
}
while (begin != end) {
ss << separator;
ss << *begin++;
}
return ss.str();
}
static inline bool _is_number(const char *arg) {
std::istringstream iss{std::string(arg)};
float f;
iss >> std::noskipws >> f;
return iss.eof() && !iss.fail();
}
std::string _desc;
std::string _bin;
bool _help;
std::vector _arguments;
std::unordered_map> _variables;
std::unordered_map _pairs;
};
template <>
inline std::string ArgumentParser::get(const std::string &name) {
std::string t = _delimit(name);
if (_pairs.find(t) != _pairs.end()) t = _pairs[t];
auto v = _variables.find(t);
if (v != _variables.end()) {
return _join(v->second.begin(), v->second.end());
}
return "";
}
template <>
inline bool ArgumentParser::get(const std::string &name) {
return exists(name);
}
template <>
inline std::vector ArgumentParser::getv(
const std::string &name) {
std::string t = _delimit(name);
if (_pairs.find(t) != _pairs.end()) t = _pairs[t];
auto v = _variables.find(t);
if (v != _variables.end()) {
return v->second;
}
return std::vector();
}
#endif