// Test constructor body throwing (in inheritance tree). #include "../aux_/oteststream.hpp" #include #include #include #include #include #include #include boost::contract::aux::test::oteststream out; struct c { void invariant() const { out << "c::inv" << std::endl; } static void static_invariant() { out << "c::static_inv" << std::endl; } struct e {}; virtual void f(boost::contract::virtual_* v = 0) { boost::contract::guard c = boost::contract::public_function(v, this) .precondition([&] { out << "c::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(false); // To check derived pre. }) .postcondition([&] { out << "c::f::post" << std::endl; }) ; out << "c::f::body" << std::endl; throw c::e(); } }; struct b #define BASES public c : BASES { typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types; #undef BASES void invariant() const { out << "b::inv" << std::endl; } static void static_invariant() { out << "b::static_inv" << std::endl; } struct e {}; virtual void f(boost::contract::virtual_* v = 0) /* override */ { boost::contract::guard c = boost::contract::public_function( v, &b::f, this) .precondition([&] { out << "b::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(false); // To check derived pre. }) .postcondition([&] { out << "b::f::post" << std::endl; }) ; out << "b::f::body" << std::endl; throw b::e(); } BOOST_CONTRACT_OVERRIDE(f) }; struct a #define BASES public b : BASES { typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types; #undef BASES void invariant() const { out << "a::inv" << std::endl; } static void static_invariant() { out << "a::static_inv" << std::endl; } struct e {}; void f(boost::contract::virtual_* v = 0) /* override */ { boost::contract::guard c = boost::contract::public_function( v, &a::f, this) .precondition([&] { out << "a::f::pre" << std::endl; }) .postcondition([&] { out << "a::f::post" << std::endl; }) ; out << "a::f::body" << std::endl; throw a::e(); } BOOST_CONTRACT_OVERRIDE(f) }; int main() { std::ostringstream ok; a aa; c& ca = aa; // Test as virtual call via polymorphism. bool threw = false; out.str(""); try { ca.f(); } catch(a::e const&) { threw = true; } BOOST_TEST(threw); ok.str(""); ok << "c::static_inv" << std::endl << "c::inv" << std::endl << "b::static_inv" << std::endl << "b::inv" << std::endl << "a::static_inv" << std::endl << "a::inv" << std::endl << "c::f::pre" << std::endl << "b::f::pre" << std::endl << "a::f::pre" << std::endl << "a::f::body" << std::endl << "c::static_inv" << std::endl << "c::inv" << std::endl << "b::static_inv" << std::endl << "b::inv" << std::endl << "a::static_inv" << std::endl << "a::inv" << std::endl // Test no post (but still subcontracted inv) because body threw. ; BOOST_TEST(out.eq(ok.str())); return boost::report_errors(); }