mirror of
https://github.com/TartanLlama/expected.git
synced 2025-08-03 19:04:29 +02:00
Emplace tests
This commit is contained in:
13
expected.hpp
13
expected.hpp
@@ -242,6 +242,9 @@ using expected_enable_from_other = detail::enable_if_t<
|
|||||||
|
|
||||||
/// \exclude
|
/// \exclude
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
struct no_init_t{};
|
||||||
|
static constexpr no_init_t no_init{};
|
||||||
|
|
||||||
// Implements the storage of the values, and ensures that the destructor is
|
// Implements the storage of the values, and ensures that the destructor is
|
||||||
// trivial if it can be.
|
// trivial if it can be.
|
||||||
//
|
//
|
||||||
@@ -252,6 +255,7 @@ template <class T, class E, bool = std::is_trivially_destructible<T>::value,
|
|||||||
bool = std::is_trivially_destructible<E>::value>
|
bool = std::is_trivially_destructible<E>::value>
|
||||||
struct expected_storage_base {
|
struct expected_storage_base {
|
||||||
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
||||||
|
constexpr expected_storage_base(no_init_t) : m_has_val(false) {}
|
||||||
|
|
||||||
template <class... Args,
|
template <class... Args,
|
||||||
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
||||||
@@ -297,6 +301,7 @@ struct expected_storage_base {
|
|||||||
// so the destructor of the `expected` can be trivial.
|
// so the destructor of the `expected` can be trivial.
|
||||||
template <class T, class E> struct expected_storage_base<T, E, true, true> {
|
template <class T, class E> struct expected_storage_base<T, E, true, true> {
|
||||||
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
||||||
|
constexpr expected_storage_base(no_init_t) : m_has_val(false) {}
|
||||||
|
|
||||||
template <class... Args,
|
template <class... Args,
|
||||||
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
||||||
@@ -335,6 +340,7 @@ template <class T, class E> struct expected_storage_base<T, E, true, true> {
|
|||||||
// T is trivial, E is not.
|
// T is trivial, E is not.
|
||||||
template <class T, class E> struct expected_storage_base<T, E, true, false> {
|
template <class T, class E> struct expected_storage_base<T, E, true, false> {
|
||||||
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
||||||
|
constexpr expected_storage_base(no_init_t) : m_has_val(false) {}
|
||||||
|
|
||||||
template <class... Args,
|
template <class... Args,
|
||||||
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
||||||
@@ -378,6 +384,7 @@ template <class T, class E> struct expected_storage_base<T, E, true, false> {
|
|||||||
// E is trivial, T is not.
|
// E is trivial, T is not.
|
||||||
template <class T, class E> struct expected_storage_base<T, E, false, true> {
|
template <class T, class E> struct expected_storage_base<T, E, false, true> {
|
||||||
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
constexpr expected_storage_base() : m_val(T{}), m_has_val(true) {}
|
||||||
|
constexpr expected_storage_base(no_init_t) : m_has_val(false) {}
|
||||||
|
|
||||||
template <class... Args,
|
template <class... Args,
|
||||||
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
detail::enable_if_t<std::is_constructible<T, Args &&...>::value> * =
|
||||||
@@ -581,7 +588,8 @@ struct expected_copy_base<T, E, false> : expected_operations_base<T, E> {
|
|||||||
using expected_operations_base<T, E>::expected_operations_base;
|
using expected_operations_base<T, E>::expected_operations_base;
|
||||||
|
|
||||||
expected_copy_base() = default;
|
expected_copy_base() = default;
|
||||||
expected_copy_base(const expected_copy_base &rhs) {
|
expected_copy_base(const expected_copy_base &rhs) :
|
||||||
|
expected_operations_base<T,E>(no_init) {
|
||||||
if (rhs.has_value()) {
|
if (rhs.has_value()) {
|
||||||
this->construct(rhs.get());
|
this->construct(rhs.get());
|
||||||
} else {
|
} else {
|
||||||
@@ -616,7 +624,8 @@ struct expected_move_base<T, E, false> : expected_copy_base<T, E> {
|
|||||||
expected_move_base(const expected_move_base &rhs) = default;
|
expected_move_base(const expected_move_base &rhs) = default;
|
||||||
|
|
||||||
expected_move_base(expected_move_base &&rhs) noexcept(
|
expected_move_base(expected_move_base &&rhs) noexcept(
|
||||||
std::is_nothrow_move_constructible<T>::value) {
|
std::is_nothrow_move_constructible<T>::value) :
|
||||||
|
expected_copy_base<T,E>(no_init) {
|
||||||
if (rhs.has_value()) {
|
if (rhs.has_value()) {
|
||||||
this->construct(std::move(rhs.get()));
|
this->construct(std::move(rhs.get()));
|
||||||
} else {
|
} else {
|
||||||
|
@@ -1,4 +1,50 @@
|
|||||||
#include "catch.hpp"
|
#include "catch.hpp"
|
||||||
#include "expected.hpp"
|
#include "expected.hpp"
|
||||||
|
#include <memory>
|
||||||
|
#include <vector>
|
||||||
|
#include <tuple>
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
struct takes_init_and_variadic {
|
||||||
|
std::vector<int> v;
|
||||||
|
std::tuple<int, int> t;
|
||||||
|
template <class... Args>
|
||||||
|
takes_init_and_variadic(std::initializer_list<int> l, Args &&... args)
|
||||||
|
: v(l), t(std::forward<Args>(args)...) {}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE("Emplace", "[emplace]") {
|
TEST_CASE("Emplace", "[emplace]") {
|
||||||
|
{
|
||||||
|
tl::expected<std::unique_ptr<int>,int> e;
|
||||||
|
e.emplace(new int{42});
|
||||||
|
REQUIRE(e);
|
||||||
|
REQUIRE(**e == 42);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
tl::expected<std::vector<int>,int> e;
|
||||||
|
e.emplace({0,1});
|
||||||
|
REQUIRE(e);
|
||||||
|
REQUIRE((*e)[0] == 0);
|
||||||
|
REQUIRE((*e)[1] == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
tl::expected<std::tuple<int,int>,int> e;
|
||||||
|
e.emplace(2,3);
|
||||||
|
REQUIRE(e);
|
||||||
|
REQUIRE(std::get<0>(*e) == 2);
|
||||||
|
REQUIRE(std::get<1>(*e) == 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
tl::expected<takes_init_and_variadic,int> e = tl::make_unexpected(0);
|
||||||
|
e.emplace({0,1}, 2, 3);
|
||||||
|
REQUIRE(e);
|
||||||
|
REQUIRE(e->v[0] == 0);
|
||||||
|
REQUIRE(e->v[1] == 1);
|
||||||
|
REQUIRE(std::get<0>(e->t) == 2);
|
||||||
|
REQUIRE(std::get<1>(e->t) == 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Reference in New Issue
Block a user