1 // Copyright Louis Dionne 2013-2017
2 // Distributed under the Boost Software License, Version 1.0.
3 // (See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)
5 #ifndef BOOST_HANA_TEST_LAWS_SEARCHABLE_HPP
6 #define BOOST_HANA_TEST_LAWS_SEARCHABLE_HPP
8 #include <boost/hana/all.hpp>
9 #include <boost/hana/all_of.hpp>
10 #include <boost/hana/any.hpp>
11 #include <boost/hana/any_of.hpp>
12 #include <boost/hana/assert.hpp>
13 #include <boost/hana/at_key.hpp>
14 #include <boost/hana/bool.hpp>
15 #include <boost/hana/concat.hpp>
16 #include <boost/hana/concept/comparable.hpp>
17 #include <boost/hana/concept/searchable.hpp>
18 #include <boost/hana/concept/sequence.hpp>
19 #include <boost/hana/contains.hpp>
20 #include <boost/hana/core/is_a.hpp>
21 #include <boost/hana/core/when.hpp>
22 #include <boost/hana/equal.hpp>
23 #include <boost/hana/find.hpp>
24 #include <boost/hana/find_if.hpp>
25 #include <boost/hana/for_each.hpp>
26 #include <boost/hana/functional/always.hpp>
27 #include <boost/hana/functional/capture.hpp>
28 #include <boost/hana/functional/compose.hpp>
29 #include <boost/hana/functional/partial.hpp>
30 #include <boost/hana/is_disjoint.hpp>
31 #include <boost/hana/is_subset.hpp>
32 #include <boost/hana/lazy.hpp>
33 #include <boost/hana/none.hpp>
34 #include <boost/hana/none_of.hpp>
35 #include <boost/hana/not.hpp>
36 #include <boost/hana/optional.hpp>
37 #include <boost/hana/transform.hpp>
38 #include <boost/hana/tuple.hpp>
40 #include <laws/base.hpp>
41 #include <support/numeric.hpp>
44 namespace boost { namespace hana { namespace test {
45 template <typename S, typename = when<true>>
46 struct TestSearchable : TestSearchable<S, laws> {
47 using TestSearchable<S, laws>::TestSearchable;
51 struct TestSearchable<S, laws> {
52 template <typename Searchables, typename Keys>
53 TestSearchable(Searchables searchables, Keys keys) {
54 hana::for_each(searchables, [](auto xs) {
55 static_assert(Searchable<decltype(xs)>{}, "");
58 auto predicates = hana::concat(
59 hana::to_tuple(hana::transform(keys, equal.to)),
61 hana::always(false_c),
66 hana::for_each(searchables, hana::capture(searchables, keys, predicates)(
67 [](auto searchables, auto keys, auto predicates, auto xs) {
68 hana::for_each(predicates, hana::capture(xs)(
70 // any_of(xs, p) <=> !all_of(xs, negated p)
71 // <=> !none_of(xs, p)
73 hana::any_of(xs, p) ^iff^
74 hana::not_(hana::all_of(xs, hana::compose(not_, p)))
78 hana::any_of(xs, p) ^iff^
79 hana::not_(hana::none_of(xs, p))
83 // any(xs) <=> any_of(xs, id)
84 // all(xs) <=> all_of(xs, id)
85 // none(xs) <=> none_of(xs, id)
86 auto all_logicals = hana::all_of(xs, [](auto x) {
87 return hana::bool_c<hana::Logical<decltype(x)>::value>;
89 only_when_(all_logicals, hana::make_lazy([](auto xs) {
91 hana::any(xs) ^iff^ hana::any_of(xs, id)
95 hana::all(xs) ^iff^ hana::all_of(xs, hana::id)
99 hana::none(xs) ^iff^ hana::none_of(xs, hana::id)
103 // find_if(xs, always(false_c)) == nothing
104 BOOST_HANA_CONSTANT_CHECK(hana::equal(
105 hana::find_if(xs, hana::always(hana::false_c)),
109 hana::for_each(searchables, hana::capture(xs)([](auto xs, auto ys) {
110 // is_subset(xs, ys) <=> all_of(xs, [](auto x) { return contains(ys, x); })
112 hana::is_subset(xs, ys) ^iff^
113 hana::all_of(xs, hana::partial(hana::contains, ys))
116 // is_disjoint(xs, ys) <=> none_of(xs, [](auto x) { return contains(ys, x); })
118 hana::is_disjoint(xs, ys) ^iff^
119 hana::none_of(xs, hana::partial(hana::contains, ys))
123 hana::for_each(keys, hana::capture(xs)([](auto xs, auto key) {
124 // find(xs, x) == find_if(xs, [](auto y) { return y == x; })
125 BOOST_HANA_CHECK(hana::equal(
127 hana::find_if(xs, hana::equal.to(key))
130 // contains(xs, x) <=> any_of(xs, [](auto y) { return y == x; })
132 hana::contains(xs, key) ^iff^
133 hana::any_of(xs, hana::equal.to(key))
136 only_when_(hana::contains(xs, key),
137 hana::make_lazy([](auto xs, auto key) {
138 // at_key(xs, key) == find(xs, key).value()
139 BOOST_HANA_CHECK(hana::equal(
140 hana::at_key(xs, key),
141 hana::find(xs, key).value()
150 template <typename S>
151 struct TestSearchable<S, when<Sequence<S>::value>>
152 : TestSearchable<S, laws>
155 using x = _constant<i>;
160 struct undefined { };
162 template <typename Xs, typename Keys>
163 TestSearchable(Xs xs, Keys keys)
164 : TestSearchable<S, laws>{xs, keys}
166 constexpr auto list = make<S>;
168 BOOST_HANA_CONSTEXPR_LAMBDA auto is_even = [](auto x) {
172 auto c = numeric; // a minimal comparable
173 auto logical = numeric;
175 //////////////////////////////////////////////////////////////////
177 //////////////////////////////////////////////////////////////////
178 BOOST_HANA_CONSTANT_CHECK(
179 hana::not_(hana::any_of(list(), equal.to(x<9>{})))
182 BOOST_HANA_CONSTANT_CHECK(
183 hana::not_(hana::any_of(list(x<0>{}), equal.to(x<9>{})))
185 BOOST_HANA_CONSTANT_CHECK(
186 hana::any_of(list(x<0>{}), equal.to(x<0>{}))
188 BOOST_HANA_CONSTANT_CHECK(
189 hana::any_of(list(x<0>{}, invalid<1>{}), equal.to(x<0>{}))
191 BOOST_HANA_CONSTANT_CHECK(
192 hana::any_of(list(x<0>{}, invalid<1>{}, invalid<2>{}), equal.to(x<0>{}))
195 BOOST_HANA_CONSTANT_CHECK(
196 hana::not_(hana::any_of(list(x<0>{}, x<1>{}), equal.to(x<9>{})))
198 BOOST_HANA_CONSTANT_CHECK(
199 hana::any_of(list(x<0>{}, x<1>{}), equal.to(x<1>{}))
201 BOOST_HANA_CONSTANT_CHECK(
202 hana::any_of(list(x<0>{}, x<1>{}, invalid<2>{}), equal.to(x<1>{}))
204 BOOST_HANA_CONSTANT_CHECK(
205 hana::any_of(list(x<0>{}, x<1>{}, invalid<2>{}, invalid<3>{}), equal.to(x<1>{}))
208 BOOST_HANA_CONSTANT_CHECK(
209 hana::not_(hana::any_of(list(x<0>{}, x<1>{}, x<2>{}), equal.to(x<9>{})))
211 BOOST_HANA_CONSTANT_CHECK(
212 hana::any_of(list(x<0>{}, x<1>{}, x<2>{}), equal.to(x<2>{}))
214 BOOST_HANA_CONSTANT_CHECK(
215 hana::any_of(list(x<0>{}, x<1>{}, x<2>{}, nothing), equal.to(x<2>{}))
217 BOOST_HANA_CONSTANT_CHECK(
218 hana::any_of(list(x<0>{}, x<1>{}, x<2>{}, nothing, nothing), equal.to(x<2>{}))
221 BOOST_HANA_CONSTANT_CHECK(
222 hana::not_(hana::any_of(list(), invalid<>{}))
226 BOOST_HANA_CONSTEXPR_CHECK(
227 hana::any_of(list(c(0)), equal.to(c(0)))
229 BOOST_HANA_CONSTEXPR_CHECK(
230 hana::not_(hana::any_of(list(c(0)), equal.to(c(1))))
232 BOOST_HANA_CONSTEXPR_CHECK(
233 hana::not_(hana::any_of(list(1), is_even))
235 BOOST_HANA_CONSTEXPR_CHECK(
236 hana::any_of(list(2), is_even)
238 BOOST_HANA_CONSTEXPR_CHECK(
239 hana::any_of(list(1, 2), is_even)
241 BOOST_HANA_CONSTEXPR_CHECK(
242 hana::not_(hana::any_of(list(1, 3), is_even))
244 BOOST_HANA_CONSTEXPR_CHECK(
245 hana::any_of(list(1, 3, 4), is_even)
248 //////////////////////////////////////////////////////////////////
250 //////////////////////////////////////////////////////////////////
251 BOOST_HANA_CONSTANT_CHECK(hana::not_(hana::any(list())));
252 BOOST_HANA_CONSTEXPR_CHECK(hana::any(list(logical(true))));
253 BOOST_HANA_CONSTEXPR_CHECK(hana::not_(hana::any(list(logical(false)))));
256 //////////////////////////////////////////////////////////////////
258 //////////////////////////////////////////////////////////////////
259 BOOST_HANA_CONSTANT_CHECK(
260 all_of(list(), invalid<>{})
262 BOOST_HANA_CONSTEXPR_CHECK(
263 all_of(list(c(0)), equal.to(c(0)))
265 BOOST_HANA_CONSTEXPR_CHECK(
266 not_(all_of(list(c(0)), equal.to(c(1))))
268 BOOST_HANA_CONSTEXPR_CHECK(not_(
269 all_of(list(c(0), c(1)), equal.to(c(0)))
271 BOOST_HANA_CONSTEXPR_CHECK(
272 all_of(list(c(0), c(0)), equal.to(c(0)))
275 BOOST_HANA_CONSTEXPR_CHECK(not_(all_of(list(1), is_even)));
276 BOOST_HANA_CONSTEXPR_CHECK(all_of(list(2), is_even));
277 BOOST_HANA_CONSTEXPR_CHECK(all_of(list(2, 4), is_even));
278 BOOST_HANA_CONSTEXPR_CHECK(not_(all_of(list(1, 2), is_even)));
279 BOOST_HANA_CONSTEXPR_CHECK(not_(all_of(list(1, 3), is_even)));
280 BOOST_HANA_CONSTEXPR_CHECK(not_(all_of(list(1, 3, 4), is_even)));
283 //////////////////////////////////////////////////////////////////
285 //////////////////////////////////////////////////////////////////
286 BOOST_HANA_CONSTANT_CHECK(all(list()));
287 BOOST_HANA_CONSTEXPR_CHECK(all(list(logical(true))));
288 BOOST_HANA_CONSTEXPR_CHECK(not_(all(list(logical(false)))));
289 BOOST_HANA_CONSTEXPR_CHECK(all(list(logical(true), logical(true))));
290 BOOST_HANA_CONSTEXPR_CHECK(not_(all(list(logical(true), logical(false)))));
292 //////////////////////////////////////////////////////////////////
294 //////////////////////////////////////////////////////////////////
295 BOOST_HANA_CONSTANT_CHECK(none_of(list(), invalid<>{}));
296 BOOST_HANA_CONSTEXPR_CHECK(none_of(list(c(0)), equal.to(c(1))));
297 BOOST_HANA_CONSTEXPR_CHECK(not_(none_of(list(c(0)), equal.to(c(0)))));
299 BOOST_HANA_CONSTEXPR_CHECK(none_of(list(1), is_even));
300 BOOST_HANA_CONSTEXPR_CHECK(not_(none_of(list(2), is_even)));
301 BOOST_HANA_CONSTEXPR_CHECK(not_(none_of(list(1, 2), is_even)));
302 BOOST_HANA_CONSTEXPR_CHECK(none_of(list(1, 3), is_even));
303 BOOST_HANA_CONSTEXPR_CHECK(not_(none_of(list(1, 3, 4), is_even)));
306 //////////////////////////////////////////////////////////////////
308 //////////////////////////////////////////////////////////////////
309 BOOST_HANA_CONSTANT_CHECK(none(list()));
310 BOOST_HANA_CONSTEXPR_CHECK(none(list(logical(false))));
311 BOOST_HANA_CONSTEXPR_CHECK(not_(none(list(logical(true)))));
313 //////////////////////////////////////////////////////////////////
315 //////////////////////////////////////////////////////////////////
317 BOOST_HANA_CONSTANT_CHECK(equal(
318 find_if(list(), equal.to(x<9>{})),
322 BOOST_HANA_CONSTANT_CHECK(equal(
323 find_if(list(x<0>{}), equal.to(x<9>{})),
326 BOOST_HANA_CONSTANT_CHECK(equal(
327 find_if(list(x<0>{}), equal.to(x<0>{})),
330 BOOST_HANA_CONSTANT_CHECK(equal(
331 find_if(list(x<0>{}, invalid<1>{}), equal.to(x<0>{})),
334 BOOST_HANA_CONSTANT_CHECK(equal(
335 find_if(list(x<0>{}, invalid<1>{}, invalid<2>{}), equal.to(x<0>{})),
339 BOOST_HANA_CONSTANT_CHECK(equal(
340 find_if(list(x<0>{}, x<1>{}), equal.to(x<9>{})),
343 BOOST_HANA_CONSTANT_CHECK(equal(
344 find_if(list(x<0>{}, x<1>{}), equal.to(x<1>{})),
347 BOOST_HANA_CONSTANT_CHECK(equal(
348 find_if(list(x<0>{}, x<1>{}, invalid<2>{}), equal.to(x<1>{})),
351 BOOST_HANA_CONSTANT_CHECK(equal(
352 find_if(list(x<0>{}, x<1>{}, invalid<2>{}, invalid<3>{}), equal.to(x<1>{})),
356 BOOST_HANA_CONSTANT_CHECK(equal(
357 find_if(list(x<0>{}, x<1>{}, x<2>{}), equal.to(x<9>{})),
360 BOOST_HANA_CONSTANT_CHECK(equal(
361 find_if(list(x<0>{}, x<1>{}, x<2>{}), equal.to(x<2>{})),
364 BOOST_HANA_CONSTANT_CHECK(equal(
365 find_if(list(x<0>{}, x<1>{}, x<2>{}, nothing), equal.to(x<2>{})),
368 BOOST_HANA_CONSTANT_CHECK(equal(
369 find_if(list(x<0>{}, x<1>{}, x<2>{}, nothing, nothing), equal.to(x<2>{})),
373 // Make sure find_if works with an lvalue sequence. hana::tuple
374 // used to have a bug that broke this.
375 auto const const_lvalue = list(x<0>{}, x<1>{}, x<2>{});
376 BOOST_HANA_CONSTANT_CHECK(equal(
377 find_if(const_lvalue, equal.to(x<1>{})),
381 auto lvalue = list(x<0>{}, x<1>{}, x<2>{});
382 BOOST_HANA_CONSTANT_CHECK(equal(
383 find_if(lvalue, equal.to(x<1>{})),
388 //////////////////////////////////////////////////////////////////
390 //////////////////////////////////////////////////////////////////
392 BOOST_HANA_CONSTANT_CHECK(equal(
393 find(list(), invalid<>{}),
397 BOOST_HANA_CONSTANT_CHECK(equal(
398 find(list(x<0>{}), x<9>{}),
401 BOOST_HANA_CONSTANT_CHECK(equal(
402 find(list(x<0>{}), x<0>{}),
405 BOOST_HANA_CONSTANT_CHECK(equal(
406 find(list(x<0>{}, invalid<1>{}), x<0>{}),
409 BOOST_HANA_CONSTANT_CHECK(equal(
410 find(list(x<0>{}, invalid<1>{}, invalid<2>{}), x<0>{}),
414 BOOST_HANA_CONSTANT_CHECK(equal(
415 find(list(x<0>{}, x<1>{}), x<9>{}),
418 BOOST_HANA_CONSTANT_CHECK(equal(
419 find(list(x<0>{}, x<1>{}), x<1>{}),
422 BOOST_HANA_CONSTANT_CHECK(equal(
423 find(list(x<0>{}, x<1>{}, invalid<2>{}), x<1>{}),
426 BOOST_HANA_CONSTANT_CHECK(equal(
427 find(list(x<0>{}, x<1>{}, invalid<2>{}, invalid<3>{}), x<1>{}),
431 BOOST_HANA_CONSTANT_CHECK(equal(
432 find(list(x<0>{}, x<1>{}, x<2>{}), x<9>{}),
435 BOOST_HANA_CONSTANT_CHECK(equal(
436 find(list(x<0>{}, x<1>{}, x<2>{}), x<2>{}),
439 BOOST_HANA_CONSTANT_CHECK(equal(
440 find(list(x<0>{}, x<1>{}, x<2>{}, nothing), x<2>{}),
443 BOOST_HANA_CONSTANT_CHECK(equal(
444 find(list(x<0>{}, x<1>{}, x<2>{}, nothing, nothing), x<2>{}),
449 //////////////////////////////////////////////////////////////////
451 //////////////////////////////////////////////////////////////////
453 BOOST_HANA_CONSTANT_CHECK(equal(
454 at_key(list(x<0>{}), x<0>{}),
458 BOOST_HANA_CONSTANT_CHECK(equal(
459 at_key(list(x<0>{}, x<1>{}), x<0>{}),
462 BOOST_HANA_CONSTANT_CHECK(equal(
463 at_key(list(x<0>{}, x<1>{}), x<1>{}),
467 BOOST_HANA_CONSTANT_CHECK(equal(
468 at_key(list(x<0>{}, x<1>{}, x<2>{}), x<0>{}),
471 BOOST_HANA_CONSTANT_CHECK(equal(
472 at_key(list(x<0>{}, x<1>{}, x<2>{}), x<1>{}),
475 BOOST_HANA_CONSTANT_CHECK(equal(
476 at_key(list(x<0>{}, x<1>{}, x<2>{}), x<2>{}),
481 //////////////////////////////////////////////////////////////////
483 //////////////////////////////////////////////////////////////////
485 BOOST_HANA_CONSTANT_CHECK(not_(
486 contains(list(), undefined{})
489 BOOST_HANA_CONSTANT_CHECK(not_(
490 contains(list(x<0>{}), x<999>{})
493 BOOST_HANA_CONSTANT_CHECK(
494 contains(list(x<0>{}), x<0>{})
497 BOOST_HANA_CONSTANT_CHECK(
498 contains(list(x<0>{}, x<1>{}), x<1>{})
501 BOOST_HANA_CONSTANT_CHECK(
502 contains(list(x<0>{}, x<1>{}, x<3>{}), x<3>{})
505 BOOST_HANA_CONSTEXPR_CHECK(contains(list(c(0)), c(0)));
506 BOOST_HANA_CONSTEXPR_CHECK(not_(contains(list(c(0)), c(1))));
509 BOOST_HANA_CONSTANT_CHECK(
510 list(x<0>{}, x<1>{}) ^contains^ x<1>{}
514 //////////////////////////////////////////////////////////////////
516 //////////////////////////////////////////////////////////////////
517 BOOST_HANA_CONSTEXPR_CHECK(c(0) ^in^ list(c(0)));
518 BOOST_HANA_CONSTEXPR_CHECK(not_(c(1) ^in^ list(c(0))));
520 //////////////////////////////////////////////////////////////////
522 //////////////////////////////////////////////////////////////////
524 BOOST_HANA_CONSTANT_CHECK(is_subset(
525 list(), list(undefined{})
528 BOOST_HANA_CONSTANT_CHECK(not_(is_subset(
529 list(undefined{}), list()
532 BOOST_HANA_CONSTEXPR_CHECK(is_subset(
533 list(c(0)), list(c(0))
536 BOOST_HANA_CONSTEXPR_CHECK(is_subset(
537 list(c(0)), list(c(0), c(1))
540 BOOST_HANA_CONSTEXPR_CHECK(not_(is_subset(
541 list(c(0)), list(c(1))
545 BOOST_HANA_CONSTANT_CHECK(
546 list(x<0>{}) ^is_subset^ list(x<1>{}, x<0>{})
549 BOOST_HANA_CONSTANT_CHECK(
550 list(x<0>{}, x<1>{}) ^is_subset^ list(x<1>{}, x<0>{})
554 //////////////////////////////////////////////////////////////////
556 //////////////////////////////////////////////////////////////////
558 BOOST_HANA_CONSTANT_CHECK(is_disjoint(
563 BOOST_HANA_CONSTANT_CHECK(is_disjoint(
567 BOOST_HANA_CONSTANT_CHECK(is_disjoint(
572 BOOST_HANA_CONSTANT_CHECK(is_disjoint(
576 BOOST_HANA_CONSTANT_CHECK(is_disjoint(
580 BOOST_HANA_CONSTANT_CHECK(not_(is_disjoint(
586 BOOST_HANA_CONSTANT_CHECK(not_(is_disjoint(
587 list(x<0>{}, x<1>{}),
590 BOOST_HANA_CONSTANT_CHECK(not_(is_disjoint(
594 BOOST_HANA_CONSTANT_CHECK(not_(is_disjoint(
595 list(x<1>{}, x<0>{}),
598 BOOST_HANA_CONSTANT_CHECK(not_(is_disjoint(
599 list(x<0>{}, x<1>{}),
602 BOOST_HANA_CONSTANT_CHECK(is_disjoint(
603 list(x<0>{}, x<1>{}),
609 }}} // end namespace boost::hana::test
611 #endif // !BOOST_HANA_TEST_LAWS_SEARCHABLE_HPP