blob: 161b80877ea12f6480f3e74d7e35f13b29de71ec [file] [log] [blame]
William A. Kennington IIIe847ef82018-11-02 17:29:15 -07001#include <stdplus/handle/copyable.hpp>
Patrick Williamsd1984dd2023-05-10 16:12:44 -05002
3#include <optional>
William A. Kennington IIIe847ef82018-11-02 17:29:15 -07004#include <string>
5#include <utility>
6#include <vector>
7
Patrick Williamsd1984dd2023-05-10 16:12:44 -05008#include <gtest/gtest.h>
9
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070010namespace stdplus
11{
12namespace
13{
14
15static std::vector<int> reffed;
16static int stored_ref = 0;
17static std::vector<int> dropped;
18static int stored_drop = 0;
19
William A. Kennington III7910ae02019-11-11 12:13:25 -080020struct SimpleRef
21{
22 int operator()(const int& i) noexcept
23 {
24 reffed.push_back(i);
25 return i + 1;
26 }
27};
28
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070029int ref(const int& i)
30{
William A. Kennington III7910ae02019-11-11 12:13:25 -080031 return SimpleRef()(i);
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070032}
33
William A. Kennington III7910ae02019-11-11 12:13:25 -080034struct SimpleDrop
35{
36 void operator()(int&& i) noexcept
37 {
38 dropped.push_back(std::move(i));
39 }
40};
41
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070042void drop(int&& i)
43{
William A. Kennington III7910ae02019-11-11 12:13:25 -080044 SimpleDrop()(std::move(i));
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070045}
46
William A. Kennington III7910ae02019-11-11 12:13:25 -080047struct StoreRef
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070048{
William A. Kennington III7910ae02019-11-11 12:13:25 -080049 int operator()(const int& i, std::string&, int& si)
50 {
51 reffed.push_back(i);
52 // Make sure we can update the stored data
53 stored_ref = si++;
54 return i + 1;
55 }
56};
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070057
William A. Kennington III7910ae02019-11-11 12:13:25 -080058struct StoreDrop
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070059{
William A. Kennington III7910ae02019-11-11 12:13:25 -080060 void operator()(int&& i, std::string&, int& si)
61 {
62 dropped.push_back(std::move(i));
63 // Make sure we can update the stored data
64 stored_drop = si++;
65 }
66};
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070067
William A. Kennington III7910ae02019-11-11 12:13:25 -080068using SimpleHandleOld = Copyable<int>::Handle<drop, ref>;
69using SimpleHandle = Copyable<int>::HandleF<SimpleDrop, SimpleRef>;
70using StoreHandle =
71 Copyable<int, std::string, int>::HandleF<StoreDrop, StoreRef>;
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070072
William A. Kennington III1007be92019-11-11 12:18:37 -080073static_assert(std::is_nothrow_copy_constructible_v<SimpleHandle>);
74static_assert(std::is_nothrow_move_constructible_v<SimpleHandle>);
75static_assert(std::is_nothrow_move_assignable_v<SimpleHandle>);
76static_assert(std::is_nothrow_destructible_v<SimpleHandle>);
77static_assert(noexcept(std::declval<SimpleHandle>().reset()));
78static_assert(!std::is_nothrow_copy_constructible_v<StoreHandle>);
79// http://cplusplus.github.io/LWG/lwg-active.html#2116
80// static_assert(std::is_nothrow_move_constructible_v<StoreHandle>);
81static_assert(!std::is_nothrow_move_assignable_v<StoreHandle>);
82static_assert(!std::is_nothrow_destructible_v<StoreHandle>);
83static_assert(!noexcept(std::declval<StoreHandle>().reset()));
84
William A. Kennington IIIe847ef82018-11-02 17:29:15 -070085class CopyableHandleTest : public ::testing::Test
86{
87 protected:
88 void SetUp()
89 {
90 reffed.clear();
91 dropped.clear();
92 }
93
94 void TearDown()
95 {
96 EXPECT_TRUE(reffed.empty());
97 EXPECT_TRUE(dropped.empty());
98 }
99};
100
101TEST_F(CopyableHandleTest, EmptyNoStorage)
102{
William A. Kennington III7910ae02019-11-11 12:13:25 -0800103 SimpleHandleOld h(std::nullopt);
William A. Kennington IIIe847ef82018-11-02 17:29:15 -0700104 EXPECT_FALSE(h);
105 EXPECT_THROW(h.value(), std::bad_optional_access);
106 h.reset();
107 EXPECT_FALSE(h);
108 EXPECT_THROW(h.value(), std::bad_optional_access);
109}
110
111TEST_F(CopyableHandleTest, EmptyWithStorage)
112{
113 auto maybeV = std::nullopt;
114 StoreHandle h(maybeV, "str", 5);
115 EXPECT_FALSE(h);
116 EXPECT_THROW(h.value(), std::bad_optional_access);
117 h.reset(maybeV);
118 EXPECT_FALSE(h);
119 EXPECT_THROW(h.value(), std::bad_optional_access);
120}
121
122TEST_F(CopyableHandleTest, SimplePopulated)
123{
124 constexpr int expected = 3;
125 {
126 int val = expected;
127 SimpleHandle h(std::move(val));
128 EXPECT_TRUE(h);
129 EXPECT_EQ(expected, *h);
130 EXPECT_EQ(expected, h.value());
131 EXPECT_TRUE(dropped.empty());
132 }
133 EXPECT_EQ(std::vector{expected}, dropped);
134 dropped.clear();
135}
136
137TEST_F(CopyableHandleTest, OptionalPopulated)
138{
139 constexpr int expected = 3;
140 {
141 std::optional<int> maybeVal{expected};
142 SimpleHandle h(std::move(maybeVal));
143 EXPECT_TRUE(h);
144 EXPECT_EQ(expected, *h);
145 EXPECT_EQ(expected, h.value());
146 EXPECT_TRUE(dropped.empty());
147 }
148 EXPECT_TRUE(reffed.empty());
149 EXPECT_EQ(std::vector{expected}, dropped);
150 dropped.clear();
151 {
152 const std::optional<int> maybeVal{expected};
153 SimpleHandle h(maybeVal);
154 EXPECT_TRUE(h);
155 EXPECT_EQ(expected + 1, *h);
156 EXPECT_EQ(expected + 1, h.value());
157 EXPECT_EQ(std::vector{expected}, reffed);
158 reffed.clear();
159 EXPECT_TRUE(dropped.empty());
160 }
161 EXPECT_EQ(std::vector{expected + 1}, dropped);
162 dropped.clear();
163}
164
165TEST_F(CopyableHandleTest, SimplePopulatedWithStorage)
166{
167 constexpr int expected = 3;
168 {
169 StoreHandle h(expected, std::string{"str"}, 5);
170 EXPECT_TRUE(h);
171 EXPECT_EQ(expected + 1, *h);
172 EXPECT_EQ(expected + 1, h.value());
173 EXPECT_EQ(5, stored_ref);
174 EXPECT_EQ(std::vector{expected}, reffed);
175 reffed.clear();
176 EXPECT_TRUE(dropped.empty());
177 }
178 EXPECT_EQ(6, stored_drop);
179 EXPECT_EQ(std::vector{expected + 1}, dropped);
180 dropped.clear();
181}
182
183TEST_F(CopyableHandleTest, ResetPopulatedWithStorage)
184{
185 constexpr int expected = 3;
186 const std::string s{"str"};
187 StoreHandle h(int{expected}, s, 5);
188 EXPECT_TRUE(dropped.empty());
189 h.reset(std::nullopt);
190 EXPECT_FALSE(h);
191 EXPECT_THROW(h.value(), std::bad_optional_access);
192 EXPECT_EQ(5, stored_drop);
193 EXPECT_EQ(std::vector{expected}, dropped);
194 dropped.clear();
195}
196
197TEST_F(CopyableHandleTest, ResetNewPopulated)
198{
199 constexpr int expected = 3, expected2 = 10;
200 {
201 SimpleHandle h(int{expected});
202 EXPECT_TRUE(dropped.empty());
203 h.reset(int{expected2});
204 EXPECT_TRUE(h);
205 EXPECT_EQ(expected2, *h);
206 EXPECT_EQ(expected2, h.value());
207 EXPECT_EQ(std::vector{expected}, dropped);
208 dropped.clear();
209 }
210 EXPECT_EQ(std::vector{expected2}, dropped);
211 dropped.clear();
212}
213
214TEST_F(CopyableHandleTest, ResetCopyPopulated)
215{
216 constexpr int expected = 3, expected2 = 10;
217 {
218 SimpleHandle h(int{expected});
219 EXPECT_TRUE(reffed.empty());
220 EXPECT_TRUE(dropped.empty());
221 const std::optional<int> maybe2{expected2};
222 h.reset(maybe2);
223 EXPECT_TRUE(h);
224 EXPECT_EQ(expected2 + 1, *h);
225 EXPECT_EQ(expected2 + 1, h.value());
226 EXPECT_EQ(std::vector{expected2}, reffed);
227 reffed.clear();
228 EXPECT_EQ(std::vector{expected}, dropped);
229 dropped.clear();
230 }
231 EXPECT_EQ(std::vector{expected2 + 1}, dropped);
232 dropped.clear();
233}
234
235TEST_F(CopyableHandleTest, ResetCopyPopulatedWithStorage)
236{
237 constexpr int expected = 3, expected2 = 10;
238 {
239 StoreHandle h(int{expected}, "str", 5);
240 EXPECT_TRUE(dropped.empty());
241 h.reset(expected2);
242 EXPECT_TRUE(h);
243 EXPECT_EQ(expected2 + 1, *h);
244 EXPECT_EQ(expected2 + 1, h.value());
245 EXPECT_EQ(5, stored_ref);
246 EXPECT_EQ(std::vector{expected2}, reffed);
247 reffed.clear();
248 EXPECT_EQ(6, stored_drop);
249 EXPECT_EQ(std::vector{expected}, dropped);
250 dropped.clear();
251 }
252 EXPECT_EQ(7, stored_drop);
253 EXPECT_EQ(std::vector{expected2 + 1}, dropped);
254 dropped.clear();
255}
256
257TEST_F(CopyableHandleTest, MoveConstructWithStorage)
258{
259 constexpr int expected = 3;
260 StoreHandle h1(int{expected}, "str", 5);
261 {
262 StoreHandle h2(std::move(h1));
263 EXPECT_TRUE(dropped.empty());
264 EXPECT_FALSE(h1);
265 EXPECT_THROW(h1.value(), std::bad_optional_access);
266 EXPECT_TRUE(h2);
267 EXPECT_EQ(expected, *h2);
268 EXPECT_EQ(expected, h2.value());
269 }
270 EXPECT_EQ(5, stored_drop);
271 EXPECT_EQ(std::vector{expected}, dropped);
272 dropped.clear();
273}
274
275TEST_F(CopyableHandleTest, MoveAssignWithStorage)
276{
277 constexpr int expected = 3, expected2 = 10;
278 {
279 StoreHandle h1(int{expected}, "str", 5);
280 StoreHandle h2(int{expected2}, "str", 10);
281 EXPECT_TRUE(dropped.empty());
282
283 h2 = std::move(h1);
284 EXPECT_EQ(10, stored_drop);
285 EXPECT_EQ(std::vector{expected2}, dropped);
286 dropped.clear();
287 EXPECT_FALSE(h1);
288 EXPECT_THROW(h1.value(), std::bad_optional_access);
289 EXPECT_TRUE(h2);
290 EXPECT_EQ(expected, *h2);
291 EXPECT_EQ(expected, h2.value());
292 }
293 EXPECT_EQ(5, stored_drop);
294 EXPECT_EQ(std::vector{expected}, dropped);
295 dropped.clear();
296}
297
298TEST_F(CopyableHandleTest, CopyConstructSrcEmptyWithStorage)
299{
300 StoreHandle h1(std::nullopt, "str", 5);
301 StoreHandle h2(h1);
302}
303
304TEST_F(CopyableHandleTest, CopyConstructWithStorage)
305{
306 constexpr int expected = 3;
307 StoreHandle h1(int{expected}, "str", 5);
308 StoreHandle h2(h1);
309 EXPECT_EQ(5, stored_ref);
310 EXPECT_EQ(std::vector{expected}, reffed);
311 reffed.clear();
312
313 h1.reset();
314 EXPECT_EQ(5, stored_drop);
315 EXPECT_EQ(std::vector{expected}, dropped);
316 dropped.clear();
317 h2.reset();
318 EXPECT_EQ(6, stored_drop);
319 EXPECT_EQ(std::vector{expected + 1}, dropped);
320 dropped.clear();
321}
322
323TEST_F(CopyableHandleTest, CopyAssignBothEmptyWithStorage)
324{
325 StoreHandle h1(std::nullopt, "str", 5);
326 StoreHandle h2(std::nullopt, "str", 10);
327 h2 = h1;
328}
329
330TEST_F(CopyableHandleTest, CopyAssignSrcEmptyWithStorage)
331{
332 constexpr int expected = 3;
333 StoreHandle h1(std::nullopt, "str", 5);
334 StoreHandle h2(int{expected}, "str", 10);
335 h2 = h1;
336 EXPECT_EQ(10, stored_drop);
337 EXPECT_EQ(std::vector{expected}, dropped);
338 dropped.clear();
339}
340
341TEST_F(CopyableHandleTest, CopyAssignDstEmptyWithStorage)
342{
343 constexpr int expected = 3;
344 StoreHandle h1(int{expected}, "str", 5);
345 StoreHandle h2(std::nullopt, "str", 10);
346 h2 = h1;
347 EXPECT_EQ(5, stored_ref);
348 EXPECT_EQ(std::vector{expected}, reffed);
349 reffed.clear();
350
351 h1.reset();
352 EXPECT_EQ(5, stored_drop);
353 EXPECT_EQ(std::vector{expected}, dropped);
354 dropped.clear();
355 h2.reset();
356 EXPECT_EQ(6, stored_drop);
357 EXPECT_EQ(std::vector{expected + 1}, dropped);
358 dropped.clear();
359}
360
361TEST_F(CopyableHandleTest, CopyAssignWithStorage)
362{
363 constexpr int expected = 3, expected2 = 15;
364 StoreHandle h1(int{expected}, "str", 5);
365 StoreHandle h2(int{expected2}, "str", 10);
366 h2 = h1;
367 EXPECT_EQ(10, stored_drop);
368 EXPECT_EQ(std::vector{expected2}, dropped);
369 dropped.clear();
370 EXPECT_EQ(5, stored_ref);
371 EXPECT_EQ(std::vector{expected}, reffed);
372 reffed.clear();
373
374 h1.reset();
375 EXPECT_EQ(5, stored_drop);
376 EXPECT_EQ(std::vector{expected}, dropped);
377 dropped.clear();
378 h2.reset();
379 EXPECT_EQ(6, stored_drop);
380 EXPECT_EQ(std::vector{expected + 1}, dropped);
381 dropped.clear();
382}
383
384} // namespace
385} // namespace stdplus