blob: fa954196196b0ad899b5339a537049ea04f0634f [file] [log] [blame]
#include <stdplus/handle/copyable.hpp>
#include <optional>
#include <string>
#include <utility>
#include <vector>
#include <gtest/gtest.h>
namespace stdplus
{
namespace
{
static std::vector<int> reffed;
static int stored_ref = 0;
static std::vector<int> dropped;
static int stored_drop = 0;
struct SimpleRef
{
int operator()(const int& i) noexcept
{
reffed.push_back(i);
return i + 1;
}
};
int ref(const int& i)
{
return SimpleRef()(i);
}
struct SimpleDrop
{
void operator()(int&& i) noexcept
{
dropped.push_back(std::move(i));
}
};
void drop(int&& i)
{
SimpleDrop()(std::move(i));
}
struct StoreRef
{
int operator()(const int& i, std::string&, int& si)
{
reffed.push_back(i);
// Make sure we can update the stored data
stored_ref = si++;
return i + 1;
}
};
struct StoreDrop
{
void operator()(int&& i, std::string&, int& si)
{
dropped.push_back(std::move(i));
// Make sure we can update the stored data
stored_drop = si++;
}
};
using SimpleHandleOld = Copyable<int>::Handle<drop, ref>;
using SimpleHandle = Copyable<int>::HandleF<SimpleDrop, SimpleRef>;
using StoreHandle =
Copyable<int, std::string, int>::HandleF<StoreDrop, StoreRef>;
static_assert(std::is_nothrow_copy_constructible_v<SimpleHandle>);
static_assert(std::is_nothrow_move_constructible_v<SimpleHandle>);
static_assert(std::is_nothrow_move_assignable_v<SimpleHandle>);
static_assert(std::is_nothrow_destructible_v<SimpleHandle>);
static_assert(noexcept(std::declval<SimpleHandle>().reset()));
static_assert(!std::is_nothrow_copy_constructible_v<StoreHandle>);
// http://cplusplus.github.io/LWG/lwg-active.html#2116
// static_assert(std::is_nothrow_move_constructible_v<StoreHandle>);
static_assert(!std::is_nothrow_move_assignable_v<StoreHandle>);
static_assert(!std::is_nothrow_destructible_v<StoreHandle>);
static_assert(!noexcept(std::declval<StoreHandle>().reset()));
class CopyableHandleTest : public ::testing::Test
{
protected:
void SetUp()
{
reffed.clear();
dropped.clear();
}
void TearDown()
{
EXPECT_TRUE(reffed.empty());
EXPECT_TRUE(dropped.empty());
}
};
TEST_F(CopyableHandleTest, EmptyNoStorage)
{
SimpleHandleOld h(std::nullopt);
EXPECT_FALSE(h);
EXPECT_THROW(h.value(), std::bad_optional_access);
h.reset();
EXPECT_FALSE(h);
EXPECT_THROW(h.value(), std::bad_optional_access);
}
TEST_F(CopyableHandleTest, EmptyWithStorage)
{
auto maybeV = std::nullopt;
StoreHandle h(maybeV, "str", 5);
EXPECT_FALSE(h);
EXPECT_THROW(h.value(), std::bad_optional_access);
h.reset(maybeV);
EXPECT_FALSE(h);
EXPECT_THROW(h.value(), std::bad_optional_access);
}
TEST_F(CopyableHandleTest, SimplePopulated)
{
constexpr int expected = 3;
{
int val = expected;
SimpleHandle h(std::move(val));
EXPECT_TRUE(h);
EXPECT_EQ(expected, *h);
EXPECT_EQ(expected, h.value());
EXPECT_TRUE(dropped.empty());
}
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, OptionalPopulated)
{
constexpr int expected = 3;
{
std::optional<int> maybeVal{expected};
SimpleHandle h(std::move(maybeVal));
EXPECT_TRUE(h);
EXPECT_EQ(expected, *h);
EXPECT_EQ(expected, h.value());
EXPECT_TRUE(dropped.empty());
}
EXPECT_TRUE(reffed.empty());
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
{
const std::optional<int> maybeVal{expected};
SimpleHandle h(maybeVal);
EXPECT_TRUE(h);
EXPECT_EQ(expected + 1, *h);
EXPECT_EQ(expected + 1, h.value());
EXPECT_EQ(std::vector{expected}, reffed);
reffed.clear();
EXPECT_TRUE(dropped.empty());
}
EXPECT_EQ(std::vector{expected + 1}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, SimplePopulatedWithStorage)
{
constexpr int expected = 3;
{
StoreHandle h(expected, std::string{"str"}, 5);
EXPECT_TRUE(h);
EXPECT_EQ(expected + 1, *h);
EXPECT_EQ(expected + 1, h.value());
EXPECT_EQ(5, stored_ref);
EXPECT_EQ(std::vector{expected}, reffed);
reffed.clear();
EXPECT_TRUE(dropped.empty());
}
EXPECT_EQ(6, stored_drop);
EXPECT_EQ(std::vector{expected + 1}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, ResetPopulatedWithStorage)
{
constexpr int expected = 3;
const std::string s{"str"};
StoreHandle h(int{expected}, s, 5);
EXPECT_TRUE(dropped.empty());
h.reset(std::nullopt);
EXPECT_FALSE(h);
EXPECT_THROW(h.value(), std::bad_optional_access);
EXPECT_EQ(5, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, ResetNewPopulated)
{
constexpr int expected = 3, expected2 = 10;
{
SimpleHandle h(int{expected});
EXPECT_TRUE(dropped.empty());
h.reset(int{expected2});
EXPECT_TRUE(h);
EXPECT_EQ(expected2, *h);
EXPECT_EQ(expected2, h.value());
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
EXPECT_EQ(std::vector{expected2}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, ResetCopyPopulated)
{
constexpr int expected = 3, expected2 = 10;
{
SimpleHandle h(int{expected});
EXPECT_TRUE(reffed.empty());
EXPECT_TRUE(dropped.empty());
const std::optional<int> maybe2{expected2};
h.reset(maybe2);
EXPECT_TRUE(h);
EXPECT_EQ(expected2 + 1, *h);
EXPECT_EQ(expected2 + 1, h.value());
EXPECT_EQ(std::vector{expected2}, reffed);
reffed.clear();
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
EXPECT_EQ(std::vector{expected2 + 1}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, ResetCopyPopulatedWithStorage)
{
constexpr int expected = 3, expected2 = 10;
{
StoreHandle h(int{expected}, "str", 5);
EXPECT_TRUE(dropped.empty());
h.reset(expected2);
EXPECT_TRUE(h);
EXPECT_EQ(expected2 + 1, *h);
EXPECT_EQ(expected2 + 1, h.value());
EXPECT_EQ(5, stored_ref);
EXPECT_EQ(std::vector{expected2}, reffed);
reffed.clear();
EXPECT_EQ(6, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
EXPECT_EQ(7, stored_drop);
EXPECT_EQ(std::vector{expected2 + 1}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, MoveConstructWithStorage)
{
constexpr int expected = 3;
StoreHandle h1(int{expected}, "str", 5);
{
StoreHandle h2(std::move(h1));
EXPECT_TRUE(dropped.empty());
EXPECT_FALSE(h1);
// NOLINTNEXTLINE(clang-analyzer-cplusplus.Move)
EXPECT_THROW(h1.value(), std::bad_optional_access);
EXPECT_TRUE(h2);
EXPECT_EQ(expected, *h2);
EXPECT_EQ(expected, h2.value());
}
EXPECT_EQ(5, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, MoveAssignWithStorage)
{
constexpr int expected = 3, expected2 = 10;
{
StoreHandle h1(int{expected}, "str", 5);
StoreHandle h2(int{expected2}, "str", 10);
EXPECT_TRUE(dropped.empty());
h2 = std::move(h1);
EXPECT_EQ(10, stored_drop);
EXPECT_EQ(std::vector{expected2}, dropped);
dropped.clear();
EXPECT_FALSE(h1);
// NOLINTNEXTLINE(clang-analyzer-cplusplus.Move)
EXPECT_THROW(h1.value(), std::bad_optional_access);
EXPECT_TRUE(h2);
EXPECT_EQ(expected, *h2);
EXPECT_EQ(expected, h2.value());
}
EXPECT_EQ(5, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, CopyConstructSrcEmptyWithStorage)
{
StoreHandle h1(std::nullopt, "str", 5);
StoreHandle h2(h1);
}
TEST_F(CopyableHandleTest, CopyConstructWithStorage)
{
constexpr int expected = 3;
StoreHandle h1(int{expected}, "str", 5);
StoreHandle h2(h1);
EXPECT_EQ(5, stored_ref);
EXPECT_EQ(std::vector{expected}, reffed);
reffed.clear();
h1.reset();
EXPECT_EQ(5, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
h2.reset();
EXPECT_EQ(6, stored_drop);
EXPECT_EQ(std::vector{expected + 1}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, CopyAssignBothEmptyWithStorage)
{
StoreHandle h1(std::nullopt, "str", 5);
StoreHandle h2(std::nullopt, "str", 10);
h2 = h1;
}
TEST_F(CopyableHandleTest, CopyAssignSrcEmptyWithStorage)
{
constexpr int expected = 3;
StoreHandle h1(std::nullopt, "str", 5);
StoreHandle h2(int{expected}, "str", 10);
h2 = h1;
EXPECT_EQ(10, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, CopyAssignDstEmptyWithStorage)
{
constexpr int expected = 3;
StoreHandle h1(int{expected}, "str", 5);
StoreHandle h2(std::nullopt, "str", 10);
h2 = h1;
EXPECT_EQ(5, stored_ref);
EXPECT_EQ(std::vector{expected}, reffed);
reffed.clear();
h1.reset();
EXPECT_EQ(5, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
h2.reset();
EXPECT_EQ(6, stored_drop);
EXPECT_EQ(std::vector{expected + 1}, dropped);
dropped.clear();
}
TEST_F(CopyableHandleTest, CopyAssignWithStorage)
{
constexpr int expected = 3, expected2 = 15;
StoreHandle h1(int{expected}, "str", 5);
StoreHandle h2(int{expected2}, "str", 10);
h2 = h1;
EXPECT_EQ(10, stored_drop);
EXPECT_EQ(std::vector{expected2}, dropped);
dropped.clear();
EXPECT_EQ(5, stored_ref);
EXPECT_EQ(std::vector{expected}, reffed);
reffed.clear();
h1.reset();
EXPECT_EQ(5, stored_drop);
EXPECT_EQ(std::vector{expected}, dropped);
dropped.clear();
h2.reset();
EXPECT_EQ(6, stored_drop);
EXPECT_EQ(std::vector{expected + 1}, dropped);
dropped.clear();
}
} // namespace
} // namespace stdplus