Files
meta.hpp/untests/meta_value_tests.cpp
2021-07-06 01:24:25 +07:00

222 lines
7.8 KiB
C++

/*******************************************************************************
* This file is part of the "https://github.com/blackmatov/meta.hpp"
* For conditions of distribution and use, see copyright notice in LICENSE.md
* Copyright (C) 2021, by Matvey Cherevko (blackmatov@gmail.com)
******************************************************************************/
#include <meta.hpp/meta_value.hpp>
#include "doctest/doctest.hpp"
#include <string>
#include <vector>
namespace
{
class clazz {};
class clazz2 {};
void func() {}
}
TEST_CASE("meta/value") {
namespace meta = meta_hpp;
using namespace std::string_literals;
CHECK(meta::value{true}.to_bool() == true);
CHECK(meta::value{1}.to_int() == 1);
CHECK(meta::value{1u}.to_uint() == 1u);
CHECK(meta::value{1.f}.to_float() == 1.f);
CHECK(meta::value{1.0}.to_double() == 1.0);
CHECK(meta::value{"meta"s}.to_string() == "meta");
CHECK(meta::value{true}.is_bool());
CHECK_FALSE(meta::value{true}.is_int());
CHECK(meta::value{1}.is_int());
CHECK_FALSE(meta::value{1}.is_uint());
CHECK(meta::value{1u}.is_uint());
CHECK_FALSE(meta::value{1u}.is_float());
CHECK(meta::value{1.f}.is_float());
CHECK_FALSE(meta::value{1.f}.is_double());
CHECK(meta::value{1.0}.is_double());
CHECK_FALSE(meta::value{1.0}.is_string());
CHECK(meta::value{"meta"s}.is_string());
CHECK_FALSE(meta::value{"meta"s}.is_bool());
CHECK(meta::value{std::in_place_type<std::int8_t>, std::int8_t{1}}.to_int8() == 1);
CHECK(meta::value{std::in_place_type<std::int16_t>, std::int16_t{1}}.to_int16() == 1);
CHECK(meta::value{std::in_place_type<std::int32_t>, std::int32_t{1}}.to_int32() == 1);
CHECK(meta::value{std::in_place_type<std::int64_t>, std::int64_t{1}}.to_int64() == 1);
CHECK(meta::value{std::in_place_type<std::ptrdiff_t>, std::ptrdiff_t{1}}.to_ptrdiff_t() == 1);
CHECK(meta::value{std::in_place_type<std::intptr_t>, std::intptr_t{1}}.to_intptr_t() == 1);
CHECK(meta::value{std::in_place_type<std::uint8_t>, std::uint8_t{1}}.to_uint8() == 1u);
CHECK(meta::value{std::in_place_type<std::uint16_t>, std::uint16_t{1}}.to_uint16() == 1u);
CHECK(meta::value{std::in_place_type<std::uint32_t>, std::uint32_t{1}}.to_uint32() == 1u);
CHECK(meta::value{std::in_place_type<std::uint64_t>, std::uint64_t{1}}.to_uint64() == 1u);
CHECK(meta::value{std::in_place_type<std::size_t>, std::size_t{1}}.to_size_t() == 1u);
CHECK(meta::value{std::in_place_type<std::uintptr_t>, std::uintptr_t{1}}.to_uintptr_t() == 1u);
CHECK(meta::value{std::in_place_type<std::int8_t>, std::int8_t{1}}.is_int8());
CHECK(meta::value{std::in_place_type<std::int16_t>, std::int16_t{1}}.is_int16());
CHECK(meta::value{std::in_place_type<std::int32_t>, std::int32_t{1}}.is_int32());
CHECK(meta::value{std::in_place_type<std::int64_t>, std::int64_t{1}}.is_int64());
CHECK(meta::value{std::in_place_type<std::ptrdiff_t>, std::ptrdiff_t{1}}.is_ptrdiff_t());
CHECK(meta::value{std::in_place_type<std::intptr_t>, std::intptr_t{1}}.is_intptr_t());
CHECK(meta::value{std::in_place_type<std::uint8_t>, std::uint8_t{1}}.is_uint8());
CHECK(meta::value{std::in_place_type<std::uint16_t>, std::uint16_t{1}}.is_uint16());
CHECK(meta::value{std::in_place_type<std::uint32_t>, std::uint32_t{1}}.is_uint32());
CHECK(meta::value{std::in_place_type<std::uint64_t>, std::uint64_t{1}}.is_uint64());
CHECK(meta::value{std::in_place_type<std::size_t>, std::size_t{1}}.is_size_t());
CHECK(meta::value{std::in_place_type<std::uintptr_t>, std::uintptr_t{1}}.is_uintptr_t());
}
TEST_CASE("meta/value/fid") {
namespace meta = meta_hpp;
using namespace std::string_literals;
CHECK(meta::value{clazz{}}.fid() == meta::get_type_family_id<clazz>());
CHECK(meta::value{clazz2{}}.fid() == meta::get_type_family_id<clazz2>());
CHECK(meta::value{clazz{}}.fid() != meta::value{clazz2{}}.fid());
{
meta::value v = clazz{};
CHECK(v.fid() == meta::get_type_family_id<clazz>());
v = clazz2{};
CHECK(v.fid() == meta::get_type_family_id<clazz2>());
}
CHECK(meta::value{std::in_place_type<int>, 1}.fid() == meta::get_type_family_id<int>());
CHECK(meta::value{std::in_place_type<std::vector<int>>, {1,2,3,4}}.fid() == meta::get_type_family_id<std::vector<int>>());
}
TEST_CASE("meta/value/data") {
namespace meta = meta_hpp;
using namespace std::string_literals;
meta::value v{1};
CHECK(*static_cast<int*>(v.data()) == 1);
*static_cast<int*>(v.data()) = 2;
CHECK(v.to_int() == 2);
CHECK(*static_cast<const int*>(std::as_const(v).data()) == 2);
}
TEST_CASE("meta/value/equal") {
namespace meta = meta_hpp;
using namespace std::string_literals;
{
CHECK(meta::value{1} == 1);
CHECK(meta::value{1} != 0);
CHECK(meta::value{1} != "hello"s);
CHECK_FALSE(meta::value{1} != 1);
CHECK_FALSE(meta::value{1} == 0);
CHECK_FALSE(meta::value{1} == "hello"s);
}
{
CHECK(1 == meta::value{1});
CHECK(0 != meta::value{1});
CHECK("hello"s != meta::value{1});
CHECK_FALSE(1 != meta::value{1});
CHECK_FALSE(0 == meta::value{1});
CHECK_FALSE("hello"s == meta::value{1});
}
{
int i = 1;
CHECK(meta::value{1} == i);
CHECK(meta::value{1} == std::as_const(i));
CHECK_FALSE(meta::value{1} != i);
CHECK_FALSE(meta::value{1} != std::as_const(i));
CHECK(i == meta::value{1});
CHECK(std::as_const(i) == meta::value{1});
CHECK_FALSE(i != meta::value{1});
CHECK_FALSE(std::as_const(i) != meta::value{1});
}
{
CHECK(meta::value{1} == meta::value{1});
CHECK(meta::value{1} != meta::value{0});
CHECK(meta::value{1} != meta::value{"hello"s});
CHECK_FALSE(meta::value{1} != meta::value{1});
CHECK_FALSE(meta::value{1} == meta::value{0});
CHECK_FALSE(meta::value{1} == meta::value{"hello"s});
CHECK(meta::value{1} == meta::value{1});
CHECK(meta::value{0} != meta::value{1});
CHECK(meta::value{"hello"s} != meta::value{1});
CHECK_FALSE(meta::value{1} != meta::value{1});
CHECK_FALSE(meta::value{0} == meta::value{1});
CHECK_FALSE(meta::value{"hello"s} == meta::value{1});
}
{
CHECK(meta::value{std::in_place_type<int>, 1} == meta::value{1});
CHECK(meta::value{1} == meta::value{std::in_place_type<int>, 1});
using vi = std::vector<int>;
CHECK(meta::value{std::in_place_type<vi>, {1,2,3}} == meta::value{vi{1,2,3}});
CHECK(meta::value{vi{1,2,3}} == meta::value{std::in_place_type<vi>, {1,2,3}});
CHECK_FALSE(meta::value{std::in_place_type<vi>, {1,2,3}} == meta::value{vi{1,2,4}});
CHECK_FALSE(meta::value{vi{1,2,4}} == meta::value{std::in_place_type<vi>, {1,2,3}});
}
{
meta::value v1{&func};
meta::value v2{&func};
CHECK(v1 == v1);
CHECK_FALSE(v1 != v1);
CHECK(v1 == v2);
CHECK_FALSE(v1 != v2);
}
{
meta::value v1{clazz{}};
meta::value v2{clazz{}};
CHECK(v1 == v1);
CHECK_FALSE(v1 != v1);
CHECK(v1 != v2);
CHECK_FALSE(v1 == v2);
}
{
const int rv = 1;
meta::value v1{&rv};
meta::value v2{&rv};
CHECK(v1 == v1);
CHECK_FALSE(v1 != v1);
}
{
meta::value v1{"hello"s};
meta::value{1}.swap(v1);
CHECK(v1 == meta::value{1});
}
}
TEST_CASE("meta/value/swap") {
namespace meta = meta_hpp;
using namespace std::string_literals;
meta::value v1{"hello"s};
meta::value v2{42};
meta::swap(v1, v2);
CHECK(v1.fid() == meta::get_type_family_id<int>());
CHECK(v1.to_int() == 42);
CHECK(v2.fid() == meta::get_type_family_id<std::string>());
CHECK(v2.to_string() == "hello");
}