summaryrefslogtreecommitdiff
path: root/test
diff options
context:
space:
mode:
authorDan Goodliffe <dan@randomdan.homeip.net>2022-10-22 17:57:47 +0100
committerDan Goodliffe <dan@randomdan.homeip.net>2022-10-22 17:57:47 +0100
commit38d7045685d4904d013ea4990a3aa7f5e78309cf (patch)
tree60915147de5e37dc4a3fab39408b6abbfc49ce09 /test
parentAdd free extend builder with placeholder network support (diff)
downloadilt-38d7045685d4904d013ea4990a3aa7f5e78309cf.tar.bz2
ilt-38d7045685d4904d013ea4990a3aa7f5e78309cf.tar.xz
ilt-38d7045685d4904d013ea4990a3aa7f5e78309cf.zip
Add magic support to printing/parsing/validating enumerations
Diffstat (limited to 'test')
-rw-r--r--test/Jamfile.jam2
-rw-r--r--test/enumDetailsData.hpp20
-rw-r--r--test/test-enumDetails.cpp50
-rw-r--r--test/test-static-enumDetails.cpp78
4 files changed, 150 insertions, 0 deletions
diff --git a/test/Jamfile.jam b/test/Jamfile.jam
index afe58c8..7606ae4 100644
--- a/test/Jamfile.jam
+++ b/test/Jamfile.jam
@@ -29,3 +29,5 @@ run test-geo.cpp ;
run test-network.cpp ;
run test-persistence.cpp : -- : [ sequence.insertion-sort [ glob fixtures/json/*.json fixtures/json/bad/*.json ] ] ;
run test-text.cpp ;
+run test-enumDetails.cpp ;
+compile test-static-enumDetails.cpp ;
diff --git a/test/enumDetailsData.hpp b/test/enumDetailsData.hpp
new file mode 100644
index 0000000..0e98af5
--- /dev/null
+++ b/test/enumDetailsData.hpp
@@ -0,0 +1,20 @@
+#pragma once
+#include <enumDetails.hpp>
+
+enum GlobalUnscoped { aa, b, c };
+enum class GlobalScoped { aa, b, c };
+namespace ns {
+ enum Unscoped { aa, b, c };
+ enum class Scoped { aa, b, c };
+}
+namespace test1 {
+ enum class DefaultDense { a, bee, ci, de };
+}
+namespace test2 {
+ enum class NumberedSparse { a = 0, bee = 3, ci = -20, de = 100 };
+}
+
+template<> struct EnumValueCollection<test2::NumberedSparse> {
+ // Any ordered integer_sequence which includes all enumeration values
+ using Vs = std::integer_sequence<int, -100, -20, 0, 3, 10, 100, 1000>;
+};
diff --git a/test/test-enumDetails.cpp b/test/test-enumDetails.cpp
new file mode 100644
index 0000000..c803cf8
--- /dev/null
+++ b/test/test-enumDetails.cpp
@@ -0,0 +1,50 @@
+#define BOOST_TEST_MODULE test_enumDetails
+
+#include "enumDetailsData.hpp"
+#include <boost/test/data/test_case.hpp>
+#include <boost/test/unit_test.hpp>
+#include <enumDetails.hpp>
+#include <stream_support.hpp>
+
+constexpr std::array INVALID_NAMES {"", "missing", "GlobalScoped::aa", "GlobalScoped", "ns::aa", "a", "bb"};
+constexpr std::array VALID_NAMES {"aa", "b", "c"};
+template<typename E> constexpr std::array VALID_VALUES {E::aa, E::b, E::c};
+// Not a template, else Boost test framework throws printing the context
+constexpr std::array INVALID_VALUES {-1, 3, 20};
+
+#define TESTS_FOR_TYPE(TYPE) \
+ BOOST_DATA_TEST_CASE(invalid_check_##TYPE, INVALID_VALUES, in) \
+ { \
+ BOOST_CHECK(!EnumDetails<TYPE>::is_valid(static_cast<TYPE>(in))); \
+ } \
+ BOOST_DATA_TEST_CASE(invalid_parse_##TYPE, INVALID_NAMES, in) \
+ { \
+ BOOST_CHECK(!EnumDetails<TYPE>::parse(in).has_value()); \
+ } \
+ BOOST_DATA_TEST_CASE(invalid_to_string_##TYPE, INVALID_VALUES, in) \
+ { \
+ BOOST_CHECK(!EnumDetails<TYPE>::to_string(static_cast<TYPE>(in)).has_value()); \
+ } \
+ BOOST_DATA_TEST_CASE(valid_check_##TYPE, VALID_VALUES<TYPE>, in) \
+ { \
+ BOOST_CHECK(EnumDetails<TYPE>::is_valid(in)); \
+ } \
+ BOOST_DATA_TEST_CASE(valid_parse_##TYPE, VALID_NAMES ^ VALID_VALUES<TYPE>, in, out) \
+ { \
+ const auto v = EnumDetails<TYPE>::parse(in); \
+ BOOST_REQUIRE(v.has_value()); \
+ BOOST_CHECK_EQUAL(v.value(), out); \
+ } \
+ BOOST_DATA_TEST_CASE(valid_to_string_##TYPE, VALID_VALUES<TYPE> ^ VALID_NAMES, in, out) \
+ { \
+ const auto v = EnumDetails<TYPE>::to_string(in); \
+ BOOST_CHECK(v.has_value()); \
+ BOOST_CHECK_EQUAL(v.value(), out); \
+ }
+
+TESTS_FOR_TYPE(GlobalScoped)
+TESTS_FOR_TYPE(GlobalUnscoped)
+using ns_unscoped = ns::Unscoped;
+using ns_scoped = ns::Scoped;
+TESTS_FOR_TYPE(ns_unscoped)
+TESTS_FOR_TYPE(ns_scoped)
diff --git a/test/test-static-enumDetails.cpp b/test/test-static-enumDetails.cpp
new file mode 100644
index 0000000..03c2203
--- /dev/null
+++ b/test/test-static-enumDetails.cpp
@@ -0,0 +1,78 @@
+#define ENUM_PROBE
+#include "enumDetailsData.hpp"
+#include <enumDetails.hpp>
+
+// Test type name
+static_assert(EnumTypeDetails<GlobalUnscoped>::typeName == "GlobalUnscoped");
+static_assert(EnumTypeDetails<GlobalScoped>::typeName == "GlobalScoped");
+static_assert(EnumTypeDetails<ns::Unscoped>::typeName == "ns::Unscoped");
+static_assert(EnumTypeDetails<ns::Scoped>::typeName == "ns::Scoped");
+
+static_assert(EnumValueDetails<GlobalUnscoped::aa>::valueName == "aa");
+static_assert(EnumValueDetails<GlobalScoped::aa>::valueName == "aa");
+static_assert(EnumValueDetails<ns::Unscoped::aa>::valueName == "aa");
+static_assert(EnumValueDetails<ns::Scoped::aa>::valueName == "aa");
+
+namespace test1 {
+ static_assert(EnumValueDetails<DefaultDense::a>::valid);
+ static_assert(EnumValueDetails<DefaultDense::de>::valid);
+ static_assert(EnumValueDetails<static_cast<DefaultDense>(0)>::valid);
+ static_assert(EnumValueDetails<static_cast<DefaultDense>(3)>::valid);
+ static_assert(!EnumValueDetails<static_cast<DefaultDense>(-1)>::valid);
+ static_assert(!EnumValueDetails<static_cast<DefaultDense>(4)>::valid);
+ static_assert(EnumValueDetails<DefaultDense::a>::valueName == "a");
+ static_assert(EnumValueDetails<DefaultDense::de>::valueName == "de");
+ using ED_DD = EnumDetails<DefaultDense>;
+ static_assert(EnumValueCollection<DefaultDense>::Vs::size() == 256);
+ static_assert(ED_DD::valid_flags.size() == 256);
+ static_assert(ED_DD::values.size() == 4);
+ static_assert(std::is_sorted(ED_DD::values.begin(), ED_DD::values.end()));
+ static_assert(ED_DD::values.at(0) == DefaultDense::a);
+ static_assert(ED_DD::values.at(3) == DefaultDense::de);
+ static_assert(ED_DD::names.at(0) == "a");
+ static_assert(ED_DD::names.at(3) == "de");
+
+ static_assert(ED_DD::is_valid(DefaultDense::a));
+ static_assert(ED_DD::is_valid(DefaultDense::de));
+ static_assert(!ED_DD::is_valid(DefaultDense(-1)));
+ static_assert(!ED_DD::parse("").has_value());
+ static_assert(!ED_DD::parse("nonsense").has_value());
+ static_assert(ED_DD::parse("bee").value() == DefaultDense::bee);
+ static_assert(ED_DD::parse("ci").value() == DefaultDense::ci);
+ static_assert(ED_DD::to_string(DefaultDense::de).value() == "de");
+ static_assert(!ED_DD::to_string(static_cast<DefaultDense>(10)).has_value());
+}
+
+namespace test2 {
+ static_assert(EnumValueDetails<NumberedSparse::bee>::valid);
+ static_assert(EnumValueDetails<static_cast<NumberedSparse>(0)>::valid);
+ static_assert(EnumValueDetails<static_cast<NumberedSparse>(3)>::valid);
+ static_assert(EnumValueDetails<static_cast<NumberedSparse>(-20)>::valid);
+ static_assert(EnumValueDetails<static_cast<NumberedSparse>(100)>::valid);
+ static_assert(!EnumValueDetails<static_cast<NumberedSparse>(2)>::valid);
+ static_assert(EnumValueDetails<NumberedSparse::a>::valueName == "a");
+ static_assert(EnumValueDetails<NumberedSparse::de>::valueName == "de");
+ using ED_NS = EnumDetails<NumberedSparse>;
+ static_assert(EnumValueCollection<NumberedSparse>::Vs::size() == 7);
+ static_assert(ED_NS::values.size() == 4);
+ static_assert(ED_NS::valid_flags.size() == 7);
+ static_assert(std::is_sorted(ED_NS::values.begin(), ED_NS::values.end()));
+ static_assert(ED_NS::values.at(0) == NumberedSparse::ci);
+ static_assert(ED_NS::values.at(1) == NumberedSparse::a);
+ static_assert(ED_NS::values.at(2) == NumberedSparse::bee);
+ static_assert(ED_NS::values.at(3) == NumberedSparse::de);
+ static_assert(ED_NS::names.at(0) == "ci");
+ static_assert(ED_NS::names.at(1) == "a");
+ static_assert(ED_NS::names.at(2) == "bee");
+ static_assert(ED_NS::names.at(3) == "de");
+
+ static_assert(ED_NS::is_valid(NumberedSparse::a));
+ static_assert(ED_NS::is_valid(NumberedSparse::de));
+ static_assert(!ED_NS::is_valid(NumberedSparse(-1)));
+ static_assert(!ED_NS::parse("").has_value());
+ static_assert(!ED_NS::parse("nonsense").has_value());
+ static_assert(ED_NS::parse("bee").value() == NumberedSparse::bee);
+ static_assert(ED_NS::parse("ci").value() == NumberedSparse::ci);
+ static_assert(ED_NS::to_string(NumberedSparse::ci).value() == "ci");
+ static_assert(!ED_NS::to_string(static_cast<NumberedSparse>(10)).has_value());
+}