// Copyright 2015-2018 Hans Dembinski // // Distributed under the Boost Software License, Version 1.0. // (See accompanying file LICENSE_1_0.txt // or copy at http://www.boost.org/LICENSE_1_0.txt) #include #include #include #include #include #include #include #include #include #include #include "std_ostream.hpp" #include "throw_exception.hpp" #include "utility_axis.hpp" #include "utility_str.hpp" int main() { using namespace boost::histogram; BOOST_TEST(std::is_nothrow_move_constructible>::value); BOOST_TEST(std::is_nothrow_move_constructible>::value); BOOST_TEST(std::is_nothrow_move_assignable>::value); BOOST_TEST(std::is_nothrow_move_assignable>::value); // bad ctor { int x[2]; boost::ignore_unused(x); BOOST_TEST_THROWS(axis::category(x + 1, x), std::invalid_argument); } // value should return copy for arithmetic types and const reference otherwise { enum class Foo { foo }; BOOST_TEST_TRAIT_SAME(axis::traits::value_type>, std::string); BOOST_TEST_TRAIT_SAME(decltype(std::declval>().value(0)), const std::string&); BOOST_TEST_TRAIT_SAME(axis::traits::value_type>, const char*); BOOST_TEST_TRAIT_SAME(decltype(std::declval>().value(0)), const char*); BOOST_TEST_TRAIT_SAME(axis::traits::value_type>, Foo); BOOST_TEST_TRAIT_SAME(decltype(std::declval>().value(0)), Foo); BOOST_TEST_TRAIT_SAME(axis::traits::value_type>, int); BOOST_TEST_TRAIT_SAME(decltype(std::declval>().value(0)), int); } // empty axis::category { axis::category a; axis::category b(std::vector(0)); BOOST_TEST_EQ(a, b); BOOST_TEST_EQ(a.size(), 0); BOOST_TEST_EQ(a.index(-1), 0); BOOST_TEST_EQ(a.index(0), 0); BOOST_TEST_EQ(a.index(1), 0); } // axis::category { std::string A("A"), B("B"), C("C"), other; axis::category a({A, B, C}, "foo"); BOOST_TEST_EQ(a.metadata(), "foo"); BOOST_TEST_EQ(static_cast&>(a).metadata(), "foo"); a.metadata() = "bar"; BOOST_TEST_EQ(static_cast&>(a).metadata(), "bar"); BOOST_TEST_EQ(a.size(), 3); BOOST_TEST_EQ(a.index(A), 0); BOOST_TEST_EQ(a.index(B), 1); BOOST_TEST_EQ(a.index(C), 2); BOOST_TEST_EQ(a.index(other), 3); BOOST_TEST_EQ(a.value(0), A); BOOST_TEST_EQ(a.value(1), B); BOOST_TEST_EQ(a.value(2), C); BOOST_TEST_THROWS(a.value(3), std::out_of_range); BOOST_TEST_EQ(str(a), "category(\"A\", \"B\", \"C\", metadata=\"bar\", options=overflow)"); } // category: copy, move { using C = axis::category; C a({1, 2, 3}); C a2(a); BOOST_TEST_EQ(a2, a); C b; BOOST_TEST_NE(a, b); b = a; BOOST_TEST_EQ(a, b); b = C{{2, 1, 3}}; BOOST_TEST_NE(a, b); b = a; BOOST_TEST_EQ(a, b); C c = std::move(b); BOOST_TEST_EQ(c, a); C d; BOOST_TEST_NE(c, d); d = std::move(c); BOOST_TEST_EQ(d, a); } // category: copy, move { using C = axis::category; C a({"A", "B", "C"}, "foo"); C a2(a); BOOST_TEST_EQ(a2, a); C b; BOOST_TEST_NE(a, b); b = a; BOOST_TEST_EQ(a, b); b = C{{"B", "A", "C"}}; BOOST_TEST_NE(a, b); b = a; BOOST_TEST_EQ(a, b); C c = std::move(b); BOOST_TEST_EQ(c, a); C d; BOOST_TEST_NE(c, d); d = std::move(c); BOOST_TEST_EQ(d, a); } // axis::category with growth { axis::category a; BOOST_TEST_EQ(a.size(), 0); BOOST_TEST_EQ(a.update(5), std::make_pair(0, -1)); BOOST_TEST_EQ(a.size(), 1); BOOST_TEST_EQ(a.update(1), std::make_pair(1, -1)); BOOST_TEST_EQ(a.size(), 2); BOOST_TEST_EQ(a.update(10), std::make_pair(2, -1)); BOOST_TEST_EQ(a.size(), 3); BOOST_TEST_EQ(a.update(10), std::make_pair(2, 0)); BOOST_TEST_EQ(a.size(), 3); BOOST_TEST_EQ(str(a), "category(5, 1, 10, options=growth)"); } // iterators { test_axis_iterator(axis::category<>({3, 1, 2}, ""), 0, 3); test_axis_iterator(axis::category({"A", "B"}, ""), 0, 2); } return boost::report_errors(); }