Files
meta.hpp/untests/meta_value_tests.cpp
2021-07-13 14:36:24 +07:00

201 lines
7.4 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());
SUBCASE("in_place") {
CHECK(meta::value{std::in_place_type<std::uint8_t>, std::uint8_t{1}}.is_uint8());
CHECK(meta::value{std::in_place_type<std::uint8_t&>, std::uint8_t{1}}.is_uint8());
CHECK(meta::value{std::in_place_type<const std::uint8_t>, std::uint8_t{1}}.is_uint8());
CHECK(meta::value{std::in_place_type<const std::uint8_t&>, std::uint8_t{1}}.is_uint8());
}
}
TEST_CASE("meta/value/fid") {
namespace meta = meta_hpp;
using namespace std::string_literals;
CHECK(meta::value{clazz{}}.has_type<clazz>());
CHECK(meta::value{clazz{}}.fid() == meta::get_family_id<clazz>());
CHECK(meta::value{clazz2{}}.has_type<clazz2>());
CHECK(meta::value{clazz2{}}.fid() == meta::get_family_id<clazz2>());
CHECK(meta::value{clazz{}}.fid() != meta::value{clazz2{}}.fid());
{
meta::value v = clazz{};
CHECK(v.fid() == meta::get_family_id<clazz>());
v = clazz2{};
CHECK(v.fid() == meta::get_family_id<clazz2>());
}
CHECK(meta::value{std::in_place_type<int>, 1}.fid() == meta::get_family_id<int>());
CHECK(meta::value{std::in_place_type<std::vector<int>>, {1,2,3,4}}.fid() == meta::get_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}.equals(1));
CHECK_FALSE(meta::value{1}.equals(0));
CHECK_FALSE(meta::value{1}.equals("hello"s));
}
{
int i = 1;
CHECK(meta::value{1}.equals(i));
CHECK(meta::value{1}.equals(std::as_const(i)));
CHECK_FALSE(meta::value{2}.equals(i));
CHECK_FALSE(meta::value{2}.equals(std::as_const(i)));
}
{
CHECK(meta::value{1}.equals(meta::value{1}));
CHECK_FALSE(meta::value{1}.equals(meta::value{0}));
CHECK_FALSE(meta::value{1}.equals(meta::value{"hello"s}));
CHECK(meta::value{1}.equals(meta::value{1}));
CHECK_FALSE(meta::value{0}.equals(meta::value{1}));
CHECK_FALSE(meta::value{"hello"s}.equals(meta::value{1}));
}
{
CHECK(meta::value{std::in_place_type<int>, 1}.equals(meta::value{1}));
CHECK(meta::value{1}.equals(meta::value{std::in_place_type<int>, 1}));
using vi = std::vector<int>;
CHECK(meta::value{std::in_place_type<vi>, {1,2,3}}.equals(meta::value{vi{1,2,3}}));
CHECK(meta::value{vi{1,2,3}}.equals(meta::value{std::in_place_type<vi>, {1,2,3}}));
CHECK_FALSE(meta::value{std::in_place_type<vi>, {1,2,3}}.equals(meta::value{vi{1,2,4}}));
CHECK_FALSE(meta::value{vi{1,2,4}}.equals(meta::value{std::in_place_type<vi>, {1,2,3}}));
}
{
meta::value v1{&func};
meta::value v2{&func};
CHECK(v1.equals(v1));
CHECK(v1.equals(v2));
}
{
meta::value v1{clazz{}};
meta::value v2{clazz{}};
CHECK(v1.equals(v1));
CHECK_FALSE(v1.equals(v2));
}
{
const int rv = 1;
meta::value v1{&rv};
meta::value v2{&rv};
CHECK(v1.equals(v1));
}
{
meta::value v1{"hello"s};
meta::value{1}.swap(v1);
CHECK(v1.equals(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_family_id<int>());
CHECK(v1.to_int() == 42);
CHECK(v2.fid() == meta::get_family_id<std::string>());
CHECK(v2.to_string() == "hello");
}