| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204 | 
//              Copyright Catch2 Authors// Distributed under the Boost Software License, Version 1.0.//   (See accompanying file LICENSE_1_0.txt or copy at//        https://www.boost.org/LICENSE_1_0.txt)// SPDX-License-Identifier: BSL-1.0#include <catch2/catch_test_macros.hpp>#include <catch2/catch_translate_exception.hpp>#include <catch2/matchers/catch_matchers_string.hpp>#include <string>#include <stdexcept>#ifdef _MSC_VER#pragma warning(disable:4702) // Unreachable code -- unconditional throws and so on#endif#ifdef __clang__#pragma clang diagnostic push#pragma clang diagnostic ignored "-Wweak-vtables"#pragma clang diagnostic ignored "-Wmissing-noreturn"#pragma clang diagnostic ignored "-Wunreachable-code"#endifnamespace {    int thisThrows() {        throw std::domain_error("expected exception");        return 1;    }    int thisDoesntThrow() {        return 0;    }    class CustomException {    public:        explicit CustomException(const std::string& msg)            : m_msg(msg) {}        std::string const& getMessage() const {            return m_msg;        }    private:        std::string m_msg;    };    class CustomStdException : public std::exception {    public:        explicit CustomStdException(const std::string& msg)            : m_msg(msg) {}        ~CustomStdException() noexcept override = default;        CustomStdException( CustomStdException const& ) = default;        CustomStdException& operator=( CustomStdException const& ) = default;        std::string const& getMessage() const {            return m_msg;        }    private:        std::string m_msg;    };    [[noreturn]] void throwCustom() {        throw CustomException("custom exception - not std");    }}TEST_CASE( "When checked exceptions are thrown they can be expected or unexpected", "[!throws]" ) {    REQUIRE_THROWS_AS( thisThrows(), std::domain_error );    REQUIRE_NOTHROW( thisDoesntThrow() );    REQUIRE_THROWS( thisThrows() );}TEST_CASE( "Expected exceptions that don't throw or unexpected exceptions fail the test", "[.][failing][!throws]" ) {    CHECK_THROWS_AS( thisThrows(), std::string );    CHECK_THROWS_AS( thisDoesntThrow(), std::domain_error );    CHECK_NOTHROW( thisThrows() );}TEST_CASE( "When unchecked exceptions are thrown directly they are always failures", "[.][failing][!throws]" ) {    throw std::domain_error( "unexpected exception" );}TEST_CASE( "An unchecked exception reports the line of the last assertion", "[.][failing][!throws]" ) {    CHECK( 1 == 1 );    throw std::domain_error( "unexpected exception" );}TEST_CASE( "When unchecked exceptions are thrown from sections they are always failures", "[.][failing][!throws]" ) {    SECTION( "section name" ) {        throw std::domain_error("unexpected exception");    }}TEST_CASE( "When unchecked exceptions are thrown from functions they are always failures", "[.][failing][!throws]" ) {    CHECK( thisThrows() == 0 );}TEST_CASE( "When unchecked exceptions are thrown during a REQUIRE the test should abort fail", "[.][failing][!throws]" ) {    REQUIRE( thisThrows() == 0 );    FAIL( "This should never happen" );}TEST_CASE( "When unchecked exceptions are thrown during a CHECK the test should continue", "[.][failing][!throws]" ) {    try {        CHECK(thisThrows() == 0);    }    catch(...) {        FAIL( "This should never happen" );    }}TEST_CASE( "When unchecked exceptions are thrown, but caught, they do not affect the test", "[!throws]" ) {    try {        throw std::domain_error( "unexpected exception" );    }    catch(...) {}}CATCH_TRANSLATE_EXCEPTION( CustomException const& ex ) {    return ex.getMessage();}CATCH_TRANSLATE_EXCEPTION( CustomStdException const& ex ) {    return ex.getMessage();}CATCH_TRANSLATE_EXCEPTION( double const& ex ) {    return Catch::Detail::stringify( ex );}TEST_CASE("Non-std exceptions can be translated", "[.][failing][!throws]" ) {    throw CustomException( "custom exception" );}TEST_CASE("Custom std-exceptions can be custom translated", "[.][failing][!throws]" ) {    throw CustomStdException( "custom std exception" );}TEST_CASE( "Custom exceptions can be translated when testing for nothrow", "[.][failing][!throws]" ) {    REQUIRE_NOTHROW( throwCustom() );}TEST_CASE( "Custom exceptions can be translated when testing for throwing as something else", "[.][failing][!throws]" ) {    REQUIRE_THROWS_AS( throwCustom(), std::exception );}TEST_CASE( "Unexpected exceptions can be translated", "[.][failing][!throws]"  ) {    throw double( 3.14 );}TEST_CASE("Thrown string literals are translated", "[.][failing][!throws]") {    throw "For some reason someone is throwing a string literal!";}TEST_CASE("thrown std::strings are translated", "[.][failing][!throws]") {    throw std::string{ "Why would you throw a std::string?" };}TEST_CASE( "Exception messages can be tested for", "[!throws]" ) {    using namespace Catch::Matchers;    SECTION( "exact match" )        REQUIRE_THROWS_WITH( thisThrows(), "expected exception" );    SECTION( "different case" )    REQUIRE_THROWS_WITH( thisThrows(), Equals( "expecteD Exception", Catch::CaseSensitive::No ) );    SECTION( "wildcarded" ) {        REQUIRE_THROWS_WITH( thisThrows(), StartsWith( "expected" ) );        REQUIRE_THROWS_WITH( thisThrows(), EndsWith( "exception" ) );        REQUIRE_THROWS_WITH( thisThrows(), ContainsSubstring( "except" ) );        REQUIRE_THROWS_WITH( thisThrows(), ContainsSubstring( "exCept", Catch::CaseSensitive::No ) );    }}TEST_CASE( "Mismatching exception messages failing the test", "[.][failing][!throws]" ) {    REQUIRE_THROWS_WITH( thisThrows(), "expected exception" );    REQUIRE_THROWS_WITH( thisThrows(), "should fail" );    REQUIRE_THROWS_WITH( thisThrows(), "expected exception" );}TEST_CASE( "#748 - captures with unexpected exceptions", "[.][failing][!throws][!shouldfail]" ) {    int answer = 42;    CAPTURE( answer );    // the message should be printed on the first two sections but not on the third    SECTION( "outside assertions" ) {        thisThrows();    }    SECTION( "inside REQUIRE_NOTHROW" ) {        REQUIRE_NOTHROW( thisThrows() );    }    SECTION( "inside REQUIRE_THROWS" ) {        REQUIRE_THROWS( thisThrows() );    }}#ifdef __clang__#pragma clang diagnostic pop#endif
 |