| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287 | 
//              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 <iostream>TEST_CASE( "INFO and WARN do not abort tests", "[messages][.]" ) {    INFO( "this is a " << "message" );    // This should output the message if a failure occurs    WARN( "this is a " << "warning" );    // This should always output the message but then continue}TEST_CASE( "#1455 - INFO and WARN can start with a linebreak", "[messages][.]" ) {    // Previously these would be hidden from the console reporter output,    // because it would fail at properly reflowing the text    INFO( "\nThis info message starts with a linebreak" );    WARN( "\nThis warning message starts with a linebreak" );}TEST_CASE( "SUCCEED counts as a test pass", "[messages]" ) {    SUCCEED( "this is a " << "success" );}TEST_CASE( "INFO gets logged on failure", "[failing][messages][.]" ) {    INFO( "this message should be logged" );    INFO( "so should this" );    int a = 2;    REQUIRE( a == 1 );}TEST_CASE( "INFO gets logged on failure, even if captured before successful assertions", "[failing][messages][.]" ) {    INFO( "this message may be logged later" );    int a = 2;    CHECK( a == 2 );    INFO( "this message should be logged" );    CHECK( a == 1 );    INFO( "and this, but later" );    CHECK( a == 0 );    INFO( "but not this" );    CHECK( a == 2 );}TEST_CASE( "FAIL aborts the test", "[failing][messages][.]" ) {    FAIL( "This is a " << "failure" );    // This should output the message and abort    WARN( "We should never see this");}TEST_CASE( "FAIL_CHECK does not abort the test", "[failing][messages][.]" ) {    FAIL_CHECK( "This is a " << "failure" );    // This should output the message then continue    WARN( "This message appears in the output");}TEST_CASE( "FAIL does not require an argument", "[failing][messages][.]" ) {    FAIL();}TEST_CASE( "SUCCEED does not require an argument", "[messages][.]" ) {   SUCCEED();}TEST_CASE( "Output from all sections is reported", "[failing][messages][.]" ) {    SECTION( "one" ) {        FAIL( "Message from section one" );    }    SECTION( "two" ) {        FAIL( "Message from section two" );    }}TEST_CASE( "Standard output from all sections is reported", "[messages][.]" ) {    SECTION( "one" ) {        std::cout << "Message from section one" << std::endl;    }    SECTION( "two" ) {        std::cout << "Message from section two" << std::endl;    }}TEST_CASE( "Standard error is reported and redirected", "[messages][.][approvals]" ) {    SECTION( "std::cerr" ) {        std::cerr << "Write to std::cerr" << std::endl;    }    SECTION( "std::clog" ) {        std::clog << "Write to std::clog" << std::endl;    }    SECTION( "Interleaved writes to cerr and clog" ) {        std::cerr << "Inter";        std::clog << "leaved";        std::cerr << ' ';        std::clog << "writes";        std::cerr << " to error";        std::clog << " streams" << std::endl;    }}TEST_CASE( "INFO is reset for each loop", "[messages][failing][.]" ) {    for( int i=0; i<100; i++ )    {        INFO( "current counter " << i );        CAPTURE( i );        REQUIRE( i < 10 );    }}TEST_CASE( "The NO_FAIL macro reports a failure but does not fail the test", "[messages]" ) {    CHECK_NOFAIL( 1 == 2 );}TEST_CASE( "just info", "[info][isolated info][messages]" ) {    INFO( "this should never be seen" );}TEST_CASE( "just failure", "[fail][isolated info][.][messages]" ) {    FAIL( "Previous info should not be seen" );}TEST_CASE( "sends information to INFO", "[.][failing]" ) {    INFO( "hi" );    int i = 7;    CAPTURE( i );    REQUIRE( false );}TEST_CASE( "Pointers can be converted to strings", "[messages][.][approvals]" ) {    int p;    WARN( "actual address of p: " << &p );    WARN( "toString(p): " << ::Catch::Detail::stringify( &p ) );}template <typename T>static void unscoped_info( T msg ) {    UNSCOPED_INFO( msg );}TEST_CASE( "just unscoped info", "[unscoped][info]" ) {    unscoped_info( "this should NOT be seen" );    unscoped_info( "this also should NOT be seen" );}TEST_CASE( "just failure after unscoped info", "[failing][.][unscoped][info]" ) {    FAIL( "previous unscoped info SHOULD not be seen" );}TEST_CASE( "print unscoped info if passing unscoped info is printed", "[unscoped][info]" ) {    unscoped_info( "this MAY be seen IF info is printed for passing assertions" );    REQUIRE( true );}TEST_CASE( "prints unscoped info on failure", "[failing][.][unscoped][info]" ) {    unscoped_info( "this SHOULD be seen" );    unscoped_info( "this SHOULD also be seen" );    REQUIRE( false );    unscoped_info( "but this should NOT be seen" );}TEST_CASE( "not prints unscoped info from previous failures", "[failing][.][unscoped][info]" ) {    unscoped_info( "this MAY be seen only for the FIRST assertion IF info is printed for passing assertions" );    REQUIRE( true );    unscoped_info( "this MAY be seen only for the SECOND assertion IF info is printed for passing assertions" );    REQUIRE( true );    unscoped_info( "this SHOULD be seen" );    REQUIRE( false );}TEST_CASE( "prints unscoped info only for the first assertion", "[failing][.][unscoped][info]" ) {    unscoped_info( "this SHOULD be seen only ONCE" );    CHECK( false );    CHECK( true );    unscoped_info( "this MAY also be seen only ONCE IF info is printed for passing assertions" );    CHECK( true );    CHECK( true );}TEST_CASE( "stacks unscoped info in loops", "[failing][.][unscoped][info]" ) {    UNSCOPED_INFO("Count 1 to 3...");    for (int i = 1; i <= 3; i++) {        unscoped_info(i);    }    CHECK( false );    UNSCOPED_INFO("Count 4 to 6...");    for (int i = 4; i <= 6; i++) {        unscoped_info(i);    }    CHECK( false );}TEST_CASE( "mix info, unscoped info and warning", "[unscoped][info]" ) {    INFO("info");    unscoped_info("unscoped info");    WARN("and warn may mix");    WARN("they are not cleared after warnings");}TEST_CASE( "CAPTURE can deal with complex expressions", "[messages][capture]" ) {    int a = 1;    int b = 2;    int c = 3;    CAPTURE( a, b, c, a + b, a+b, c > b, a == 1 );    SUCCEED();}#ifdef __clang__#pragma clang diagnostic push#pragma clang diagnostic ignored "-Wunused-value" // In (1, 2), the "1" is unused ...#endif#ifdef __GNUC__#pragma GCC diagnostic push#pragma GCC diagnostic ignored "-Wunused-value" // All the comma operators are side-effect free#endif#ifdef _MSC_VER#pragma warning(push)#pragma warning(disable:4709) // comma in indexing operator#endiftemplate <typename T1, typename T2>struct helper_1436 {    helper_1436(T1 t1_, T2 t2_):        t1{ t1_ },        t2{ t2_ }    {}    T1 t1;    T2 t2;};template <typename T1, typename T2>std::ostream& operator<<(std::ostream& out, helper_1436<T1, T2> const& helper) {    out << "{ " << helper.t1 << ", " << helper.t2 << " }";    return out;}// Clang and gcc have different names for this warning, and clang also// warns about an unused value. This warning must be disabled for C++20.#if defined(__GNUG__) && !defined(__clang__)#pragma GCC diagnostic push#pragma GCC diagnostic ignored "-Wpragmas"#pragma GCC diagnostic ignored "-Wcomma-subscript"#elif defined(__clang__)#pragma clang diagnostic push#pragma clang diagnostic ignored "-Wunknown-pragmas"#pragma clang diagnostic ignored "-Wunknown-warning-option"#pragma clang diagnostic ignored "-Wdeprecated-comma-subscript"#pragma clang diagnostic ignored "-Wunused-value"#endifTEST_CASE("CAPTURE can deal with complex expressions involving commas", "[messages][capture]") {    CAPTURE(std::vector<int>{1, 2, 3}[0, 1, 2],            std::vector<int>{1, 2, 3}[(0, 1)],            std::vector<int>{1, 2, 3}[0]);    CAPTURE((helper_1436<int, int>{12, -12}),            (helper_1436<int, int>(-12, 12)));    CAPTURE( (1, 2), (2, 3) );    SUCCEED();}#ifdef __GNUG__#pragma GCC diagnostic pop#endifTEST_CASE("CAPTURE parses string and character constants", "[messages][capture]") {    CAPTURE(("comma, in string", "escaped, \", "), "single quote in string,',", "some escapes, \\,\\\\");    CAPTURE("some, ), unmatched, } prenheses {[<");    CAPTURE('"', '\'', ',', '}', ')', '(', '{');    SUCCEED();}#ifdef __clang__#pragma clang diagnostic pop#endif#ifdef __GNUC__#pragma GCC diagnostic pop#endif#ifdef _MSC_VER#pragma warning(pop)#endif
 |