|
|
|
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
|
|
#include "cmArgumentParser.h"
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include "cmArgumentParserTypes.h"
|
|
|
|
#include "cmMakefile.h"
|
|
|
|
#include "cmMessageType.h"
|
|
|
|
#include "cmStringAlgorithms.h"
|
|
|
|
|
|
|
|
namespace ArgumentParser {
|
|
|
|
|
|
|
|
auto KeywordActionMap::Emplace(cm::string_view name, KeywordAction action)
|
|
|
|
-> std::pair<iterator, bool>
|
|
|
|
{
|
|
|
|
auto const it =
|
|
|
|
std::lower_bound(this->begin(), this->end(), name,
|
|
|
|
[](value_type const& elem, cm::string_view const& k) {
|
|
|
|
return elem.first < k;
|
|
|
|
});
|
|
|
|
return (it != this->end() && it->first == name)
|
|
|
|
? std::make_pair(it, false)
|
|
|
|
: std::make_pair(this->emplace(it, name, std::move(action)), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto KeywordActionMap::Find(cm::string_view name) const -> const_iterator
|
|
|
|
{
|
|
|
|
auto const it =
|
|
|
|
std::lower_bound(this->begin(), this->end(), name,
|
|
|
|
[](value_type const& elem, cm::string_view const& k) {
|
|
|
|
return elem.first < k;
|
|
|
|
});
|
|
|
|
return (it != this->end() && it->first == name) ? it : this->end();
|
|
|
|
}
|
|
|
|
|
|
|
|
auto PositionActionMap::Emplace(std::size_t pos, PositionAction action)
|
|
|
|
-> std::pair<iterator, bool>
|
|
|
|
{
|
|
|
|
auto const it = std::lower_bound(
|
|
|
|
this->begin(), this->end(), pos,
|
|
|
|
[](value_type const& elem, std::size_t k) { return elem.first < k; });
|
|
|
|
return (it != this->end() && it->first == pos)
|
|
|
|
? std::make_pair(it, false)
|
|
|
|
: std::make_pair(this->emplace(it, pos, std::move(action)), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto PositionActionMap::Find(std::size_t pos) const -> const_iterator
|
|
|
|
{
|
|
|
|
auto const it = std::lower_bound(
|
|
|
|
this->begin(), this->end(), pos,
|
|
|
|
[](value_type const& elem, std::size_t k) { return elem.first < k; });
|
|
|
|
return (it != this->end() && it->first == pos) ? it : this->end();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(std::function<Continue(cm::string_view)> f,
|
|
|
|
ExpectAtLeast expect)
|
|
|
|
{
|
|
|
|
this->KeywordValueFunc = std::move(f);
|
|
|
|
this->KeywordValuesExpected = expect.Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(bool& val)
|
|
|
|
{
|
|
|
|
val = true;
|
|
|
|
this->Bind(nullptr, ExpectAtLeast{ 0 });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(std::string& val)
|
|
|
|
{
|
|
|
|
this->Bind(
|
|
|
|
[&val](cm::string_view arg) -> Continue {
|
|
|
|
val = std::string(arg);
|
|
|
|
return Continue::No;
|
|
|
|
},
|
|
|
|
ExpectAtLeast{ 1 });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(NonEmpty<std::string>& val)
|
|
|
|
{
|
|
|
|
this->Bind(
|
|
|
|
[this, &val](cm::string_view arg) -> Continue {
|
|
|
|
if (arg.empty() && this->ParseResults) {
|
|
|
|
this->ParseResults->AddKeywordError(this->Keyword,
|
|
|
|
" empty string not allowed\n");
|
|
|
|
}
|
|
|
|
val.assign(std::string(arg));
|
|
|
|
return Continue::No;
|
|
|
|
},
|
|
|
|
ExpectAtLeast{ 1 });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(Maybe<std::string>& val)
|
|
|
|
{
|
|
|
|
this->Bind(
|
|
|
|
[&val](cm::string_view arg) -> Continue {
|
|
|
|
static_cast<std::string&>(val) = std::string(arg);
|
|
|
|
return Continue::No;
|
|
|
|
},
|
|
|
|
ExpectAtLeast{ 0 });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(MaybeEmpty<std::vector<std::string>>& val)
|
|
|
|
{
|
|
|
|
this->Bind(
|
|
|
|
[&val](cm::string_view arg) -> Continue {
|
|
|
|
val.emplace_back(arg);
|
|
|
|
return Continue::Yes;
|
|
|
|
},
|
|
|
|
ExpectAtLeast{ 0 });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(NonEmpty<std::vector<std::string>>& val)
|
|
|
|
{
|
|
|
|
this->Bind(
|
|
|
|
[&val](cm::string_view arg) -> Continue {
|
|
|
|
val.emplace_back(arg);
|
|
|
|
return Continue::Yes;
|
|
|
|
},
|
|
|
|
ExpectAtLeast{ 1 });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Bind(std::vector<std::vector<std::string>>& multiVal)
|
|
|
|
{
|
|
|
|
multiVal.emplace_back();
|
|
|
|
std::vector<std::string>& val = multiVal.back();
|
|
|
|
this->Bind(
|
|
|
|
[&val](cm::string_view arg) -> Continue {
|
|
|
|
val.emplace_back(arg);
|
|
|
|
return Continue::Yes;
|
|
|
|
},
|
|
|
|
ExpectAtLeast{ 0 });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::Consume(std::size_t pos, cm::string_view arg)
|
|
|
|
{
|
|
|
|
auto const it = this->Bindings.Keywords.Find(arg);
|
|
|
|
if (it != this->Bindings.Keywords.end()) {
|
|
|
|
this->FinishKeyword();
|
|
|
|
this->Keyword = it->first;
|
|
|
|
this->KeywordValuesSeen = 0;
|
|
|
|
this->DoneWithPositional = true;
|
|
|
|
if (this->Bindings.ParsedKeyword) {
|
|
|
|
this->Bindings.ParsedKeyword(*this, it->first);
|
|
|
|
}
|
|
|
|
it->second(*this);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this->KeywordValueFunc) {
|
|
|
|
switch (this->KeywordValueFunc(arg)) {
|
|
|
|
case Continue::Yes:
|
|
|
|
break;
|
|
|
|
case Continue::No:
|
|
|
|
this->KeywordValueFunc = nullptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++this->KeywordValuesSeen;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this->DoneWithPositional) {
|
|
|
|
auto const pit = this->Bindings.Positions.Find(pos);
|
|
|
|
if (pit != this->Bindings.Positions.end()) {
|
|
|
|
pit->second(*this, pos, arg);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this->UnparsedArguments) {
|
|
|
|
this->UnparsedArguments->emplace_back(arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Instance::FinishKeyword()
|
|
|
|
{
|
|
|
|
if (this->Keyword.empty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (this->KeywordValuesSeen < this->KeywordValuesExpected) {
|
|
|
|
if (this->ParseResults) {
|
|
|
|
this->ParseResults->AddKeywordError(this->Keyword,
|
|
|
|
" missing required value\n");
|
|
|
|
}
|
|
|
|
if (this->Bindings.KeywordMissingValue) {
|
|
|
|
this->Bindings.KeywordMissingValue(*this, this->Keyword);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ParseResult::MaybeReportError(cmMakefile& mf) const
|
|
|
|
{
|
|
|
|
if (*this) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
std::string e;
|
|
|
|
for (auto const& ke : this->KeywordErrors) {
|
|
|
|
e = cmStrCat(e, "Error after keyword \"", ke.first, "\":\n", ke.second);
|
|
|
|
}
|
|
|
|
mf.IssueMessage(MessageType::FATAL_ERROR, e);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace ArgumentParser
|