// Copyright 2018-2022 Emil Dotchevski and Reverge Studios, Inc. // 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) #ifdef BOOST_LEAF_TEST_SINGLE_HEADER # include "leaf.hpp" #else # include # include # include #endif #include "_test_ec.hpp" #include "lightweight_test.hpp" namespace leaf = boost::leaf; template struct info { int value; }; enum class my_error_code { ok, error1, error2, error3 }; struct e_my_error_code { my_error_code value; }; #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR struct e_std_error_code { std::error_code value; }; #endif template leaf::result f( my_error_code ec ) { if( ec==my_error_code::ok ) return R(42); else return leaf::new_error(ec, e_my_error_code{ec}, info<1>{1}, info<2>{2}, info<3>{3}); } #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR template leaf::result f_errc( Errc ec ) { return leaf::new_error(make_error_code(ec), info<1>{1}, info<2>{2}, info<3>{3}); } template leaf::result f_errc_wrapped( Errc ec ) { return leaf::new_error(e_std_error_code{make_error_code(ec)}, info<1>{1}, info<2>{2}, info<3>{3}); } #endif struct move_only { explicit move_only( int val ): value(val) { } int value; #ifndef BOOST_LEAF_NO_CXX11_REF_QUALIFIERS move_only( move_only const & ) = delete; move_only( move_only && ) = default; #endif }; int main() { // void, try_handle_all (success) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::ok)); c = answer; return { }; }, [&c] { BOOST_TEST_EQ(c, 0); c = 1; } ); BOOST_TEST_EQ(c, 42); } // void, try_handle_all (failure) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); c = answer; return { }; }, [&c]( my_error_code ec, info<1> const & x,info<2> y ) { BOOST_TEST(ec==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); BOOST_TEST_EQ(c, 0); c = 1; }, [&c]() { BOOST_TEST_EQ(c, 0); c = 2; } ); BOOST_TEST_EQ(c, 1); } #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR // void, try_handle_all (failure), match cond_x (single enum value) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f_errc(errc_a::a0)); c = answer; return { }; }, [&c]( leaf::match, cond_x::x11> ) { BOOST_TEST_EQ(c, 0); c = 1; }, [&c]( leaf::match, cond_x::x00> ec, info<1> const & x, info<2> y ) { BOOST_TEST_EQ(ec.matched, make_error_code(errc_a::a0)); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); BOOST_TEST_EQ(c, 0); c = 2; }, [&c]() { BOOST_TEST_EQ(c, 0); c = 3; } ); BOOST_TEST_EQ(c, 2); } #endif #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR // void, try_handle_all (failure), match cond_x (wrapped std::error_code) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f_errc_wrapped(errc_a::a0)); c = answer; return { }; }, [&c]( leaf::match_value, cond_x::x11> ) { BOOST_TEST_EQ(c, 0); c = 1; }, [&c]( leaf::match_value, cond_x::x00> ec, info<1> const & x, info<2> y ) { BOOST_TEST_EQ(ec.matched.value, make_error_code(errc_a::a0)); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); BOOST_TEST_EQ(c, 0); c = 2; }, [&c]() { BOOST_TEST_EQ(c, 0); c = 3; } ); BOOST_TEST_EQ(c, 2); } #endif // void, try_handle_all (failure), match enum (single enum value) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); c = answer; return { }; }, [&c]( leaf::match ) { BOOST_TEST_EQ(c, 0); c = 1; }, [&c]( leaf::match ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); BOOST_TEST_EQ(c, 0); c = 2; }, [&c]() { BOOST_TEST_EQ(c, 0); c = 3; } ); BOOST_TEST_EQ(c, 2); } // void, try_handle_all (failure), match enum (multiple enum values) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); c = answer; return { }; }, [&c]( leaf::match ) { BOOST_TEST_EQ(c, 0); c = 1; }, [&c]( leaf::match ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); BOOST_TEST_EQ(c, 0); c = 2; }, [&c]() { BOOST_TEST_EQ(c, 0); c = 3; } ); BOOST_TEST_EQ(c, 2); } // void, try_handle_all (failure), match value (single value) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); c = answer; return { }; }, [&c]( leaf::match_value ) { BOOST_TEST_EQ(c, 0); c = 1; }, [&c]( leaf::match_value ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched.value==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); BOOST_TEST_EQ(c, 0); c = 2; }, [&c]() { BOOST_TEST_EQ(c, 0); c = 3; } ); BOOST_TEST_EQ(c, 2); } // void, try_handle_all (failure), match value (multiple values) { int c=0; leaf::try_handle_all( [&c]() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); c = answer; return { }; }, [&c]( leaf::match_value ) { BOOST_TEST_EQ(c, 0); c = 1; }, [&c]( leaf::match_value ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched.value==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); BOOST_TEST_EQ(c, 0); c = 2; }, [&c]() { BOOST_TEST_EQ(c, 0); c = 3; } ); BOOST_TEST_EQ(c, 2); } ////////////////////////////////////// // int, try_handle_all (success) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::ok)); return answer; }, [] { return 1; } ); BOOST_TEST_EQ(r, 42); } // int, try_handle_all (failure) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return answer; }, []( my_error_code ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return 1; }, [] { return 2; } ); BOOST_TEST_EQ(r, 1); } #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR // int, try_handle_all (failure), match cond_x (single enum value) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f_errc(errc_a::a0)); return answer; }, []( leaf::match, cond_x::x11> ) { return 1; }, []( leaf::match, cond_x::x00> ec, info<1> const & x, info<2> y ) { BOOST_TEST_EQ(ec.matched, make_error_code(errc_a::a0)); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return 2; }, [] { return 3; } ); BOOST_TEST_EQ(r, 2); } #endif #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR // int, try_handle_all (failure), match cond_x (wrapped std::error_code) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f_errc_wrapped(errc_a::a0)); return answer; }, []( leaf::match, cond_x::x11> ) { return 1; }, []( leaf::match_value, cond_x::x00> ec, info<1> const & x, info<2> y ) { BOOST_TEST_EQ(ec.matched.value, make_error_code(errc_a::a0)); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return 2; }, [] { return 3; } ); BOOST_TEST_EQ(r, 2); } #endif // int, try_handle_all (failure), match enum (single enum value) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return answer; }, []( leaf::match ) { return 1; }, []( leaf::match ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return 2; }, [] { return 3; } ); BOOST_TEST_EQ(r, 2); } // int, try_handle_all (failure), match enum (multiple enum values) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return answer; }, []( leaf::match ) { return 1; }, []( leaf::match ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return 2; }, [] { return 3; } ); BOOST_TEST_EQ(r, 2); } // int, try_handle_all (failure), match value (single value) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return answer; }, []( leaf::match_value ) { return 1; }, []( leaf::match_value ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched.value==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return 2; }, [] { return 3; } ); BOOST_TEST_EQ(r, 2); } // int, try_handle_all (failure), match value (multiple values) { int r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return answer; }, []( leaf::match_value ) { return 1; }, []( leaf::match_value ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched.value==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return 2; }, [] { return 3; } ); BOOST_TEST_EQ(r, 2); } ////////////////////////////////////// // move_only, try_handle_all (success) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::ok)); return std::move(answer); }, [] { return move_only(1); } ); BOOST_TEST_EQ(r.value, 42); } // move_only, try_handle_all (failure) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return std::move(answer); }, []( my_error_code ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return move_only(1); }, [] { return move_only(2); } ); BOOST_TEST_EQ(r.value, 1); } #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR // move_only, try_handle_all (failure), match cond_x (single enum value) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f_errc(errc_a::a0)); return std::move(answer); }, []( leaf::match, cond_x::x11> ) { return move_only(1); }, []( leaf::match, cond_x::x00> ec, info<1> const & x, info<2> y ) { BOOST_TEST_EQ(ec.matched, make_error_code(errc_a::a0)); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return move_only(2); }, [] { return move_only(3); } ); BOOST_TEST_EQ(r.value, 2); } #endif #if BOOST_LEAF_CFG_STD_SYSTEM_ERROR // move_only, try_handle_all (failure), match cond_x (wrapped std::error_code) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f_errc_wrapped(errc_a::a0)); return std::move(answer); }, []( leaf::match, cond_x::x11> ) { return move_only(1); }, []( leaf::match_value, cond_x::x00> ec, info<1> const & x, info<2> y ) { BOOST_TEST_EQ(ec.matched.value, make_error_code(errc_a::a0)); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return move_only(2); }, [] { return move_only(3); } ); BOOST_TEST_EQ(r.value, 2); } #endif // move_only, try_handle_all (failure), match enum (single enum value) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return std::move(answer); }, []( leaf::match ) { return move_only(1); }, []( leaf::match ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return move_only(2); }, [] { return move_only(3); } ); BOOST_TEST_EQ(r.value, 2); } // move_only, try_handle_all (failure), match enum (multiple enum values) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return std::move(answer); }, []( leaf::match ) { return move_only(1); }, []( leaf::match ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return move_only(2); }, [] { return move_only(3); } ); BOOST_TEST_EQ(r.value, 2); } // move_only, try_handle_all (failure), match value (single value) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return std::move(answer); }, []( leaf::match_value ) { return move_only(1); }, []( leaf::match_value ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched.value==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return move_only(2); }, [] { return move_only(3); } ); BOOST_TEST_EQ(r.value, 2); } // move_only, try_handle_all (failure), match value (multiple values) { move_only r = leaf::try_handle_all( []() -> leaf::result { BOOST_LEAF_AUTO(answer, f(my_error_code::error1)); return std::move(answer); }, []( leaf::match_value ) { return move_only(1); }, []( leaf::match_value ec, info<1> const & x, info<2> y ) { BOOST_TEST(ec.matched.value==my_error_code::error1); BOOST_TEST_EQ(x.value, 1); BOOST_TEST_EQ(y.value, 2); return move_only(2); }, [] { return move_only(3); } ); BOOST_TEST_EQ(r.value, 2); } return boost::report_errors(); }