#define CATCH_CONFIG_ENABLE_VARIANT_STRINGMAKER #include "catch.hpp" #if defined(CATCH_CONFIG_CPP17_VARIANT) && !defined(CATCH_CONFIG_DISABLE_EXCEPTIONS) #include #include // We need 2 types with non-trivial copies/moves struct MyType1 { MyType1() = default; [[noreturn]] MyType1(MyType1 const&) { throw 1; } MyType1& operator=(MyType1 const&) { throw 3; } }; struct MyType2 { MyType2() = default; [[noreturn]] MyType2(MyType2 const&) { throw 2; } MyType2& operator=(MyType2 const&) { throw 4; } }; TEST_CASE( "variant", "[toString][variant][approvals]") { using type = std::variant; CHECK( "{ }" == ::Catch::Detail::stringify(type{}) ); type value {}; CHECK( "{ }" == ::Catch::Detail::stringify(value) ); CHECK( "{ }" == ::Catch::Detail::stringify(std::get<0>(value)) ); } TEST_CASE( "variant", "[toString][variant][approvals]") { using type = std::variant; CHECK( "0" == ::Catch::Detail::stringify(type{0}) ); } TEST_CASE( "variant", "[toString][variant][approvals]") { using type = std::variant; CHECK( "0.5f" == ::Catch::Detail::stringify(type{0.5f}) ); CHECK( "0" == ::Catch::Detail::stringify(type{0}) ); } TEST_CASE( "variant -- valueless-by-exception", "[toString][variant][approvals]" ) { using type = std::variant; type value; REQUIRE_THROWS_AS(value.emplace(MyType2{}), int); REQUIRE(value.valueless_by_exception()); CHECK("{valueless variant}" == ::Catch::Detail::stringify(value)); } TEST_CASE( "variant", "[toString][variant][approvals]") { using type = std::variant; CHECK( "\"foo\"" == ::Catch::Detail::stringify(type{"foo"}) ); CHECK( "0" == ::Catch::Detail::stringify(type{0}) ); } TEST_CASE( "variant, string>", "[toString][variant][approvals]") { using inner = std::variant; using type = std::variant; CHECK( "0.5f" == ::Catch::Detail::stringify(type{0.5f}) ); CHECK( "0" == ::Catch::Detail::stringify(type{0}) ); CHECK( "\"foo\"" == ::Catch::Detail::stringify(type{"foo"}) ); SECTION("valueless nested variant") { type value = inner{0.5f}; REQUIRE( std::holds_alternative(value) ); REQUIRE( std::holds_alternative(std::get(value)) ); REQUIRE_THROWS_AS( std::get<0>(value).emplace(MyType1{}), int ); // outer variant is still valid and contains inner REQUIRE( std::holds_alternative(value) ); // inner variant is valueless REQUIRE( std::get(value).valueless_by_exception() ); CHECK( "{valueless variant}" == ::Catch::Detail::stringify(value) ); } } TEST_CASE( "variant", "[toString][variant][approvals]" ) { using type = std::variant; CHECK( "nullptr" == ::Catch::Detail::stringify(type{nullptr}) ); CHECK( "42" == ::Catch::Detail::stringify(type{42}) ); CHECK( "\"Catch me\"" == ::Catch::Detail::stringify(type{"Catch me"}) ); } #endif // CATCH_INTERNAL_CONFIG_CPP17_VARIANT