2013-02-20 00:07:52 +01:00
|
|
|
/* Copyright 2013 MultiMC Contributors
|
|
|
|
*
|
|
|
|
* Authors: Orochimarufan <orochimarufan.x3@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef CMDUTILS_H
|
|
|
|
#define CMDUTILS_H
|
|
|
|
|
|
|
|
#include <exception>
|
2013-10-28 22:01:37 +01:00
|
|
|
#include <stdexcept>
|
2013-02-20 00:07:52 +01:00
|
|
|
|
|
|
|
#include <QString>
|
|
|
|
#include <QVariant>
|
|
|
|
#include <QHash>
|
|
|
|
#include <QStringList>
|
|
|
|
|
2013-02-25 14:44:36 -06:00
|
|
|
#include "libutil_config.h"
|
|
|
|
|
2013-02-20 00:07:52 +01:00
|
|
|
/**
|
2013-02-21 20:40:32 +01:00
|
|
|
* @file libutil/include/cmdutils.h
|
2013-07-29 00:59:35 +02:00
|
|
|
* @brief commandline parsing and processing utilities
|
2013-02-20 00:07:52 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Util {
|
|
|
|
namespace Commandline {
|
|
|
|
|
2013-07-29 00:59:35 +02:00
|
|
|
/**
|
|
|
|
* @brief split a string into argv items like a shell would do
|
|
|
|
* @param args the argument string
|
|
|
|
* @return a QStringList containing all arguments
|
|
|
|
*/
|
|
|
|
LIBUTIL_EXPORT QStringList splitArgs(QString args);
|
|
|
|
|
2013-02-20 00:07:52 +01:00
|
|
|
/**
|
|
|
|
* @brief The FlagStyle enum
|
|
|
|
* Specifies how flags are decorated
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
|
|
|
|
namespace FlagStyle
|
|
|
|
{
|
2013-08-26 02:53:29 +02:00
|
|
|
enum Enum
|
2013-02-25 14:44:36 -06:00
|
|
|
{
|
2013-02-25 13:39:07 -06:00
|
|
|
GNU, /**< --option and -o (GNU Style) */
|
|
|
|
Unix, /**< -option and -o (Unix Style) */
|
|
|
|
Windows, /**< /option and /o (Windows Style) */
|
2013-02-20 00:07:52 +01:00
|
|
|
#ifdef Q_OS_WIN32
|
2013-02-25 13:39:07 -06:00
|
|
|
Default = Windows
|
2013-02-20 00:07:52 +01:00
|
|
|
#else
|
2013-02-25 13:39:07 -06:00
|
|
|
Default = GNU
|
2013-02-20 00:07:52 +01:00
|
|
|
#endif
|
|
|
|
};
|
2013-02-25 14:44:36 -06:00
|
|
|
}
|
2013-02-20 00:07:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The ArgumentStyle enum
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
namespace ArgumentStyle
|
|
|
|
{
|
2013-08-26 02:53:29 +02:00
|
|
|
enum Enum
|
2013-02-25 14:44:36 -06:00
|
|
|
{
|
2013-02-25 13:39:07 -06:00
|
|
|
Space, /**< --option=value */
|
|
|
|
Equals, /**< --option value */
|
|
|
|
SpaceAndEquals, /**< --option[= ]value */
|
2013-02-20 00:07:52 +01:00
|
|
|
#ifdef Q_OS_WIN32
|
2013-02-25 13:39:07 -06:00
|
|
|
Default = Equals
|
2013-02-20 00:07:52 +01:00
|
|
|
#else
|
2013-02-25 13:39:07 -06:00
|
|
|
Default = SpaceAndEquals
|
2013-02-20 00:07:52 +01:00
|
|
|
#endif
|
|
|
|
};
|
2013-02-25 14:44:36 -06:00
|
|
|
}
|
|
|
|
|
2013-02-20 00:07:52 +01:00
|
|
|
/**
|
|
|
|
* @brief The ParsingError class
|
|
|
|
*/
|
2013-10-28 21:50:58 +01:00
|
|
|
class LIBUTIL_EXPORT ParsingError : public std::runtime_error
|
2013-02-20 00:07:52 +01:00
|
|
|
{
|
|
|
|
public:
|
2013-02-25 13:39:07 -06:00
|
|
|
ParsingError(const QString &what);
|
2013-02-20 00:07:52 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The Parser class
|
|
|
|
*/
|
2013-02-26 16:47:39 -06:00
|
|
|
class LIBUTIL_EXPORT Parser
|
2013-02-20 00:07:52 +01:00
|
|
|
{
|
|
|
|
public:
|
2013-02-25 13:39:07 -06:00
|
|
|
/**
|
|
|
|
* @brief Parser constructor
|
|
|
|
* @param flagStyle the FlagStyle to use in this Parser
|
|
|
|
* @param argStyle the ArgumentStyle to use in this Parser
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
Parser(FlagStyle::Enum flagStyle = FlagStyle::Default,
|
|
|
|
ArgumentStyle::Enum argStyle = ArgumentStyle::Default);
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief set the flag style
|
|
|
|
* @param style
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
void setFlagStyle(FlagStyle::Enum style);
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief get the flag style
|
|
|
|
* @return
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
FlagStyle::Enum flagStyle();
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief set the argument style
|
|
|
|
* @param style
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
void setArgumentStyle(ArgumentStyle::Enum style);
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief get the argument style
|
|
|
|
* @return
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
ArgumentStyle::Enum argumentStyle();
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief define a boolean switch
|
|
|
|
* @param name the parameter name
|
|
|
|
* @param def the default value
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
void addSwitch(QString name, bool def = false);
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief define an option that takes an additional argument
|
|
|
|
* @param name the parameter name
|
|
|
|
* @param def the default value
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
void addOption(QString name, QVariant def = QVariant());
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief define a positional argument
|
|
|
|
* @param name the parameter name
|
|
|
|
* @param required wether this argument is required
|
|
|
|
* @param def the default value
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
void addArgument(QString name, bool required = true, QVariant def = QVariant());
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief adds a flag to an existing parameter
|
|
|
|
* @param name the (existing) parameter name
|
|
|
|
* @param flag the flag character
|
|
|
|
* @see addSwitch addArgument addOption
|
|
|
|
* Note: any one parameter can only have one flag
|
|
|
|
*/
|
|
|
|
void addShortOpt(QString name, QChar flag);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief adds documentation to a Parameter
|
|
|
|
* @param name the parameter name
|
|
|
|
* @param metavar a string to be displayed as placeholder for the value
|
|
|
|
* @param doc a QString containing the documentation
|
|
|
|
* Note: on positional arguments, metavar replaces the name as displayed.
|
|
|
|
* on options , metavar replaces the value placeholder
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
void addDocumentation(QString name, QString doc, QString metavar = QString());
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief generate a help message
|
|
|
|
* @param progName the program name to use in the help message
|
|
|
|
* @param helpIndent how much the parameter documentation should be indented
|
|
|
|
* @param flagsInUsage whether we should use flags instead of options in the usage
|
|
|
|
* @return a help message
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
QString compileHelp(QString progName, int helpIndent = 22, bool flagsInUsage = true);
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief generate a short usage message
|
|
|
|
* @param progName the program name to use in the usage message
|
|
|
|
* @param useFlags whether we should use flags instead of options
|
|
|
|
* @return a usage message
|
|
|
|
*/
|
2013-02-25 14:44:36 -06:00
|
|
|
QString compileUsage(QString progName, bool useFlags = true);
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief parse
|
|
|
|
* @param argv a QStringList containing the program ARGV
|
|
|
|
* @return a QHash mapping argument names to their values
|
|
|
|
*/
|
|
|
|
QHash<QString, QVariant> parse(QStringList argv);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief clear all definitions
|
|
|
|
*/
|
|
|
|
void clear();
|
|
|
|
|
|
|
|
~Parser();
|
|
|
|
|
2013-02-20 00:07:52 +01:00
|
|
|
private:
|
2013-02-25 14:44:36 -06:00
|
|
|
FlagStyle::Enum m_flagStyle;
|
|
|
|
ArgumentStyle::Enum m_argStyle;
|
2013-02-25 22:47:03 +01:00
|
|
|
|
|
|
|
enum OptionType
|
|
|
|
{
|
|
|
|
otSwitch,
|
|
|
|
otOption
|
|
|
|
};
|
2013-02-25 13:39:07 -06:00
|
|
|
|
|
|
|
// Important: the common part MUST BE COMMON ON ALL THREE structs
|
|
|
|
struct CommonDef {
|
|
|
|
QString name;
|
|
|
|
QString doc;
|
|
|
|
QString metavar;
|
|
|
|
QVariant def;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct OptionDef {
|
|
|
|
// common
|
|
|
|
QString name;
|
|
|
|
QString doc;
|
|
|
|
QString metavar;
|
|
|
|
QVariant def;
|
|
|
|
// option
|
2013-02-25 22:47:03 +01:00
|
|
|
OptionType type;
|
2013-02-25 13:39:07 -06:00
|
|
|
QChar flag;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct PositionalDef {
|
|
|
|
// common
|
|
|
|
QString name;
|
|
|
|
QString doc;
|
|
|
|
QString metavar;
|
|
|
|
QVariant def;
|
|
|
|
// positional
|
|
|
|
bool required;
|
|
|
|
};
|
|
|
|
|
|
|
|
QHash<QString, OptionDef *> m_options;
|
|
|
|
QHash<QChar, OptionDef *> m_flags;
|
|
|
|
QHash<QString, CommonDef *> m_params;
|
|
|
|
QList<PositionalDef *> m_positionals;
|
|
|
|
QList<OptionDef *> m_optionList;
|
|
|
|
|
|
|
|
void getPrefix(QString &opt, QString &flag);
|
2013-02-20 00:07:52 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // CMDUTILS_H
|