#include #include "unit_test.hpp" #include #include #include #include #include #include #if TOML11_CPLUSPLUS_STANDARD_VERSION >= 201703L #include #endif using test_value_types = std::tuple< toml::basic_value, toml::basic_value, toml::basic_value, toml::basic_value >; namespace test { template std::basic_ostream& operator<<(std::basic_ostream& os, const std::vector& v) { os << "[ "; for(const auto& i : v) {os << i << ' ';} os << ']'; return os; } template std::basic_ostream& operator<<(std::basic_ostream& os, const std::deque& v) { os << "[ "; for(const auto& i : v) {os << i << ' ';} os << ']'; return os; } template std::basic_ostream& operator<<(std::basic_ostream& os, const std::list& v) { os << "[ "; for(const auto& i : v) {os << i << ' ';} os << ']'; return os; } template std::basic_ostream& operator<<(std::basic_ostream& os, const std::map& v) { os << "[ "; for(const auto& i : v) {os << '{' << i.first << ", " << i.second << "} ";} os << ']'; return os; } template std::basic_ostream& operator<<(std::basic_ostream& os, const std::unordered_map& v) { os << "[ "; for(const auto& i : v) {os << '{' << i.first << ", " << i.second << "} ";} os << ']'; return os; } } // test #define TOML11_TEST_FIND_OR_EXACT(toml_type, init_expr, opt_expr) \ { \ using namespace test; \ const toml::toml_type init init_expr ; \ const toml::toml_type opt opt_expr ; \ const value_type v{{"key1", value_type{{"key2", init}} }};\ BOOST_TEST(init != opt); \ BOOST_TEST(init == toml::find_or(v, "key1", "key2", opt));\ } \ /**/ BOOST_AUTO_TEST_CASE_TEMPLATE(test_find_or_exact, value_type, test_value_types) { TOML11_TEST_FIND_OR_EXACT(boolean, ( true), (false)) TOML11_TEST_FIND_OR_EXACT(integer, ( 42), ( 54)) TOML11_TEST_FIND_OR_EXACT(floating, ( 3.14), ( 2.71)) TOML11_TEST_FIND_OR_EXACT(string, ("foo"), ("bar")) TOML11_TEST_FIND_OR_EXACT(local_time, (12, 30, 45), (6, 0, 30)) TOML11_TEST_FIND_OR_EXACT(local_date, (2019, toml::month_t::Apr, 1), (1999, toml::month_t::Jan, 2)) TOML11_TEST_FIND_OR_EXACT(local_datetime, (toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45)), (toml::local_date(1999, toml::month_t::Jan, 2), toml::local_time( 6, 0, 30)) ) TOML11_TEST_FIND_OR_EXACT(offset_datetime, (toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45), toml::time_offset( 9, 0)), (toml::local_date(1999, toml::month_t::Jan, 2), toml::local_time( 6, 0, 30), toml::time_offset(-3, 0)) ) { const typename value_type::array_type init{1,2,3,4,5}; const typename value_type::array_type opt {6,7,8,9,10}; const value_type v{{"key", init}}; BOOST_TEST(init != opt); BOOST_TEST(init == toml::find_or(v, "key", opt)); } { const typename value_type::table_type init{{"key1", 42}, {"key2", "foo"}}; const typename value_type::table_type opt {{"key1", 54}, {"key2", "bar"}}; const value_type v{{"key", init}}; BOOST_TEST(init != opt); BOOST_TEST(init == toml::find_or(v, "key", opt)); } } #undef TOML11_TEST_FIND_OR_EXACT #define TOML11_TEST_FIND_OR_MOVE(toml_type, init_expr, opt_expr) \ { \ using namespace test; \ const toml::toml_type init init_expr ; \ toml::toml_type opt opt_expr ; \ value_type v{{"key1", value_type{{"key2", init}} }}; \ BOOST_TEST(init != opt); \ const auto moved = toml::find_or(std::move(v), "key1", "key2", std::move(opt));\ BOOST_TEST(init == moved); \ } \ /**/ BOOST_AUTO_TEST_CASE_TEMPLATE(test_find_or_move, value_type, test_value_types) { TOML11_TEST_FIND_OR_MOVE(boolean, ( true), (false)) TOML11_TEST_FIND_OR_MOVE(integer, ( 42), ( 54)) TOML11_TEST_FIND_OR_MOVE(floating, ( 3.14), ( 2.71)) TOML11_TEST_FIND_OR_MOVE(string, ("foo"), ("bar")) TOML11_TEST_FIND_OR_MOVE(local_time, (12, 30, 45), (6, 0, 30)) TOML11_TEST_FIND_OR_MOVE(local_date, (2019, toml::month_t::Apr, 1), (1999, toml::month_t::Jan, 2)) TOML11_TEST_FIND_OR_MOVE(local_datetime, (toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45)), (toml::local_date(1999, toml::month_t::Jan, 2), toml::local_time( 6, 0, 30)) ) TOML11_TEST_FIND_OR_MOVE(offset_datetime, (toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45), toml::time_offset( 9, 0)), (toml::local_date(1999, toml::month_t::Jan, 2), toml::local_time( 6, 0, 30), toml::time_offset(-3, 0)) ) { typename value_type::array_type init{1,2,3,4,5}; typename value_type::array_type opt {6,7,8,9,10}; value_type v{{"key", init}}; BOOST_TEST(init != opt); const auto moved = toml::find_or(std::move(v), "key", std::move(opt)); BOOST_TEST(init == moved); } { typename value_type::table_type init{{"key1", 42}, {"key2", "foo"}}; typename value_type::table_type opt {{"key1", 54}, {"key2", "bar"}}; value_type v{{"key", init}}; BOOST_TEST(init != opt); const auto moved = toml::find_or(std::move(v), "key", std::move(opt)); BOOST_TEST(init == moved); } } #undef TOML11_TEST_FIND_OR_MOVE #define TOML11_TEST_FIND_OR_MODIFY(toml_type, init_expr, opt_expr)\ { \ using namespace test; \ const toml::toml_type init init_expr ; \ toml::toml_type opt1 opt_expr ; \ toml::toml_type opt2 opt_expr ; \ value_type v{{"key1", value_type{{"key2", init}} }}; \ BOOST_TEST(init != opt1); \ toml::find_or(v, "key1", "key2", opt2) = opt1; \ BOOST_TEST(opt1 == toml::find(v, "key1", "key2"));\ } \ /**/ BOOST_AUTO_TEST_CASE_TEMPLATE(test_find_or_modify, value_type, test_value_types) { TOML11_TEST_FIND_OR_MODIFY(boolean, ( true), (false)) TOML11_TEST_FIND_OR_MODIFY(integer, ( 42), ( 54)) TOML11_TEST_FIND_OR_MODIFY(floating, ( 3.14), ( 2.71)) TOML11_TEST_FIND_OR_MODIFY(string, ("foo"), ("bar")) TOML11_TEST_FIND_OR_MODIFY(local_time, (12, 30, 45), (6, 0, 30)) TOML11_TEST_FIND_OR_MODIFY(local_date, (2019, toml::month_t::Apr, 1), (1999, toml::month_t::Jan, 2)) TOML11_TEST_FIND_OR_MODIFY(local_datetime, (toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45)), (toml::local_date(1999, toml::month_t::Jan, 2), toml::local_time( 6, 0, 30)) ) TOML11_TEST_FIND_OR_MODIFY(offset_datetime, (toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45), toml::time_offset( 9, 0)), (toml::local_date(1999, toml::month_t::Jan, 2), toml::local_time( 6, 0, 30), toml::time_offset(-3, 0)) ) { typename value_type::array_type init{1,2,3,4,5}; typename value_type::array_type opt1{6,7,8,9,10}; typename value_type::array_type opt2{6,7,8,9,10}; BOOST_TEST(init != opt1); value_type v{{"key", init}}; toml::find_or(v, "key", opt2) = opt1; BOOST_TEST(opt1 == toml::find(v, "key")); } { typename value_type::table_type init{{"key1", 42}, {"key2", "foo"}}; typename value_type::table_type opt1{{"key1", 54}, {"key2", "bar"}}; typename value_type::table_type opt2{{"key1", 54}, {"key2", "bar"}}; value_type v{{"key", init}}; BOOST_TEST(init != opt1); toml::find_or(v, "key", opt2) = opt1; BOOST_TEST(opt1 == toml::find(v, "key")); } } #undef TOML11_TEST_FIND_OR_MODIFY #define TOML11_TEST_FIND_OR_FALLBACK(init_type, opt_type) \ { \ using namespace test; \ value_type v1{{"key1", value_type{{"key3", "foo"}}}}; \ BOOST_TEST(opt_type == toml::find_or(v1, "key1", "key2", opt_type));\ value_type v2{{"key1", "foo"}}; \ BOOST_TEST(opt_type == toml::find_or(v2, "key1", "key3", opt_type));\ } \ /**/ BOOST_AUTO_TEST_CASE_TEMPLATE(test_find_or_fallback, value_type, test_value_types) { const toml::boolean boolean (true); const toml::integer integer (42); const toml::floating floating (3.14); const toml::string string ("foo"); const toml::local_time local_time (12, 30, 45); const toml::local_date local_date (2019, toml::month_t::Apr, 1); const toml::local_datetime local_datetime ( toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45)); const toml::offset_datetime offset_datetime( toml::local_date(2019, toml::month_t::Apr, 1), toml::local_time(12, 30, 45), toml::time_offset( 9, 0)); using array_type = typename value_type::array_type; using table_type = typename value_type::table_type; const array_type array{1, 2, 3, 4, 5}; const table_type table{{"key1", 42}, {"key2", "foo"}}; TOML11_TEST_FIND_OR_FALLBACK(boolean, integer ); TOML11_TEST_FIND_OR_FALLBACK(boolean, floating ); TOML11_TEST_FIND_OR_FALLBACK(boolean, string ); TOML11_TEST_FIND_OR_FALLBACK(boolean, local_time ); TOML11_TEST_FIND_OR_FALLBACK(boolean, local_date ); TOML11_TEST_FIND_OR_FALLBACK(boolean, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(boolean, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(boolean, array ); TOML11_TEST_FIND_OR_FALLBACK(boolean, table ); TOML11_TEST_FIND_OR_FALLBACK(integer, boolean ); TOML11_TEST_FIND_OR_FALLBACK(integer, floating ); TOML11_TEST_FIND_OR_FALLBACK(integer, string ); TOML11_TEST_FIND_OR_FALLBACK(integer, local_time ); TOML11_TEST_FIND_OR_FALLBACK(integer, local_date ); TOML11_TEST_FIND_OR_FALLBACK(integer, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(integer, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(integer, array ); TOML11_TEST_FIND_OR_FALLBACK(integer, table ); TOML11_TEST_FIND_OR_FALLBACK(floating, boolean ); TOML11_TEST_FIND_OR_FALLBACK(floating, integer ); TOML11_TEST_FIND_OR_FALLBACK(floating, string ); TOML11_TEST_FIND_OR_FALLBACK(floating, local_time ); TOML11_TEST_FIND_OR_FALLBACK(floating, local_date ); TOML11_TEST_FIND_OR_FALLBACK(floating, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(floating, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(floating, array ); TOML11_TEST_FIND_OR_FALLBACK(floating, table ); TOML11_TEST_FIND_OR_FALLBACK(string, boolean ); TOML11_TEST_FIND_OR_FALLBACK(string, integer ); TOML11_TEST_FIND_OR_FALLBACK(string, floating ); TOML11_TEST_FIND_OR_FALLBACK(string, local_time ); TOML11_TEST_FIND_OR_FALLBACK(string, local_date ); TOML11_TEST_FIND_OR_FALLBACK(string, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(string, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(string, array ); TOML11_TEST_FIND_OR_FALLBACK(string, table ); TOML11_TEST_FIND_OR_FALLBACK(local_time, boolean ); TOML11_TEST_FIND_OR_FALLBACK(local_time, integer ); TOML11_TEST_FIND_OR_FALLBACK(local_time, floating ); TOML11_TEST_FIND_OR_FALLBACK(local_time, string ); TOML11_TEST_FIND_OR_FALLBACK(local_time, local_date ); TOML11_TEST_FIND_OR_FALLBACK(local_time, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(local_time, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(local_time, array ); TOML11_TEST_FIND_OR_FALLBACK(local_time, table ); TOML11_TEST_FIND_OR_FALLBACK(local_date, boolean ); TOML11_TEST_FIND_OR_FALLBACK(local_date, integer ); TOML11_TEST_FIND_OR_FALLBACK(local_date, floating ); TOML11_TEST_FIND_OR_FALLBACK(local_date, string ); TOML11_TEST_FIND_OR_FALLBACK(local_date, local_time ); TOML11_TEST_FIND_OR_FALLBACK(local_date, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(local_date, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(local_date, array ); TOML11_TEST_FIND_OR_FALLBACK(local_date, table ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, boolean ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, integer ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, floating ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, string ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, local_time ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, local_date ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, array ); TOML11_TEST_FIND_OR_FALLBACK(local_datetime, table ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, boolean ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, integer ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, floating ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, string ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, local_time ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, local_date ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, array ); TOML11_TEST_FIND_OR_FALLBACK(offset_datetime, table ); TOML11_TEST_FIND_OR_FALLBACK(array, boolean ); TOML11_TEST_FIND_OR_FALLBACK(array, integer ); TOML11_TEST_FIND_OR_FALLBACK(array, floating ); TOML11_TEST_FIND_OR_FALLBACK(array, string ); TOML11_TEST_FIND_OR_FALLBACK(array, local_time ); TOML11_TEST_FIND_OR_FALLBACK(array, local_date ); TOML11_TEST_FIND_OR_FALLBACK(array, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(array, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(array, table ); TOML11_TEST_FIND_OR_FALLBACK(table, boolean ); TOML11_TEST_FIND_OR_FALLBACK(table, integer ); TOML11_TEST_FIND_OR_FALLBACK(table, floating ); TOML11_TEST_FIND_OR_FALLBACK(table, string ); TOML11_TEST_FIND_OR_FALLBACK(table, local_time ); TOML11_TEST_FIND_OR_FALLBACK(table, local_date ); TOML11_TEST_FIND_OR_FALLBACK(table, local_datetime ); TOML11_TEST_FIND_OR_FALLBACK(table, offset_datetime); TOML11_TEST_FIND_OR_FALLBACK(table, array ); } #undef TOML11_TEST_FIND_OR_FALLBACK struct move_only_type { explicit move_only_type(const std::string& n): name_(n) {} void from_toml(const toml::value& v) { this->name_ = toml::find(v, "name"); return; } move_only_type(): name_("default"){} ~move_only_type() = default; move_only_type(move_only_type&&) = default; move_only_type& operator=(move_only_type&&) = default; move_only_type(const move_only_type&) = delete; move_only_type& operator=(const move_only_type&) = delete; bool operator==(const move_only_type& other) const noexcept {return this->name_ == other.name_;} bool operator!=(const move_only_type& other) const noexcept {return this->name_ != other.name_;} bool operator< (const move_only_type& other) const noexcept {return this->name_ < other.name_;} bool operator<=(const move_only_type& other) const noexcept {return this->name_ <= other.name_;} bool operator> (const move_only_type& other) const noexcept {return this->name_ > other.name_;} bool operator>=(const move_only_type& other) const noexcept {return this->name_ >= other.name_;} std::string name_; }; std::ostream& operator<<(std::ostream& os, const move_only_type& mot) { os << mot.name_; return os; } BOOST_AUTO_TEST_CASE_TEMPLATE(test_find_or_move_only, value_type, test_value_types) { const move_only_type ref("reference"); move_only_type opt("optional"); { const value_type v{{"key1", value_type{{"key2", value_type{{"name", "reference"}} }} }}; BOOST_TEST(ref == toml::find_or(v, "key1", "key2", std::move(opt))); } }