2024-07-28 22:42:16 +08:00
|
|
|
#pragma once
|
|
|
|
#include "YYCCInternal.hpp"
|
|
|
|
|
|
|
|
#include "Constraints.hpp"
|
|
|
|
#include "EncodingHelper.hpp"
|
2024-07-29 16:58:52 +08:00
|
|
|
#include "ParserHelper.hpp"
|
2024-07-28 22:42:16 +08:00
|
|
|
#include <functional>
|
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
namespace YYCC::ArgParser {
|
|
|
|
|
|
|
|
class ArgumentList {
|
|
|
|
public:
|
|
|
|
static ArgumentList CreateFromStd(int argc, char* argv[]);
|
|
|
|
#if YYCC_OS == YYCC_OS_WINDOWS
|
|
|
|
static ArgumentList CreateFromWin32();
|
|
|
|
#endif
|
|
|
|
private:
|
|
|
|
ArgumentList(std::vector<yycc_u8string>&& arguments);
|
2024-07-29 16:58:52 +08:00
|
|
|
public:
|
|
|
|
ArgumentList(const ArgumentList&) = default;
|
|
|
|
ArgumentList& operator=(const ArgumentList&) = default;
|
|
|
|
ArgumentList(ArgumentList&&) = default;
|
|
|
|
ArgumentList& operator=(ArgumentList&&) = default;
|
2024-07-28 22:42:16 +08:00
|
|
|
|
|
|
|
public:
|
|
|
|
void Prev();
|
|
|
|
void Next();
|
2024-07-29 19:31:17 +08:00
|
|
|
const yycc_u8string& Argument() const;
|
2024-07-28 22:42:16 +08:00
|
|
|
bool IsSwitch(
|
|
|
|
bool* is_long_name = nullptr,
|
2024-07-29 16:58:52 +08:00
|
|
|
yycc_u8string* long_name = nullptr,
|
2024-07-28 22:42:16 +08:00
|
|
|
yycc_char8_t* short_name = nullptr) const;
|
2024-07-29 19:31:17 +08:00
|
|
|
bool IsParameter(yycc_u8string* val = nullptr) const;
|
2024-07-28 22:42:16 +08:00
|
|
|
bool IsEOF() const;
|
|
|
|
void Reset();
|
2024-07-29 16:58:52 +08:00
|
|
|
private:
|
|
|
|
bool IsLongNameSwitch(yycc_u8string* name_part = nullptr) const;
|
|
|
|
bool IsShortNameSwitch(yycc_char8_t* name_part = nullptr) const;
|
2024-07-28 22:42:16 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<yycc_u8string> m_Arguments;
|
|
|
|
std::vector<yycc_u8string>::const_iterator m_ArgumentsIterator;
|
|
|
|
};
|
|
|
|
|
|
|
|
class AbstractArgument {
|
|
|
|
friend class OptionContext;
|
2024-07-29 16:58:52 +08:00
|
|
|
|
|
|
|
// Long name and short name constants and checker.
|
2024-07-28 22:42:16 +08:00
|
|
|
public:
|
|
|
|
static const yycc_u8string DOUBLE_DASH;
|
|
|
|
static const yycc_char8_t DASH;
|
|
|
|
static const yycc_char8_t NO_SHORT_NAME;
|
|
|
|
static const yycc_char8_t MIN_SHORT_NAME;
|
|
|
|
static const yycc_char8_t MAX_SHORT_NAME;
|
2024-07-29 16:58:52 +08:00
|
|
|
static bool IsLegalShortName(yycc_char8_t short_name);
|
|
|
|
static bool IsLegalLongName(const yycc_u8string_view& long_name);
|
|
|
|
|
|
|
|
// Constructor & destructor
|
2024-07-28 22:42:16 +08:00
|
|
|
public:
|
|
|
|
AbstractArgument(
|
|
|
|
const yycc_char8_t* long_name, yycc_char8_t short_name = AbstractArgument::NO_SHORT_NAME,
|
|
|
|
const yycc_char8_t* description = nullptr, const yycc_char8_t* argument_example = nullptr,
|
|
|
|
bool is_optional = false);
|
|
|
|
virtual ~AbstractArgument();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual bool Parse(ArgumentList& al) = 0;
|
|
|
|
virtual void Reset() = 0;
|
|
|
|
|
|
|
|
public:
|
|
|
|
bool HasLongName() const;
|
|
|
|
const yycc_u8string& GetLongName() const;
|
|
|
|
bool HasShortName() const;
|
|
|
|
yycc_char8_t GetShortName() const;
|
|
|
|
bool HasDescription() const;
|
|
|
|
const yycc_u8string& GetDescription() const;
|
|
|
|
bool HasArgumentExample() const;
|
|
|
|
const yycc_u8string& GetArgumentExample() const;
|
|
|
|
bool IsOptional() const;
|
|
|
|
private:
|
|
|
|
yycc_u8string m_LongName;
|
|
|
|
yycc_char8_t m_ShortName;
|
|
|
|
yycc_u8string m_Description;
|
|
|
|
yycc_u8string m_ArgumentExample;
|
|
|
|
bool m_IsOptional;
|
|
|
|
|
|
|
|
public:
|
|
|
|
bool IsCaptured() const;
|
|
|
|
private:
|
|
|
|
void SetCaptured(bool is_captured);
|
|
|
|
bool m_IsCaptured;
|
|
|
|
};
|
|
|
|
|
|
|
|
class OptionContext {
|
|
|
|
public:
|
|
|
|
OptionContext(
|
|
|
|
const yycc_char8_t* summary, const yycc_char8_t* description,
|
|
|
|
std::initializer_list<AbstractArgument*> arguments);
|
|
|
|
~OptionContext();
|
|
|
|
|
|
|
|
public:
|
2024-07-29 16:58:52 +08:00
|
|
|
bool Parse(ArgumentList& al);
|
2024-07-28 22:42:16 +08:00
|
|
|
void Reset();
|
2024-07-29 19:31:17 +08:00
|
|
|
void Help() const;
|
2024-07-28 22:42:16 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
yycc_u8string m_Summary;
|
|
|
|
yycc_u8string m_Description;
|
|
|
|
|
|
|
|
std::vector<AbstractArgument*> m_Arguments;
|
|
|
|
std::map<yycc_u8string, AbstractArgument*> m_LongNameMap;
|
|
|
|
std::map<yycc_char8_t, AbstractArgument*> m_ShortNameMap;
|
|
|
|
};
|
|
|
|
|
|
|
|
#pragma region Argument Presets
|
|
|
|
|
|
|
|
template<typename _Ty, std::enable_if_t<std::is_arithmetic_v<_Ty> && !std::is_same_v<_Ty, bool>, int> = 0>
|
|
|
|
class NumberArgument : public AbstractArgument {
|
|
|
|
public:
|
|
|
|
NumberArgument(
|
|
|
|
const yycc_char8_t* long_name, yycc_char8_t short_name,
|
|
|
|
const yycc_char8_t* description = nullptr, const yycc_char8_t* argument_example = nullptr,
|
|
|
|
bool is_optional = false,
|
|
|
|
Constraints::Constraint<_Ty> constraint = Constraints::Constraint<_Ty> {}) :
|
|
|
|
AbstractArgument(long_name, short_name, description, argument_example, is_optional), m_Data(), m_Constraint(constraint) {}
|
|
|
|
virtual ~NumberArgument() {}
|
|
|
|
|
|
|
|
public:
|
|
|
|
_Ty Get() const {
|
|
|
|
if (!IsCaptured()) throw std::runtime_error("try fetching data from a not captured argument.");
|
|
|
|
return m_Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2024-07-29 16:58:52 +08:00
|
|
|
virtual bool Parse(ArgumentList& al) override {
|
|
|
|
// try get corresponding value
|
|
|
|
yycc_u8string strval;
|
|
|
|
al.Next();
|
2024-07-29 19:31:17 +08:00
|
|
|
if (al.IsEOF() || !al.IsParameter(&strval)) {
|
2024-07-29 16:58:52 +08:00
|
|
|
al.Prev();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// try parsing value
|
|
|
|
if (!YYCC::ParserHelper::TryParse<_Ty>(strval, m_Data)) return false;
|
|
|
|
// check constraint
|
|
|
|
if (m_Constraint.IsValid() && !m_Constraint.m_CheckFct(m_Data))
|
|
|
|
return false;
|
|
|
|
// okey
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
virtual void Reset() override {
|
|
|
|
std::memset(&m_Data, 0, sizeof(m_Data));
|
|
|
|
}
|
2024-07-28 22:42:16 +08:00
|
|
|
|
|
|
|
protected:
|
|
|
|
_Ty m_Data;
|
|
|
|
Constraints::Constraint<_Ty> m_Constraint;
|
|
|
|
};
|
|
|
|
|
|
|
|
class SwitchArgument : public AbstractArgument {
|
|
|
|
public:
|
|
|
|
SwitchArgument(
|
|
|
|
const yycc_char8_t* long_name, yycc_char8_t short_name,
|
2024-07-29 19:31:17 +08:00
|
|
|
const yycc_char8_t* description = nullptr) :
|
|
|
|
// bool switch must be optional, because it is false if no given switch.
|
|
|
|
// bool switch doesn't have argument, so it doesn't have example property.
|
|
|
|
AbstractArgument(long_name, short_name, description, nullptr, true), m_Data(false) {}
|
2024-07-28 22:42:16 +08:00
|
|
|
virtual ~SwitchArgument() {}
|
|
|
|
|
|
|
|
public:
|
|
|
|
bool Get() const { return m_Data; }
|
|
|
|
|
|
|
|
protected:
|
2024-07-29 16:58:52 +08:00
|
|
|
virtual bool Parse(ArgumentList& al) override {
|
|
|
|
m_Data = true;
|
|
|
|
return true;
|
|
|
|
}
|
2024-07-28 22:42:16 +08:00
|
|
|
virtual void Reset() override { m_Data = false; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
bool m_Data;
|
|
|
|
};
|
|
|
|
|
|
|
|
class StringArgument : public AbstractArgument {
|
|
|
|
public:
|
|
|
|
StringArgument(
|
|
|
|
const yycc_char8_t* long_name, yycc_char8_t short_name,
|
|
|
|
const yycc_char8_t* description = nullptr, const yycc_char8_t* argument_example = nullptr,
|
|
|
|
bool is_optional = false,
|
2024-07-29 16:58:52 +08:00
|
|
|
Constraints::Constraint<yycc_u8string> constraint = Constraints::Constraint<yycc_u8string> {}) :
|
2024-07-28 22:42:16 +08:00
|
|
|
AbstractArgument(long_name, short_name, description, argument_example, is_optional), m_Data(), m_Constraint(constraint) {}
|
|
|
|
virtual ~StringArgument() {}
|
|
|
|
|
|
|
|
public:
|
|
|
|
const yycc_u8string& Get() const {
|
|
|
|
if (!IsCaptured()) throw std::runtime_error("try fetching data from a not captured argument.");
|
|
|
|
return m_Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2024-07-29 16:58:52 +08:00
|
|
|
virtual bool Parse(ArgumentList& al) override {
|
|
|
|
// try get corresponding value
|
|
|
|
al.Next();
|
2024-07-29 19:31:17 +08:00
|
|
|
if (al.IsEOF() || !al.IsParameter(&m_Data)) {
|
2024-07-29 16:58:52 +08:00
|
|
|
al.Prev();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// check constraint
|
|
|
|
if (m_Constraint.IsValid() && !m_Constraint.m_CheckFct(m_Data))
|
|
|
|
return false;
|
|
|
|
// okey
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
virtual void Reset() override {
|
|
|
|
m_Data.clear();
|
|
|
|
}
|
2024-07-28 22:42:16 +08:00
|
|
|
|
|
|
|
protected:
|
|
|
|
yycc_u8string m_Data;
|
2024-07-29 16:58:52 +08:00
|
|
|
Constraints::Constraint<yycc_u8string> m_Constraint;
|
2024-07-28 22:42:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
#pragma endregion
|
|
|
|
|
|
|
|
|
|
|
|
}
|