/* Copyright 2013-2021 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.
 */

#pragma once

#include <exception>
#include <stdexcept>

#include <QString>
#include <QVariant>
#include <QHash>
#include <QStringList>

/**
 * @file libutil/include/cmdutils.h
 * @brief commandline parsing and processing utilities
 */

namespace Commandline
{

/**
 * @brief split a string into argv items like a shell would do
 * @param args the argument string
 * @return a QStringList containing all arguments
 */
QStringList splitArgs(QString args);

/**
 * @brief The FlagStyle enum
 * Specifies how flags are decorated
 */

namespace FlagStyle
{
enum Enum
{
    GNU,     /**< --option and -o (GNU Style) */
    Unix,    /**< -option and -o  (Unix Style) */
    Windows, /**< /option and /o  (Windows Style) */
#ifdef Q_OS_WIN32
    Default = Windows
#else
        Default = GNU
#endif
};
}

/**
 * @brief The ArgumentStyle enum
 */
namespace ArgumentStyle
{
enum Enum
{
    Space,          /**< --option value */
    Equals,         /**< --option=value */
    SpaceAndEquals, /**< --option[= ]value */
#ifdef Q_OS_WIN32
    Default = Equals
#else
        Default = SpaceAndEquals
#endif
};
}

/**
 * @brief The ParsingError class
 */
class ParsingError : public std::runtime_error
{
public:
    ParsingError(const QString &what);
};

/**
 * @brief The Parser class
 */
class Parser
{
public:
    /**
     * @brief Parser constructor
     * @param flagStyle the FlagStyle to use in this Parser
     * @param argStyle the ArgumentStyle to use in this Parser
     */
    Parser(FlagStyle::Enum flagStyle = FlagStyle::Default,
           ArgumentStyle::Enum argStyle = ArgumentStyle::Default);

    /**
     * @brief set the flag style
     * @param style
     */
    void setFlagStyle(FlagStyle::Enum style);

    /**
     * @brief get the flag style
     * @return
     */
    FlagStyle::Enum flagStyle();

    /**
     * @brief set the argument style
     * @param style
     */
    void setArgumentStyle(ArgumentStyle::Enum style);

    /**
     * @brief get the argument style
     * @return
     */
    ArgumentStyle::Enum argumentStyle();

    /**
     * @brief define a boolean switch
     * @param name the parameter name
     * @param def the default value
     */
    void addSwitch(QString name, bool def = false);

    /**
     * @brief define an option that takes an additional argument
     * @param name the parameter name
     * @param def the default value
     */
    void addOption(QString name, QVariant def = QVariant());

    /**
     * @brief define a positional argument
     * @param name the parameter name
     * @param required wether this argument is required
     * @param def the default value
     */
    void addArgument(QString name, bool required = true, QVariant def = QVariant());

    /**
     * @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
     */
    void addDocumentation(QString name, QString doc, QString metavar = QString());

    /**
     * @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
     */
    QString compileHelp(QString progName, int helpIndent = 22, bool flagsInUsage = true);

    /**
     * @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
     */
    QString compileUsage(QString progName, bool useFlags = true);

    /**
     * @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();

private:
    FlagStyle::Enum m_flagStyle;
    ArgumentStyle::Enum m_argStyle;

    enum OptionType
    {
        otSwitch,
        otOption
    };

    // 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
        OptionType type;
        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);
};
}