mirror of
https://github.com/quantum5/Zalgo.git
synced 2025-04-24 13:41:57 -04:00
308 lines
11 KiB
C
308 lines
11 KiB
C
|
/**
|
||
|
* Copyright (C) 2010 Johannes Weißl <jargon@molb.org>
|
||
|
* License: your favourite BSD-style license
|
||
|
*
|
||
|
* git clone http://github.com/weisslj/cpp-optparse.git
|
||
|
*
|
||
|
* This is yet another option parser for C++. It is modelled after the
|
||
|
* excellent Python optparse API. Although incomplete, anyone familiar to
|
||
|
* optparse should feel at home:
|
||
|
* http://docs.python.org/library/optparse.html
|
||
|
*
|
||
|
* Design decisions:
|
||
|
* - elegant and easy usage more important than speed / flexibility
|
||
|
* - shortness more important than feature completeness
|
||
|
* * no unicode
|
||
|
* * no checking for user programming errors
|
||
|
*
|
||
|
* Why not use getopt/getopt_long?
|
||
|
* - not C++ / not completely POSIX
|
||
|
* - too cumbersome to use, would need lot of additional code
|
||
|
*
|
||
|
* Why not use Boost.Program_options?
|
||
|
* - boost not installed on all target platforms (esp. cluster, HPC, ...)
|
||
|
* - too big to include just for option handling:
|
||
|
* 322 *.h (44750 lines) + 7 *.cpp (2078 lines)
|
||
|
*
|
||
|
* Why not use tclap/Opag/Options/CmdLine/Anyoption/Argument_helper/...?
|
||
|
* - no reason, writing one is faster than code inspection :-)
|
||
|
* - similarity to Python desired for faster learning curve
|
||
|
*
|
||
|
* Future work:
|
||
|
* - nargs > 1?
|
||
|
* - comments?
|
||
|
*
|
||
|
* Python only features:
|
||
|
* - conflict handlers
|
||
|
* - adding new actions
|
||
|
*
|
||
|
*
|
||
|
* Example:
|
||
|
*
|
||
|
* using optparse::OptionParser;
|
||
|
*
|
||
|
* OptionParser parser = OptionParser() .description("just an example");
|
||
|
*
|
||
|
* parser.add_option("-f", "--file") .dest("filename")
|
||
|
* .help("write report to FILE") .metavar("FILE");
|
||
|
* parser.add_option("-q", "--quiet")
|
||
|
* .action("store_false") .dest("verbose") .set_default("1")
|
||
|
* .help("don't print status messages to stdout");
|
||
|
*
|
||
|
* optparse::Values options = parser.parse_args(argc, argv);
|
||
|
* vector<string> args = parser.args();
|
||
|
*
|
||
|
* if (options.get("verbose"))
|
||
|
* cout << options["filename"] << endl;
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#ifndef OPTIONPARSER_H_
|
||
|
#define OPTIONPARSER_H_
|
||
|
|
||
|
#include <string>
|
||
|
#include <vector>
|
||
|
#include <list>
|
||
|
#include <map>
|
||
|
#include <set>
|
||
|
#include <iostream>
|
||
|
#include <sstream>
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
# pragma warning(push)
|
||
|
# pragma warning(disable: 4512)
|
||
|
#endif
|
||
|
namespace optparse {
|
||
|
|
||
|
class OptionParser;
|
||
|
class OptionGroup;
|
||
|
class Option;
|
||
|
class Values;
|
||
|
class Value;
|
||
|
class Callback;
|
||
|
|
||
|
typedef std::map<std::string,std::string> strMap;
|
||
|
typedef std::map<std::string,std::list<std::string> > lstMap;
|
||
|
typedef std::map<std::string,Option const*> optMap;
|
||
|
|
||
|
const char* const SUPPRESS_HELP = "SUPPRESS" "HELP";
|
||
|
const char* const SUPPRESS_USAGE = "SUPPRESS" "USAGE";
|
||
|
|
||
|
//! Class for automatic conversion from string -> anytype
|
||
|
class Value {
|
||
|
public:
|
||
|
Value() : str(), valid(false) {}
|
||
|
Value(const std::string& v) : str(v), valid(true) {}
|
||
|
operator const char*() { return str.c_str(); }
|
||
|
operator bool() { bool t; return (valid && (std::istringstream(str) >> t)) ? t : false; }
|
||
|
operator short() { short t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator unsigned short() { unsigned short t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator int() { int t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator unsigned int() { unsigned int t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator long() { long t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator unsigned long() { unsigned long t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator float() { float t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator double() { double t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
operator long double() { long double t; return (valid && (std::istringstream(str) >> t)) ? t : 0; }
|
||
|
private:
|
||
|
const std::string str;
|
||
|
bool valid;
|
||
|
};
|
||
|
|
||
|
class Values {
|
||
|
public:
|
||
|
Values() : _map() {}
|
||
|
const std::string& operator[] (const std::string& d) const;
|
||
|
std::string& operator[] (const std::string& d) { return _map[d]; }
|
||
|
bool is_set(const std::string& d) const { return _map.find(d) != _map.end(); }
|
||
|
bool is_set_by_user(const std::string& d) const { return _userSet.find(d) != _userSet.end(); }
|
||
|
void is_set_by_user(const std::string& d, bool yes);
|
||
|
Value get(const std::string& d) const { return (is_set(d)) ? Value((*this)[d]) : Value(); }
|
||
|
|
||
|
typedef std::list<std::string>::iterator iterator;
|
||
|
typedef std::list<std::string>::const_iterator const_iterator;
|
||
|
std::list<std::string>& all(const std::string& d) { return _appendMap[d]; }
|
||
|
const std::list<std::string>& all(const std::string& d) const { return _appendMap.find(d)->second; }
|
||
|
|
||
|
private:
|
||
|
strMap _map;
|
||
|
lstMap _appendMap;
|
||
|
std::set<std::string> _userSet;
|
||
|
};
|
||
|
|
||
|
class OptionParser {
|
||
|
public:
|
||
|
OptionParser();
|
||
|
virtual ~OptionParser() {}
|
||
|
|
||
|
OptionParser& usage(const std::string& u) { set_usage(u); return *this; }
|
||
|
OptionParser& version(const std::string& v) { _version = v; return *this; }
|
||
|
OptionParser& description(const std::string& d) { _description = d; return *this; }
|
||
|
OptionParser& add_help_option(bool h) { _add_help_option = h; return *this; }
|
||
|
OptionParser& add_version_option(bool v) { _add_version_option = v; return *this; }
|
||
|
OptionParser& prog(const std::string& p) { _prog = p; return *this; }
|
||
|
OptionParser& epilog(const std::string& e) { _epilog = e; return *this; }
|
||
|
OptionParser& set_defaults(const std::string& dest, const std::string& val) {
|
||
|
_defaults[dest] = val; return *this;
|
||
|
}
|
||
|
OptionParser& enable_interspersed_args() { _interspersed_args = true; return *this; }
|
||
|
OptionParser& disable_interspersed_args() { _interspersed_args = false; return *this; }
|
||
|
OptionParser& add_option_group(const OptionGroup& group);
|
||
|
|
||
|
const std::string& usage() const { return _usage; }
|
||
|
const std::string& version() const { return _version; }
|
||
|
const std::string& description() const { return _description; }
|
||
|
bool add_help_option() const { return _add_help_option; }
|
||
|
bool add_version_option() const { return _add_version_option; }
|
||
|
const std::string& prog() const { return _prog; }
|
||
|
const std::string& epilog() const { return _epilog; }
|
||
|
bool interspersed_args() const { return _interspersed_args; }
|
||
|
|
||
|
Option& add_option(const std::string& opt);
|
||
|
Option& add_option(const std::string& opt1, const std::string& opt2);
|
||
|
Option& add_option(const std::string& opt1, const std::string& opt2, const std::string& opt3);
|
||
|
Option& add_option(const std::vector<std::string>& opt);
|
||
|
|
||
|
Values& parse_args(int argc, char const* const* argv);
|
||
|
Values& parse_args(const std::vector<std::string>& args);
|
||
|
template<typename InputIterator>
|
||
|
Values& parse_args(InputIterator begin, InputIterator end) {
|
||
|
return parse_args(std::vector<std::string>(begin, end));
|
||
|
}
|
||
|
|
||
|
const std::list<std::string>& args() const { return _leftover; }
|
||
|
std::vector<std::string> args() {
|
||
|
return std::vector<std::string>(_leftover.begin(), _leftover.end());
|
||
|
}
|
||
|
|
||
|
std::string format_help() const;
|
||
|
std::string format_option_help(unsigned int indent = 2) const;
|
||
|
void print_help() const;
|
||
|
|
||
|
void set_usage(const std::string& u);
|
||
|
std::string get_usage() const;
|
||
|
void print_usage(std::ostream& out) const;
|
||
|
void print_usage() const;
|
||
|
|
||
|
std::string get_version() const;
|
||
|
void print_version(std::ostream& out) const;
|
||
|
void print_version() const;
|
||
|
|
||
|
void error(const std::string& msg) const;
|
||
|
void exit() const;
|
||
|
|
||
|
private:
|
||
|
const Option& lookup_short_opt(const std::string& opt) const;
|
||
|
const Option& lookup_long_opt(const std::string& opt) const;
|
||
|
|
||
|
void handle_short_opt(const std::string& opt, const std::string& arg);
|
||
|
void handle_long_opt(const std::string& optstr);
|
||
|
|
||
|
void process_opt(const Option& option, const std::string& opt, const std::string& value);
|
||
|
|
||
|
std::string format_usage(const std::string& u) const;
|
||
|
|
||
|
std::string _usage;
|
||
|
std::string _version;
|
||
|
std::string _description;
|
||
|
bool _add_help_option;
|
||
|
bool _add_version_option;
|
||
|
std::string _prog;
|
||
|
std::string _epilog;
|
||
|
bool _interspersed_args;
|
||
|
|
||
|
Values _values;
|
||
|
|
||
|
std::list<Option> _opts;
|
||
|
optMap _optmap_s;
|
||
|
optMap _optmap_l;
|
||
|
strMap _defaults;
|
||
|
std::list<OptionGroup const*> _groups;
|
||
|
|
||
|
std::list<std::string> _remaining;
|
||
|
std::list<std::string> _leftover;
|
||
|
};
|
||
|
|
||
|
class OptionGroup : public OptionParser {
|
||
|
public:
|
||
|
OptionGroup(const OptionParser& p, const std::string& t, const std::string& d = "") :
|
||
|
_parser(p), _title(t), _group_description(d) {}
|
||
|
virtual ~OptionGroup() {}
|
||
|
|
||
|
OptionGroup& title(const std::string& t) { _title = t; return *this; }
|
||
|
OptionGroup& group_description(const std::string& d) { _group_description = d; return *this; }
|
||
|
const std::string& title() const { return _title; }
|
||
|
const std::string& group_description() const { return _group_description; }
|
||
|
|
||
|
private:
|
||
|
const OptionParser& _parser;
|
||
|
std::string _title;
|
||
|
std::string _group_description;
|
||
|
};
|
||
|
|
||
|
class Option {
|
||
|
public:
|
||
|
Option() : _action("store"), _type("string"), _nargs(1), _callback(0) {}
|
||
|
virtual ~Option() {}
|
||
|
|
||
|
Option& action(const std::string& a);
|
||
|
Option& type(const std::string& t) { _type = t; return *this; }
|
||
|
Option& dest(const std::string& d) { _dest = d; return *this; }
|
||
|
Option& set_default(const std::string& d) { _default = d; return *this; }
|
||
|
template<typename T>
|
||
|
Option& set_default(T t) { std::ostringstream ss; ss << t; _default = ss.str(); return *this; }
|
||
|
Option& nargs(size_t n) { _nargs = n; return *this; }
|
||
|
Option& set_const(const std::string& c) { _const = c; return *this; }
|
||
|
template<typename InputIterator>
|
||
|
Option& choices(InputIterator begin, InputIterator end) {
|
||
|
_choices.assign(begin, end); type("choice"); return *this;
|
||
|
}
|
||
|
Option& help(const std::string& h) { _help = h; return *this; }
|
||
|
Option& metavar(const std::string& m) { _metavar = m; return *this; }
|
||
|
Option& callback(Callback& c) { _callback = &c; return *this; }
|
||
|
|
||
|
const std::string& action() const { return _action; }
|
||
|
const std::string& type() const { return _type; }
|
||
|
const std::string& dest() const { return _dest; }
|
||
|
const std::string& get_default() const { return _default; }
|
||
|
size_t nargs() const { return _nargs; }
|
||
|
const std::string& get_const() const { return _const; }
|
||
|
const std::list<std::string>& choices() const { return _choices; }
|
||
|
const std::string& help() const { return _help; }
|
||
|
const std::string& metavar() const { return _metavar; }
|
||
|
Callback* callback() const { return _callback; }
|
||
|
|
||
|
private:
|
||
|
std::string check_type(const std::string& opt, const std::string& val) const;
|
||
|
std::string format_option_help(unsigned int indent = 2) const;
|
||
|
std::string format_help(unsigned int indent = 2) const;
|
||
|
|
||
|
std::set<std::string> _short_opts;
|
||
|
std::set<std::string> _long_opts;
|
||
|
|
||
|
std::string _action;
|
||
|
std::string _type;
|
||
|
std::string _dest;
|
||
|
std::string _default;
|
||
|
size_t _nargs;
|
||
|
std::string _const;
|
||
|
std::list<std::string> _choices;
|
||
|
std::string _help;
|
||
|
std::string _metavar;
|
||
|
Callback* _callback;
|
||
|
|
||
|
friend class OptionParser;
|
||
|
};
|
||
|
|
||
|
class Callback {
|
||
|
public:
|
||
|
virtual void operator() (const Option& option, const std::string& opt, const std::string& val, const OptionParser& parser) = 0;
|
||
|
virtual ~Callback() {}
|
||
|
};
|
||
|
|
||
|
}
|
||
|
#ifdef _MSC_VER
|
||
|
# pragma warning(pop)
|
||
|
#endif
|
||
|
#endif
|