2020-08-30 15:43:45 +02:00
|
|
|
|
|
|
|
// Copyright Catch2 Authors
|
|
|
|
// Distributed under the Boost Software License, Version 1.0.
|
2022-10-28 11:22:53 +02:00
|
|
|
// (See accompanying file LICENSE.txt or copy at
|
2020-08-30 15:43:45 +02:00
|
|
|
// https://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
2022-04-14 17:16:01 +02:00
|
|
|
|
2020-08-08 18:18:27 +02:00
|
|
|
#include <catch2/internal/catch_clara.hpp>
|
|
|
|
#include <catch2/internal/catch_console_width.hpp>
|
|
|
|
#include <catch2/internal/catch_platform.hpp>
|
|
|
|
#include <catch2/internal/catch_string_manip.hpp>
|
|
|
|
#include <catch2/internal/catch_textflow.hpp>
|
2023-12-23 16:06:30 +01:00
|
|
|
#include <catch2/internal/catch_reusable_string_stream.hpp>
|
2020-08-08 18:18:27 +02:00
|
|
|
|
2022-04-14 17:16:01 +02:00
|
|
|
#include <algorithm>
|
|
|
|
#include <ostream>
|
|
|
|
|
2020-08-08 18:18:27 +02:00
|
|
|
namespace {
|
|
|
|
bool isOptPrefix( char c ) {
|
|
|
|
return c == '-'
|
|
|
|
#ifdef CATCH_PLATFORM_WINDOWS
|
|
|
|
|| c == '/'
|
|
|
|
#endif
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
2023-12-26 14:01:15 +01:00
|
|
|
Catch::StringRef normaliseOpt( Catch::StringRef optName ) {
|
|
|
|
if ( optName[0] == '-'
|
|
|
|
#if defined(CATCH_PLATFORM_WINDOWS)
|
|
|
|
|| optName[0] == '/'
|
2020-08-08 18:18:27 +02:00
|
|
|
#endif
|
2023-12-26 14:01:15 +01:00
|
|
|
) {
|
|
|
|
return optName.substr( 1, optName.size() );
|
|
|
|
}
|
|
|
|
|
|
|
|
return optName;
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
|
2023-12-27 16:19:49 +01:00
|
|
|
static size_t find_first_separator(Catch::StringRef sr) {
|
|
|
|
auto is_separator = []( char c ) {
|
|
|
|
return c == ' ' || c == ':' || c == '=';
|
|
|
|
};
|
|
|
|
size_t pos = 0;
|
|
|
|
while (pos < sr.size()) {
|
|
|
|
if (is_separator(sr[pos])) { return pos; }
|
|
|
|
++pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Catch::StringRef::npos;
|
|
|
|
}
|
|
|
|
|
2020-08-08 18:18:27 +02:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace Catch {
|
|
|
|
namespace Clara {
|
|
|
|
namespace Detail {
|
|
|
|
|
|
|
|
void TokenStream::loadBuffer() {
|
|
|
|
m_tokenBuffer.clear();
|
|
|
|
|
|
|
|
// Skip any empty strings
|
|
|
|
while ( it != itEnd && it->empty() ) {
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( it != itEnd ) {
|
2023-12-27 16:19:49 +01:00
|
|
|
StringRef next = *it;
|
2020-08-08 18:18:27 +02:00
|
|
|
if ( isOptPrefix( next[0] ) ) {
|
2023-12-27 16:19:49 +01:00
|
|
|
auto delimiterPos = find_first_separator(next);
|
|
|
|
if ( delimiterPos != StringRef::npos ) {
|
2020-08-08 18:18:27 +02:00
|
|
|
m_tokenBuffer.push_back(
|
|
|
|
{ TokenType::Option,
|
|
|
|
next.substr( 0, delimiterPos ) } );
|
|
|
|
m_tokenBuffer.push_back(
|
|
|
|
{ TokenType::Argument,
|
2023-12-27 16:19:49 +01:00
|
|
|
next.substr( delimiterPos + 1, next.size() ) } );
|
2020-08-08 18:18:27 +02:00
|
|
|
} else {
|
|
|
|
if ( next[1] != '-' && next.size() > 2 ) {
|
2023-12-27 16:19:49 +01:00
|
|
|
// Combined short args, e.g. "-ab" for "-a -b"
|
2020-08-08 18:18:27 +02:00
|
|
|
for ( size_t i = 1; i < next.size(); ++i ) {
|
|
|
|
m_tokenBuffer.push_back(
|
2023-12-27 16:19:49 +01:00
|
|
|
{ TokenType::Option,
|
|
|
|
next.substr( i, 1 ) } );
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m_tokenBuffer.push_back(
|
|
|
|
{ TokenType::Option, next } );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m_tokenBuffer.push_back(
|
|
|
|
{ TokenType::Argument, next } );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TokenStream::TokenStream( Args const& args ):
|
|
|
|
TokenStream( args.m_args.begin(), args.m_args.end() ) {}
|
|
|
|
|
2020-09-06 13:10:43 +02:00
|
|
|
TokenStream::TokenStream( Iterator it_, Iterator itEnd_ ):
|
|
|
|
it( it_ ), itEnd( itEnd_ ) {
|
2020-08-08 18:18:27 +02:00
|
|
|
loadBuffer();
|
|
|
|
}
|
|
|
|
|
|
|
|
TokenStream& TokenStream::operator++() {
|
|
|
|
if ( m_tokenBuffer.size() >= 2 ) {
|
|
|
|
m_tokenBuffer.erase( m_tokenBuffer.begin() );
|
|
|
|
} else {
|
|
|
|
if ( it != itEnd )
|
|
|
|
++it;
|
|
|
|
loadBuffer();
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
ParserResult convertInto( std::string const& source,
|
|
|
|
std::string& target ) {
|
|
|
|
target = source;
|
|
|
|
return ParserResult::ok( ParseResultType::Matched );
|
|
|
|
}
|
|
|
|
|
|
|
|
ParserResult convertInto( std::string const& source,
|
|
|
|
bool& target ) {
|
|
|
|
std::string srcLC = toLower( source );
|
|
|
|
|
|
|
|
if ( srcLC == "y" || srcLC == "1" || srcLC == "true" ||
|
|
|
|
srcLC == "yes" || srcLC == "on" ) {
|
|
|
|
target = true;
|
|
|
|
} else if ( srcLC == "n" || srcLC == "0" || srcLC == "false" ||
|
|
|
|
srcLC == "no" || srcLC == "off" ) {
|
|
|
|
target = false;
|
|
|
|
} else {
|
|
|
|
return ParserResult::runtimeError(
|
|
|
|
"Expected a boolean value but did not recognise: '" +
|
2021-10-03 10:09:10 +02:00
|
|
|
source + '\'' );
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
return ParserResult::ok( ParseResultType::Matched );
|
|
|
|
}
|
|
|
|
|
2020-08-20 23:16:06 +02:00
|
|
|
size_t ParserBase::cardinality() const { return 1; }
|
|
|
|
|
2020-08-08 18:18:27 +02:00
|
|
|
InternalParseResult ParserBase::parse( Args const& args ) const {
|
2023-12-27 16:19:49 +01:00
|
|
|
return parse( static_cast<std::string>(args.exeName()), TokenStream( args ) );
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ParseState::ParseState( ParseResultType type,
|
2023-12-27 21:11:34 +01:00
|
|
|
TokenStream remainingTokens ):
|
|
|
|
m_type( type ), m_remainingTokens( CATCH_MOVE(remainingTokens) ) {}
|
2020-08-08 18:18:27 +02:00
|
|
|
|
|
|
|
ParserResult BoundFlagRef::setFlag( bool flag ) {
|
|
|
|
m_ref = flag;
|
|
|
|
return ParserResult::ok( ParseResultType::Matched );
|
|
|
|
}
|
|
|
|
|
2020-08-20 23:16:06 +02:00
|
|
|
ResultBase::~ResultBase() = default;
|
|
|
|
|
|
|
|
bool BoundRef::isContainer() const { return false; }
|
|
|
|
|
|
|
|
bool BoundRef::isFlag() const { return false; }
|
|
|
|
|
|
|
|
bool BoundFlagRefBase::isFlag() const { return true; }
|
|
|
|
|
|
|
|
} // namespace Detail
|
2020-08-08 18:18:27 +02:00
|
|
|
|
|
|
|
Detail::InternalParseResult Arg::parse(std::string const&,
|
2023-12-27 21:11:34 +01:00
|
|
|
Detail::TokenStream tokens) const {
|
2020-08-08 18:18:27 +02:00
|
|
|
auto validationResult = validate();
|
|
|
|
if (!validationResult)
|
|
|
|
return Detail::InternalParseResult(validationResult);
|
|
|
|
|
2023-12-27 21:11:34 +01:00
|
|
|
auto token = *tokens;
|
2020-08-08 18:18:27 +02:00
|
|
|
if (token.type != Detail::TokenType::Argument)
|
|
|
|
return Detail::InternalParseResult::ok(Detail::ParseState(
|
2023-12-27 21:11:34 +01:00
|
|
|
ParseResultType::NoMatch, CATCH_MOVE(tokens)));
|
2020-08-08 18:18:27 +02:00
|
|
|
|
|
|
|
assert(!m_ref->isFlag());
|
|
|
|
auto valueRef =
|
|
|
|
static_cast<Detail::BoundValueRefBase*>(m_ref.get());
|
|
|
|
|
2023-12-27 16:19:49 +01:00
|
|
|
auto result = valueRef->setValue(static_cast<std::string>(token.token));
|
2023-12-27 21:11:34 +01:00
|
|
|
if ( !result )
|
|
|
|
return Detail::InternalParseResult( result );
|
2020-08-08 18:18:27 +02:00
|
|
|
else
|
2023-12-27 21:11:34 +01:00
|
|
|
return Detail::InternalParseResult::ok(
|
|
|
|
Detail::ParseState( ParseResultType::Matched,
|
|
|
|
CATCH_MOVE( ++tokens ) ) );
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Opt::Opt(bool& ref) :
|
|
|
|
ParserRefImpl(std::make_shared<Detail::BoundFlagRef>(ref)) {}
|
|
|
|
|
2023-12-23 16:06:30 +01:00
|
|
|
Detail::HelpColumns Opt::getHelpColumns() const {
|
|
|
|
ReusableStringStream oss;
|
2020-08-08 18:18:27 +02:00
|
|
|
bool first = true;
|
|
|
|
for (auto const& opt : m_optNames) {
|
|
|
|
if (first)
|
|
|
|
first = false;
|
|
|
|
else
|
|
|
|
oss << ", ";
|
|
|
|
oss << opt;
|
|
|
|
}
|
|
|
|
if (!m_hint.empty())
|
|
|
|
oss << " <" << m_hint << '>';
|
2023-12-23 16:06:30 +01:00
|
|
|
return { oss.str(), m_description };
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
|
2023-12-27 16:19:49 +01:00
|
|
|
bool Opt::isMatch(StringRef optToken) const {
|
2020-08-08 18:18:27 +02:00
|
|
|
auto normalisedToken = normaliseOpt(optToken);
|
|
|
|
for (auto const& name : m_optNames) {
|
|
|
|
if (normaliseOpt(name) == normalisedToken)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Detail::InternalParseResult Opt::parse(std::string const&,
|
2023-12-27 21:11:34 +01:00
|
|
|
Detail::TokenStream tokens) const {
|
2020-08-08 18:18:27 +02:00
|
|
|
auto validationResult = validate();
|
|
|
|
if (!validationResult)
|
|
|
|
return Detail::InternalParseResult(validationResult);
|
|
|
|
|
2023-12-27 21:11:34 +01:00
|
|
|
if (tokens &&
|
|
|
|
tokens->type == Detail::TokenType::Option) {
|
|
|
|
auto const& token = *tokens;
|
2020-08-08 18:18:27 +02:00
|
|
|
if (isMatch(token.token)) {
|
|
|
|
if (m_ref->isFlag()) {
|
|
|
|
auto flagRef =
|
|
|
|
static_cast<Detail::BoundFlagRefBase*>(
|
|
|
|
m_ref.get());
|
|
|
|
auto result = flagRef->setFlag(true);
|
|
|
|
if (!result)
|
|
|
|
return Detail::InternalParseResult(result);
|
|
|
|
if (result.value() ==
|
|
|
|
ParseResultType::ShortCircuitAll)
|
|
|
|
return Detail::InternalParseResult::ok(Detail::ParseState(
|
2023-12-27 21:11:34 +01:00
|
|
|
result.value(), CATCH_MOVE(tokens)));
|
2020-08-08 18:18:27 +02:00
|
|
|
} else {
|
|
|
|
auto valueRef =
|
|
|
|
static_cast<Detail::BoundValueRefBase*>(
|
|
|
|
m_ref.get());
|
2023-12-27 21:11:34 +01:00
|
|
|
++tokens;
|
|
|
|
if (!tokens)
|
2020-08-08 18:18:27 +02:00
|
|
|
return Detail::InternalParseResult::runtimeError(
|
|
|
|
"Expected argument following " +
|
|
|
|
token.token);
|
2023-12-27 21:11:34 +01:00
|
|
|
auto const& argToken = *tokens;
|
2020-08-08 18:18:27 +02:00
|
|
|
if (argToken.type != Detail::TokenType::Argument)
|
|
|
|
return Detail::InternalParseResult::runtimeError(
|
|
|
|
"Expected argument following " +
|
|
|
|
token.token);
|
2023-12-27 16:19:49 +01:00
|
|
|
const auto result = valueRef->setValue(static_cast<std::string>(argToken.token));
|
2020-08-08 18:18:27 +02:00
|
|
|
if (!result)
|
|
|
|
return Detail::InternalParseResult(result);
|
|
|
|
if (result.value() ==
|
|
|
|
ParseResultType::ShortCircuitAll)
|
|
|
|
return Detail::InternalParseResult::ok(Detail::ParseState(
|
2023-12-27 21:11:34 +01:00
|
|
|
result.value(), CATCH_MOVE(tokens)));
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
return Detail::InternalParseResult::ok(Detail::ParseState(
|
2023-12-27 21:11:34 +01:00
|
|
|
ParseResultType::Matched, CATCH_MOVE(++tokens)));
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return Detail::InternalParseResult::ok(
|
2023-12-27 21:11:34 +01:00
|
|
|
Detail::ParseState(ParseResultType::NoMatch, CATCH_MOVE(tokens)));
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Detail::Result Opt::validate() const {
|
|
|
|
if (m_optNames.empty())
|
|
|
|
return Detail::Result::logicError("No options supplied to Opt");
|
|
|
|
for (auto const& name : m_optNames) {
|
|
|
|
if (name.empty())
|
|
|
|
return Detail::Result::logicError(
|
|
|
|
"Option name cannot be empty");
|
|
|
|
#ifdef CATCH_PLATFORM_WINDOWS
|
|
|
|
if (name[0] != '-' && name[0] != '/')
|
|
|
|
return Detail::Result::logicError(
|
|
|
|
"Option name must begin with '-' or '/'");
|
|
|
|
#else
|
|
|
|
if (name[0] != '-')
|
|
|
|
return Detail::Result::logicError(
|
|
|
|
"Option name must begin with '-'");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return ParserRefImpl::validate();
|
|
|
|
}
|
|
|
|
|
|
|
|
ExeName::ExeName() :
|
|
|
|
m_name(std::make_shared<std::string>("<executable>")) {}
|
|
|
|
|
|
|
|
ExeName::ExeName(std::string& ref) : ExeName() {
|
|
|
|
m_ref = std::make_shared<Detail::BoundValueRef<std::string>>(ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
Detail::InternalParseResult
|
|
|
|
ExeName::parse(std::string const&,
|
2023-12-27 21:11:34 +01:00
|
|
|
Detail::TokenStream tokens) const {
|
2020-08-08 18:18:27 +02:00
|
|
|
return Detail::InternalParseResult::ok(
|
2023-12-27 21:11:34 +01:00
|
|
|
Detail::ParseState(ParseResultType::NoMatch, CATCH_MOVE(tokens)));
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ParserResult ExeName::set(std::string const& newName) {
|
|
|
|
auto lastSlash = newName.find_last_of("\\/");
|
|
|
|
auto filename = (lastSlash == std::string::npos)
|
|
|
|
? newName
|
|
|
|
: newName.substr(lastSlash + 1);
|
|
|
|
|
|
|
|
*m_name = filename;
|
|
|
|
if (m_ref)
|
|
|
|
return m_ref->setValue(filename);
|
|
|
|
else
|
|
|
|
return ParserResult::ok(ParseResultType::Matched);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Parser& Parser::operator|=( Parser const& other ) {
|
|
|
|
m_options.insert( m_options.end(),
|
|
|
|
other.m_options.begin(),
|
|
|
|
other.m_options.end() );
|
|
|
|
m_args.insert(
|
|
|
|
m_args.end(), other.m_args.begin(), other.m_args.end() );
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<Detail::HelpColumns> Parser::getHelpColumns() const {
|
|
|
|
std::vector<Detail::HelpColumns> cols;
|
2023-12-23 16:06:30 +01:00
|
|
|
cols.reserve( m_options.size() );
|
2020-08-08 18:18:27 +02:00
|
|
|
for ( auto const& o : m_options ) {
|
2023-12-23 16:06:30 +01:00
|
|
|
cols.push_back(o.getHelpColumns());
|
2020-08-08 18:18:27 +02:00
|
|
|
}
|
|
|
|
return cols;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Parser::writeToStream( std::ostream& os ) const {
|
|
|
|
if ( !m_exeName.name().empty() ) {
|
|
|
|
os << "usage:\n"
|
|
|
|
<< " " << m_exeName.name() << ' ';
|
|
|
|
bool required = true, first = true;
|
|
|
|
for ( auto const& arg : m_args ) {
|
|
|
|
if ( first )
|
|
|
|
first = false;
|
|
|
|
else
|
|
|
|
os << ' ';
|
|
|
|
if ( arg.isOptional() && required ) {
|
|
|
|
os << '[';
|
|
|
|
required = false;
|
|
|
|
}
|
|
|
|
os << '<' << arg.hint() << '>';
|
|
|
|
if ( arg.cardinality() == 0 )
|
|
|
|
os << " ... ";
|
|
|
|
}
|
|
|
|
if ( !required )
|
|
|
|
os << ']';
|
|
|
|
if ( !m_options.empty() )
|
|
|
|
os << " options";
|
|
|
|
os << "\n\nwhere options are:\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
auto rows = getHelpColumns();
|
|
|
|
size_t consoleWidth = CATCH_CONFIG_CONSOLE_WIDTH;
|
|
|
|
size_t optWidth = 0;
|
|
|
|
for ( auto const& cols : rows )
|
|
|
|
optWidth = ( std::max )( optWidth, cols.left.size() + 2 );
|
|
|
|
|
|
|
|
optWidth = ( std::min )( optWidth, consoleWidth / 2 );
|
|
|
|
|
2023-12-27 21:43:40 +01:00
|
|
|
for ( auto& cols : rows ) {
|
|
|
|
auto row = TextFlow::Column( CATCH_MOVE(cols.left) )
|
2020-08-08 18:18:27 +02:00
|
|
|
.width( optWidth )
|
|
|
|
.indent( 2 ) +
|
|
|
|
TextFlow::Spacer( 4 ) +
|
2023-12-23 16:06:30 +01:00
|
|
|
TextFlow::Column( static_cast<std::string>(cols.descriptions) )
|
2020-08-08 18:18:27 +02:00
|
|
|
.width( consoleWidth - 7 - optWidth );
|
|
|
|
os << row << '\n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Detail::Result Parser::validate() const {
|
|
|
|
for ( auto const& opt : m_options ) {
|
|
|
|
auto result = opt.validate();
|
|
|
|
if ( !result )
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
for ( auto const& arg : m_args ) {
|
|
|
|
auto result = arg.validate();
|
|
|
|
if ( !result )
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
return Detail::Result::ok();
|
|
|
|
}
|
|
|
|
|
|
|
|
Detail::InternalParseResult
|
|
|
|
Parser::parse( std::string const& exeName,
|
2023-12-27 21:11:34 +01:00
|
|
|
Detail::TokenStream tokens ) const {
|
2020-08-08 18:18:27 +02:00
|
|
|
|
|
|
|
struct ParserInfo {
|
|
|
|
ParserBase const* parser = nullptr;
|
|
|
|
size_t count = 0;
|
|
|
|
};
|
|
|
|
std::vector<ParserInfo> parseInfos;
|
|
|
|
parseInfos.reserve( m_options.size() + m_args.size() );
|
|
|
|
for ( auto const& opt : m_options ) {
|
|
|
|
parseInfos.push_back( { &opt, 0 } );
|
|
|
|
}
|
|
|
|
for ( auto const& arg : m_args ) {
|
|
|
|
parseInfos.push_back( { &arg, 0 } );
|
|
|
|
}
|
|
|
|
|
|
|
|
m_exeName.set( exeName );
|
|
|
|
|
|
|
|
auto result = Detail::InternalParseResult::ok(
|
2023-12-27 21:11:34 +01:00
|
|
|
Detail::ParseState( ParseResultType::NoMatch, CATCH_MOVE(tokens) ) );
|
2020-08-08 18:18:27 +02:00
|
|
|
while ( result.value().remainingTokens() ) {
|
|
|
|
bool tokenParsed = false;
|
|
|
|
|
|
|
|
for ( auto& parseInfo : parseInfos ) {
|
|
|
|
if ( parseInfo.parser->cardinality() == 0 ||
|
|
|
|
parseInfo.count < parseInfo.parser->cardinality() ) {
|
|
|
|
result = parseInfo.parser->parse(
|
2023-12-27 21:11:34 +01:00
|
|
|
exeName, CATCH_MOVE(result).value().remainingTokens() );
|
2020-08-08 18:18:27 +02:00
|
|
|
if ( !result )
|
|
|
|
return result;
|
|
|
|
if ( result.value().type() !=
|
|
|
|
ParseResultType::NoMatch ) {
|
|
|
|
tokenParsed = true;
|
|
|
|
++parseInfo.count;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( result.value().type() == ParseResultType::ShortCircuitAll )
|
|
|
|
return result;
|
|
|
|
if ( !tokenParsed )
|
|
|
|
return Detail::InternalParseResult::runtimeError(
|
|
|
|
"Unrecognised token: " +
|
|
|
|
result.value().remainingTokens()->token );
|
|
|
|
}
|
|
|
|
// !TBD Check missing required options
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Args::Args(int argc, char const* const* argv) :
|
|
|
|
m_exeName(argv[0]), m_args(argv + 1, argv + argc) {}
|
|
|
|
|
2023-12-27 16:19:49 +01:00
|
|
|
Args::Args(std::initializer_list<StringRef> args) :
|
2020-08-08 18:18:27 +02:00
|
|
|
m_exeName(*args.begin()),
|
|
|
|
m_args(args.begin() + 1, args.end()) {}
|
|
|
|
|
|
|
|
|
|
|
|
Help::Help( bool& showHelpFlag ):
|
|
|
|
Opt( [&]( bool flag ) {
|
|
|
|
showHelpFlag = flag;
|
|
|
|
return ParserResult::ok( ParseResultType::ShortCircuitAll );
|
|
|
|
} ) {
|
|
|
|
static_cast<Opt&> ( *this )(
|
|
|
|
"display usage information" )["-?"]["-h"]["--help"]
|
|
|
|
.optional();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Clara
|
|
|
|
} // namespace Catch
|