2
0
mirror of https://github.com/boostorg/parser.git synced 2026-01-25 18:32:17 +00:00
Files
parser/test/compile_attribute.cpp
Zach Laine 852559ce67 static_assert inside the prase() overloads that take an attr out-param that
the parser generates an attribute.

Fixes #2.
2020-09-01 01:24:12 -05:00

557 lines
19 KiB
C++

// Copyright (C) 2018 T. Zachary Laine
//
// Distributed under the Boost Software License, Version 1.0. (See
// accompanying file LICENSE_1_0.txt or copy at
// http://www.boost.org/LICENSE_1_0.txt)
#include <boost/parser/parser.hpp>
#include <boost/mpl/assert.hpp>
#include <boost/type_traits/is_same.hpp>
using namespace boost::parser;
using boost::is_same;
using boost::hana::tuple;
void compile_attribute_non_unicode()
{
{
char const r[] = "";
{
constexpr auto parser = char_;
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<char>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
}
{
char const chars[] = "";
auto first = std::begin(chars);
auto const last = std::end(chars);
{
constexpr auto parser = char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<char>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
}
}
void compile_attribute_unicode_utf8()
{
{
char8_t const chars[] = u8"";
auto first = std::begin(chars);
auto const last = std::end(chars);
{
constexpr auto parser = char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<uint32_t>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<uint32_t>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<uint32_t, std::string>>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<std::string, uint32_t>>>));
}
}
{
char const chars[] = "";
auto const r = boost::text::as_utf8(chars);
{
constexpr auto parser = char_;
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<uint32_t>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<uint32_t>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<uint32_t, std::string>>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(r, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<std::string, uint32_t>>>));
}
}
}
void compile_attribute_unicode_utf32()
{
{
int const chars[] = {0};
auto first = std::begin(chars);
auto const last = std::end(chars);
{
constexpr auto parser = char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<int>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<int>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<int, std::string>>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<std::string, int>>>));
}
}
{
uint32_t const chars[] = {0};
auto first = std::begin(chars);
auto const last = std::end(chars);
{
constexpr auto parser = char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<uint32_t>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<uint32_t>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<uint32_t, std::string>>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<std::string, uint32_t>>>));
}
}
{
char32_t const chars[] = {0};
auto first = std::begin(chars);
auto const last = std::end(chars);
{
constexpr auto parser = char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char32_t>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<char32_t>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<char32_t, std::string>>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<std::string, char32_t>>>));
}
}
{
{
constexpr auto parser = char_;
using attr_t = decltype(parse(U"", parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char32_t>>));
}
{
constexpr auto parser = *char_;
using attr_t = decltype(parse(U"", parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::vector<char32_t>>>));
}
{
constexpr auto parser = string("foo");
using attr_t = decltype(parse(U"", parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = char_ >> string("foo");
using attr_t = decltype(parse(U"", parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<char32_t, std::string>>>));
}
{
constexpr auto parser = string("foo") >> char_;
using attr_t = decltype(parse(U"", parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<tuple<std::string, char32_t>>>));
}
}
}
void compile_attribute()
{
compile_attribute_non_unicode();
compile_attribute_unicode_utf8();
compile_attribute_unicode_utf32();
char const chars[] = "";
auto first = std::begin(chars);
auto const last = std::end(chars);
{
constexpr auto parser = eps;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
constexpr auto parser = eol;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
constexpr auto parser = eoi;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
constexpr auto parser = attr(3.0);
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<double>>));
}
{
constexpr auto parser = attr('c');
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char>>));
}
{
constexpr auto parser = cp;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<uint32_t>>));
}
{
constexpr auto parser = cu;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char>>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = 'c'_l;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = u8'c'_l;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = U'c'_l;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = "str"_l;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = u8"str"_l;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = U"str"_l;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = 'c'_p;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char>>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = u8'c'_p;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char>>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = U'c'_p;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<char>>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = "str"_p;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = u8"str"_p;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
using namespace boost::parser::literals;
constexpr auto parser = U"str"_p;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
constexpr auto parser = lit('c');
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
constexpr auto parser = lit(U'c');
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
constexpr auto parser = lit("str");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
std::string str = "str";
auto parser = lit(str);
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
{
constexpr auto parser = bool_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<bool>>));
}
{
constexpr auto parser = bin;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<unsigned int>>));
}
{
constexpr auto parser = oct;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<unsigned int>>));
}
{
constexpr auto parser = hex;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<unsigned int>>));
}
{
constexpr auto parser = ushort_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<unsigned short>>));
}
{
constexpr auto parser = uint_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<unsigned int>>));
}
{
constexpr auto parser = ulong_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<unsigned long>>));
}
{
constexpr auto parser = ulong_long;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<unsigned long long>>));
}
{
constexpr auto parser = short_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<short>>));
}
{
constexpr auto parser = int_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<int>>));
}
{
constexpr auto parser = long_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<long>>));
}
{
constexpr auto parser = long_long;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<long long>>));
}
{
constexpr auto parser = float_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<float>>));
}
{
constexpr auto parser = double_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<double>>));
}
{
symbols<float> parser;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<float>>));
}
{
auto const c = [](auto & ctx) { return true; };
constexpr auto parser = if_(c)[double_];
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<double>>));
}
{
auto const x = [](auto & ctx) { return 2; };
auto parser = switch_(x)(0, double_)(2, int_);
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::variant<double, int>>>));
}
{
constexpr auto parser = int_ | int_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<int>>));
}
{
constexpr auto parser = double_ | int_;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<attr_t, std::optional<std::variant<double, int>>>));
}
{
constexpr auto parser = double_ | int_ | eps;
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<
attr_t,
std::optional<std::optional<std::variant<double, int>>>>));
}
{
constexpr auto parser = *cu >> string("str");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT(
(is_same<
attr_t,
std::optional<tuple<std::vector<char>, std::string>>>));
}
{
constexpr auto parser = cu >> string("str");
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, std::optional<std::string>>));
}
{
auto a = [](auto & ctx) {};
constexpr auto parser = cu[a];
using attr_t = decltype(parse(first, last, parser));
BOOST_MPL_ASSERT((is_same<attr_t, bool>));
}
}