// Copyright 2017, 2021 Peter Dimov. // Distributed under the Boost Software License, Version 1.0. // https://www.boost.org/LICENSE_1_0.txt #include #include #include using namespace boost::system; struct X { static int instances; int v_; X(): v_() { ++instances; } explicit X( int v ): v_( v ) { ++instances; } X( int v1, int v2 ): v_( v1+v2 ) { ++instances; } X( int v1, int v2, int v3 ): v_( v1+v2+v3 ) { ++instances; } X( X const& r ): v_( r.v_ ) { ++instances; } X( X&& r ): v_( r.v_ ) { r.v_ = 0; ++instances; } X& operator=( X const& ) = delete; ~X() { --instances; } }; bool operator==( X const & x1, X const & x2 ) { return x1.v_ == x2.v_; } std::ostream& operator<<( std::ostream& os, X const & x ) { os << "X:" << x.v_; return os; } int X::instances = 0; int main() { { result r; result r2( std::move( r ) ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 0 ); } { result r2( result{} ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 0 ); } BOOST_TEST_EQ( X::instances, 0 ); { result r; result r2( std::move( r ) ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), X() ); BOOST_TEST_EQ( X::instances, 2 ); } BOOST_TEST_EQ( X::instances, 0 ); { result r2( result{} ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), X() ); BOOST_TEST_EQ( X::instances, 1 ); } BOOST_TEST_EQ( X::instances, 0 ); { result r( 1 ); result r2( std::move( r ) ); BOOST_TEST( r.has_value() ); BOOST_TEST( !r.has_error() ); BOOST_TEST_EQ( r.value(), 1 ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 1 ); } { result r2( result( 1 ) ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 1 ); } BOOST_TEST_EQ( X::instances, 0 ); { result r( 1 ); result r2( std::move( r ) ); BOOST_TEST( r.has_value() ); BOOST_TEST( !r.has_error() ); BOOST_TEST_EQ( r.value().v_, 0 ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value().v_, 1 ); BOOST_TEST_EQ( X::instances, 2 ); } BOOST_TEST_EQ( X::instances, 0 ); { result r2( result( 1 ) ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value().v_, 1 ); BOOST_TEST_EQ( X::instances, 1 ); } BOOST_TEST_EQ( X::instances, 0 ); { auto ec = make_error_code( errc::invalid_argument ); result r( ec ); result r2( std::move( r ) ); BOOST_TEST( !r2.has_value() ); BOOST_TEST( r2.has_error() ); BOOST_TEST_EQ( r2.error(), ec ); } { auto ec = make_error_code( errc::invalid_argument ); result r2( result{ ec } ); BOOST_TEST( !r2.has_value() ); BOOST_TEST( r2.has_error() ); BOOST_TEST_EQ( r2.error(), ec ); } BOOST_TEST_EQ( X::instances, 0 ); { result r( 1 ); result r2( std::move( r ) ); BOOST_TEST( !r.has_value() ); BOOST_TEST( r.has_error() ); BOOST_TEST_EQ( r.error().v_, 0 ); BOOST_TEST( !r2.has_value() ); BOOST_TEST( r2.has_error() ); BOOST_TEST_EQ( r2.error().v_, 1 ); BOOST_TEST_EQ( X::instances, 2 ); } BOOST_TEST_EQ( X::instances, 0 ); { result r2( result( 1 ) ); BOOST_TEST( !r2.has_value() ); BOOST_TEST( r2.has_error() ); BOOST_TEST_EQ( r2.error().v_, 1 ); BOOST_TEST_EQ( X::instances, 1 ); } BOOST_TEST_EQ( X::instances, 0 ); // { result r; result r2( std::move( r ) ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); } { result r2( result{} ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); } { auto ec = make_error_code( errc::invalid_argument ); result r( ec ); result r2( std::move( r ) ); BOOST_TEST( !r2.has_value() ); BOOST_TEST( r2.has_error() ); BOOST_TEST_EQ( r2.error(), ec ); } { auto ec = make_error_code( errc::invalid_argument ); result r2( result{ ec } ); BOOST_TEST( !r2.has_value() ); BOOST_TEST( r2.has_error() ); BOOST_TEST_EQ( r2.error(), ec ); } // { int x1 = 1; result r1( x1 ); result r2( std::move( r1 ) ); BOOST_TEST( r1.has_value() ); BOOST_TEST( !r1.has_error() ); BOOST_TEST_EQ( r1.value(), 1 ); BOOST_TEST_EQ( &*r1, &x1 ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 1 ); BOOST_TEST_EQ( &*r2, &x1 ); } { int const x1 = 1; result r1( x1 ); result r2( std::move( r1 ) ); BOOST_TEST( r1.has_value() ); BOOST_TEST( !r1.has_error() ); BOOST_TEST_EQ( r1.value(), 1 ); BOOST_TEST_EQ( &*r1, &x1 ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 1 ); BOOST_TEST_EQ( &*r2, &x1 ); } { int x1 = 1; result r2(( result( x1 ) )); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 1 ); BOOST_TEST_EQ( &*r2, &x1 ); } { int const x1 = 1; result r2(( result( x1 ) )); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value(), 1 ); BOOST_TEST_EQ( &*r2, &x1 ); } { X x1( 1 ); result r1( x1 ); result r2( std::move( r1 ) ); BOOST_TEST( r1.has_value() ); BOOST_TEST( !r1.has_error() ); BOOST_TEST_EQ( r1.value().v_, 1 ); BOOST_TEST_EQ( &*r1, &x1 ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value().v_, 1 ); BOOST_TEST_EQ( &*r2, &x1 ); BOOST_TEST_EQ( X::instances, 1 ); } BOOST_TEST_EQ( X::instances, 0 ); { X x1( 1 ); result r2(( result( x1 ) )); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value().v_, 1 ); BOOST_TEST_EQ( &*r2, &x1 ); BOOST_TEST_EQ( X::instances, 1 ); } BOOST_TEST_EQ( X::instances, 0 ); { X const x1( 1 ); result r1( x1 ); result r2( std::move( r1 ) ); BOOST_TEST( r1.has_value() ); BOOST_TEST( !r1.has_error() ); BOOST_TEST_EQ( r1.value().v_, 1 ); BOOST_TEST_EQ( &*r1, &x1 ); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value().v_, 1 ); BOOST_TEST_EQ( &*r2, &x1 ); BOOST_TEST_EQ( X::instances, 1 ); } BOOST_TEST_EQ( X::instances, 0 ); { X const x1( 1 ); result r2(( result( x1 ) )); BOOST_TEST( r2.has_value() ); BOOST_TEST( !r2.has_error() ); BOOST_TEST_EQ( r2.value().v_, 1 ); BOOST_TEST_EQ( &*r2, &x1 ); BOOST_TEST_EQ( X::instances, 1 ); } BOOST_TEST_EQ( X::instances, 0 ); // return boost::report_errors(); }