// 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 subcontracting with sparse and complex inheritance graph. #include "../detail/oteststream.hpp" #include #include #include #include #include #include #include boost::contract::test::detail::oteststream out; struct j { static void static_invariant() { out << "j::static_inv" << std::endl; } void invariant() const { out << "j::inv" << std::endl; } virtual void f(char ch, boost::contract::virtual_* v = 0) { boost::contract::check c = boost::contract::public_function(v, this) .precondition([&] { out << "j::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(ch == 'j'); }) .old([] { out << "j::f::old" << std::endl; }) .postcondition([] { out << "j::f::post" << std::endl; }) ; out << "j::f::body" << std::endl; } }; struct i { static void static_invariant() { out << "i::static_inv" << std::endl; } void invariant() const { out << "i::inv" << std::endl; } virtual void f(char ch, boost::contract::virtual_* v = 0) { boost::contract::check c = boost::contract::public_function(v, this) .precondition([&] { out << "i::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(ch == 'i'); }) .old([] { out << "i::f::old" << std::endl; }) .postcondition([] { out << "i::f::post" << std::endl; }) ; out << "i::f::body" << std::endl; } }; struct k {}; struct h #define BASES public j : BASES { typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types; #undef BASES static void static_invariant() { out << "h::static_inv" << std::endl; } void invariant() const { out << "h::inv" << std::endl; } virtual void f(char ch, boost::contract::virtual_* v = 0) /* override */ { boost::contract::check c = boost::contract::public_function< override_f>(v, &h::f, this, ch) .precondition([&] { out << "h::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(ch == 'h'); }) .old([] { out << "h::f::old" << std::endl; }) .postcondition([] { out << "h::f::post" << std::endl; }) ; out << "h::f::body" << std::endl; } BOOST_CONTRACT_OVERRIDE(f) }; struct e #define BASES public virtual i : BASES { typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types; #undef BASES static void static_invariant() { out << "e::static_inv" << std::endl; } void invariant() const { out << "e::inv" << std::endl; } virtual void f(char ch, boost::contract::virtual_* v = 0) /* override */ { boost::contract::check c = boost::contract::public_function< override_f>(v, &e::f, this, ch) .precondition([&] { out << "e::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(ch == 'e'); }) .old([] { out << "e::f::old" << std::endl; }) .postcondition([] { out << "e::f::post" << std::endl; }) ; out << "e::f::body" << std::endl; } BOOST_CONTRACT_OVERRIDE(f) }; struct d #define BASES public k, virtual public i : BASES { typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types; #undef BASES static void static_invariant() { out << "d::static_inv" << std::endl; } void invariant() const { out << "d::inv" << std::endl; } }; struct c { static void static_invariant() { out << "c::static_inv" << std::endl; } void invariant() const { out << "c::inv" << std::endl; } virtual void f(char ch, 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(ch == 'c'); }) .old([] { out << "c::f::old" << std::endl; }) .postcondition([] { out << "c::f::post" << std::endl; }) ; out << "c::f::body" << std::endl; } }; struct b #define BASES public c, public d : 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; } }; struct x {}; struct y {}; struct z {}; struct a #define BASES public b, public x, public e, protected y, public h, \ private z : 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; } virtual void f(char ch, boost::contract::virtual_* v = 0) /* override */ { boost::contract::check c = boost::contract::public_function< override_f>(v, &a::f, this, ch) .precondition([&] { out << "a::f::pre" << std::endl; BOOST_CONTRACT_ASSERT(ch == 'a'); }) .old([] { out << "a::f::old" << std::endl; }) .postcondition([] { out << "a::f::post" << std::endl; }) ; out << "a::f::body" << std::endl; } BOOST_CONTRACT_OVERRIDE(f) }; int main() { std::ostringstream ok; a aa; out.str(""); aa.f('a'); ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "c::static_inv" << std::endl << "c::inv" << std::endl << "i::static_inv" << std::endl << "i::inv" << std::endl << "e::static_inv" << std::endl << "e::inv" << std::endl << "j::static_inv" << std::endl << "j::inv" << std::endl << "h::static_inv" << std::endl << "h::inv" << std::endl << "a::static_inv" << std::endl << "a::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "c::f::pre" << std::endl << "i::f::pre" << std::endl << "e::f::pre" << std::endl << "j::f::pre" << std::endl << "h::f::pre" << std::endl << "a::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "c::f::old" << std::endl << "i::f::old" << std::endl << "e::f::old" << std::endl << "j::f::old" << std::endl << "h::f::old" << std::endl << "a::f::old" << std::endl #endif << "a::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "c::static_inv" << std::endl << "c::inv" << std::endl << "i::static_inv" << std::endl << "i::inv" << std::endl << "e::static_inv" << std::endl << "e::inv" << std::endl << "j::static_inv" << std::endl << "j::inv" << std::endl << "h::static_inv" << std::endl << "h::inv" << std::endl << "a::static_inv" << std::endl << "a::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS << "c::f::old" << std::endl << "c::f::post" << std::endl << "i::f::old" << std::endl << "i::f::post" << std::endl << "e::f::old" << std::endl << "e::f::post" << std::endl << "j::f::old" << std::endl << "j::f::post" << std::endl << "h::f::old" << std::endl << "h::f::post" << std::endl // No old call here because not a base object. << "a::f::post" << std::endl #endif ; BOOST_TEST(out.eq(ok.str())); c cc; out.str(""); cc.f('c'); ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "c::static_inv" << std::endl << "c::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "c::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "c::f::old" << std::endl #endif << "c::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "c::static_inv" << std::endl << "c::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS // No old call here because not a base object. << "c::f::post" << std::endl #endif ; BOOST_TEST(out.eq(ok.str())); d dd; out.str(""); dd.f('i'); // d's f inherited from i. ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "i::static_inv" << std::endl << "i::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "i::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "i::f::old" << std::endl #endif << "i::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "i::static_inv" << std::endl << "i::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS // No old call here because not a base object. << "i::f::post" << std::endl #endif ; BOOST_TEST(out.eq(ok.str())); e ee; out.str(""); ee.f('e'); ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "i::static_inv" << std::endl << "i::inv" << std::endl << "e::static_inv" << std::endl << "e::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "i::f::pre" << std::endl << "e::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "i::f::old" << std::endl << "e::f::old" << std::endl #endif << "e::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "i::static_inv" << std::endl << "i::inv" << std::endl << "e::static_inv" << std::endl << "e::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS << "i::f::old" << std::endl << "i::f::post" << std::endl // No old call here because not a base object. << "e::f::post" << std::endl #endif ; BOOST_TEST(out.eq(ok.str())); i ii; out.str(""); ii.f('i'); ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "i::static_inv" << std::endl << "i::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "i::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "i::f::old" << std::endl #endif << "i::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "i::static_inv" << std::endl << "i::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS // No old call here because not a base object. << "i::f::post" << std::endl #endif ; BOOST_TEST(out.eq(ok.str())); h hh; out.str(""); hh.f('h'); ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "j::static_inv" << std::endl << "j::inv" << std::endl << "h::static_inv" << std::endl << "h::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "j::f::pre" << std::endl << "h::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "j::f::old" << std::endl << "h::f::old" << std::endl #endif << "h::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "j::static_inv" << std::endl << "j::inv" << std::endl << "h::static_inv" << std::endl << "h::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS << "j::f::old" << std::endl << "j::f::post" << std::endl // No old call here because not a base object. << "h::f::post" << std::endl #endif ; BOOST_TEST(out.eq(ok.str())); j jj; out.str(""); jj.f('j'); ok.str(""); ok #ifndef BOOST_CONTRACT_NO_ENTRY_INVARIANTS << "j::static_inv" << std::endl << "j::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_PRECONDITIONS << "j::f::pre" << std::endl #endif #ifndef BOOST_CONTRACT_NO_OLDS << "j::f::old" << std::endl #endif << "j::f::body" << std::endl #ifndef BOOST_CONTRACT_NO_EXIT_INVARIANTS << "j::static_inv" << std::endl << "j::inv" << std::endl #endif #ifndef BOOST_CONTRACT_NO_POSTCONDITIONS // No old call here because not a base object. << "j::f::post" << std::endl #endif ; BOOST_TEST(out.eq(ok.str())); return boost::report_errors(); }