// Copyright (C) 2008-2018 Lorenzo Caminiti // Distributed under the Boost Software License, Version 1.0 (see accompanying // file LICENSE_1_0.txt or a copy at http://www.boost.org/LICENSE_1_0.txt). // See: http://www.boost.org/doc/libs/release/libs/contract/doc/html/index.html // Test throw from public function (derived and bases) .old(). #include "../detail/oteststream.hpp" #include #include #include #include #include #include #include boost::contract::test::detail::oteststream out; struct c_err {}; // Global decl so visible in MSVC10 lambdas. struct c { static void static_invariant() { out << "c::static_inv" << std::endl; } void invariant() const { out << "c::inv" << std::endl; } virtual void f(boost::contract::virtual_* v = 0) { boost::contract::check c = boost::contract::public_function(v, this) .precondition([] { out << "c::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(false); // To check derived pre. }) .old([] { out << "c::f::old" << std::endl; throw c_err(); // Test this throws. }) .postcondition([] { out << "c::f::post" << std::endl; }) .except([] { out << "c::f::except" << std::endl; }) ; out << "c::f::body" << std::endl; } }; struct b_err {}; // Global decl so visible in MSVC10 lambdas. struct b #define BASES public c : BASES { typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types; #undef BASES static void static_invariant() { out << "b::static_inv" << std::endl; } void invariant() const { out << "b::inv" << std::endl; } virtual void f(boost::contract::virtual_* v = 0) /* override */ { boost::contract::check c = boost::contract::public_function( v, &b::f, this) .precondition([] { out << "b::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(false); // To check derived pre. }) .old([] { out << "b::f::old" << std::endl; throw b_err(); // Test this throws. }) .postcondition([] { out << "b::f::post" << std::endl; }) .except([] { out << "b::f::except" << std::endl; }) ; out << "b::f::body" << std::endl; } BOOST_CONTRACT_OVERRIDE(f) }; struct a_err {}; // Global decl so visible in MSVC10 lambdas. struct a #define BASES public b : BASES { typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types; #undef BASES static void static_invariant() { out << "a::static_inv" << std::endl; } void invariant() const { out << "a::inv" << std::endl; } void f(boost::contract::virtual_* v = 0) /* override */ { boost::contract::check c = boost::contract::public_function( v, &a::f, this) .precondition([] { out << "a::f::pre" << std::endl; }) .old([] { out << "a::f::old" << std::endl; throw a_err(); // Test this throws. }) .postcondition([] { out << "a::f::post" << std::endl; }) .except([] { out << "a::f::except" << std::endl; }) ; out << "a::f::body" << std::endl; } BOOST_CONTRACT_OVERRIDE(f) }; int main() { std::ostringstream ok; boost::contract::set_old_failure([] (boost::contract::from) { throw; }); a aa; b& ba = aa; // Test as virtual call via polymorphism. try { out.str(""); ba.f(); #ifndef BOOST_CONTRACT_NO_OLDS BOOST_TEST(false); } catch(c_err const&) { #endif ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "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 #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "c::f::pre" << std::endl << "b::f::pre" << std::endl << "a::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "c::f::old" << std::endl // Test this threw. #else << "a::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "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 #endif #endif ; BOOST_TEST(out.eq(ok.str())); } catch(...) { BOOST_TEST(false); } return boost::report_errors(); }