#include <catch2/catch_test_macros.hpp>
#include <catch2/reporters/catch_reporter_event_listener.hpp>
#include <catch2/reporters/catch_reporter_registrars.hpp>
#include <catch2/catch_test_case_info.hpp>
#include <iostream>
namespace
{
std::string ws(
int
const
level) {
return
std::string( 2 * level,
' '
);
}
std::ostream& operator<<(std::ostream& out, Catch::Tag t) {
return
out <<
"original: "
<< t.original;
}
template
<
typename
T >
std::ostream& operator<<( std::ostream& os, std::vector<T>
const
& v ) {
os <<
"{ "
;
for
(
const
auto
& x : v )
os << x <<
", "
;
return
os <<
"}"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::SourceLineInfo
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- file: "
<< info.file <<
"\n"
<< ws(level+1) <<
"- line: "
<< info.line <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, Catch::MessageInfo
const
& info ) {
os << ws(level+1) <<
"- macroName: '"
<< info.macroName <<
"'\n"
<< ws(level+1) <<
"- message '"
<< info.message <<
"'\n"
;
print( os,level+1 ,
"- lineInfo"
, info.lineInfo );
os << ws(level+1) <<
"- sequence "
<< info.sequence <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, std::vector<Catch::MessageInfo>
const
& v ) {
os << ws(level ) << title <<
":\n"
;
for
(
const
auto
& x : v )
{
os << ws(level+1) <<
"{\n"
;
print( os, level+2, x );
os << ws(level+1) <<
"}\n"
;
}
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::TestRunInfo
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- name: "
<< info.name <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::Counts
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- total(): "
<< info.total() <<
"\n"
<< ws(level+1) <<
"- allPassed(): "
<< info.allPassed() <<
"\n"
<< ws(level+1) <<
"- allOk(): "
<< info.allOk() <<
"\n"
<< ws(level+1) <<
"- passed: "
<< info.passed <<
"\n"
<< ws(level+1) <<
"- failed: "
<< info.failed <<
"\n"
<< ws(level+1) <<
"- failedButOk: "
<< info.failedButOk <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::Totals
const
& info ) {
os << ws(level) << title <<
":\n"
;
print( os, level+1,
"- assertions"
, info.assertions );
print( os, level+1,
"- testCases"
, info.testCases );
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::TestRunStats
const
& info ) {
os << ws(level) << title <<
":\n"
;
print( os, level+1 ,
"- runInfo"
, info.runInfo );
print( os, level+1 ,
"- totals"
, info.totals );
os << ws(level+1) <<
"- aborting: "
<< info.aborting <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::TestCaseInfo
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- isHidden(): "
<< info.isHidden() <<
"\n"
<< ws(level+1) <<
"- throws(): "
<< info.throws() <<
"\n"
<< ws(level+1) <<
"- okToFail(): "
<< info.okToFail() <<
"\n"
<< ws(level+1) <<
"- expectedToFail(): "
<< info.expectedToFail() <<
"\n"
<< ws(level+1) <<
"- tagsAsString(): '"
<< info.tagsAsString() <<
"'\n"
<< ws(level+1) <<
"- name: '"
<< info.name <<
"'\n"
<< ws(level+1) <<
"- className: '"
<< info.className <<
"'\n"
<< ws(level+1) <<
"- tags: "
<< info.tags <<
"\n"
;
print( os, level+1 ,
"- lineInfo"
, info.lineInfo );
os << ws(level+1) <<
"- properties (flags): 0x"
<< std::hex <<
static_cast
<uint32_t>(info.properties) << std::dec <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::TestCaseStats
const
& info ) {
os << ws(level ) << title <<
":\n"
;
print( os, level+1 ,
"- testInfo"
, *info.testInfo );
print( os, level+1 ,
"- totals"
, info.totals );
os << ws(level+1) <<
"- stdOut: "
<< info.stdOut <<
"\n"
<< ws(level+1) <<
"- stdErr: "
<< info.stdErr <<
"\n"
<< ws(level+1) <<
"- aborting: "
<< info.aborting <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::SectionInfo
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- name: "
<< info.name <<
"\n"
;
print( os, level+1 ,
"- lineInfo"
, info.lineInfo );
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::SectionStats
const
& info ) {
os << ws(level ) << title <<
":\n"
;
print( os, level+1 ,
"- sectionInfo"
, info.sectionInfo );
print( os, level+1 ,
"- assertions"
, info.assertions );
os << ws(level+1) <<
"- durationInSeconds: "
<< info.durationInSeconds <<
"\n"
<< ws(level+1) <<
"- missingAssertions: "
<< info.missingAssertions <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::AssertionInfo
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- macroName: '"
<< info.macroName <<
"'\n"
;
print( os, level+1 ,
"- lineInfo"
, info.lineInfo );
os << ws(level+1) <<
"- capturedExpression: '"
<< info.capturedExpression <<
"'\n"
<< ws(level+1) <<
"- resultDisposition (flags): 0x"
<< std::hex << info.resultDisposition << std::dec <<
"\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::AssertionResultData
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- reconstructExpression(): '"
<< info.reconstructExpression() <<
"'\n"
<< ws(level+1) <<
"- message: '"
<< info.message <<
"'\n"
<< ws(level+1) <<
"- lazyExpression: '"
<<
"(info.lazyExpression)"
<<
"'\n"
<< ws(level+1) <<
"- resultType: '"
<< info.resultType <<
"'\n"
;
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::AssertionResult
const
& info ) {
os << ws(level ) << title <<
":\n"
<< ws(level+1) <<
"- isOk(): "
<< info.isOk() <<
"\n"
<< ws(level+1) <<
"- succeeded(): "
<< info.succeeded() <<
"\n"
<< ws(level+1) <<
"- getResultType(): "
<< info.getResultType() <<
"\n"
<< ws(level+1) <<
"- hasExpression(): "
<< info.hasExpression() <<
"\n"
<< ws(level+1) <<
"- hasMessage(): "
<< info.hasMessage() <<
"\n"
<< ws(level+1) <<
"- getExpression(): '"
<< info.getExpression() <<
"'\n"
<< ws(level+1) <<
"- getExpressionInMacro(): '"
<< info.getExpressionInMacro() <<
"'\n"
<< ws(level+1) <<
"- hasExpandedExpression(): "
<< info.hasExpandedExpression() <<
"\n"
<< ws(level+1) <<
"- getExpandedExpression(): "
<< info.getExpandedExpression() <<
"'\n"
<< ws(level+1) <<
"- getMessage(): '"
<< info.getMessage() <<
"'\n"
;
print( os, level+1 ,
"- getSourceInfo(): "
, info.getSourceInfo() );
os << ws(level+1) <<
"- getTestMacroName(): '"
<< info.getTestMacroName() <<
"'\n"
;
print( os, level+1 ,
"- *** m_info (AssertionInfo)"
, info.m_info );
print( os, level+1 ,
"- *** m_resultData (AssertionResultData)"
, info.m_resultData );
}
void
print( std::ostream& os,
int
const
level, std::string
const
& title, Catch::AssertionStats
const
& info ) {
os << ws(level ) << title <<
":\n"
;
print( os, level+1 ,
"- assertionResult"
, info.assertionResult );
print( os, level+1 ,
"- infoMessages"
, info.infoMessages );
print( os, level+1 ,
"- totals"
, info.totals );
}
char
const
* dashed_line =
"--------------------------------------------------------------------------"
;
struct
MyListener : Catch::EventListenerBase {
using
EventListenerBase::EventListenerBase;
~MyListener() override;
void
testRunStarting( Catch::TestRunInfo
const
& testRunInfo ) override {
std::cout
<< std::boolalpha
<<
"\nEvent: testRunStarting:\n"
;
print( std::cout, 1,
"- testRunInfo"
, testRunInfo );
}
void
testRunEnded( Catch::TestRunStats
const
& testRunStats ) override {
std::cout
<< dashed_line
<<
"\nEvent: testRunEnded:\n"
;
print( std::cout, 1,
"- testRunStats"
, testRunStats );
}
void
skipTest( Catch::TestCaseInfo
const
& testInfo ) override {
std::cout
<< dashed_line
<<
"\nEvent: skipTest:\n"
;
print( std::cout, 1,
"- testInfo"
, testInfo );
}
void
testCaseStarting( Catch::TestCaseInfo
const
& testInfo ) override {
std::cout
<< dashed_line
<<
"\nEvent: testCaseStarting:\n"
;
print( std::cout, 1,
"- testInfo"
, testInfo );
}
void
testCaseEnded( Catch::TestCaseStats
const
& testCaseStats ) override {
std::cout <<
"\nEvent: testCaseEnded:\n"
;
print( std::cout, 1,
"testCaseStats"
, testCaseStats );
}
void
sectionStarting( Catch::SectionInfo
const
& sectionInfo ) override {
std::cout <<
"\nEvent: sectionStarting:\n"
;
print( std::cout, 1,
"- sectionInfo"
, sectionInfo );
}
void
sectionEnded( Catch::SectionStats
const
& sectionStats ) override {
std::cout <<
"\nEvent: sectionEnded:\n"
;
print( std::cout, 1,
"- sectionStats"
, sectionStats );
}
void
assertionStarting( Catch::AssertionInfo
const
& assertionInfo ) override {
std::cout <<
"\nEvent: assertionStarting:\n"
;
print( std::cout, 1,
"- assertionInfo"
, assertionInfo );
}
void
assertionEnded( Catch::AssertionStats
const
& assertionStats ) override {
std::cout <<
"\nEvent: assertionEnded:\n"
;
print( std::cout, 1,
"- assertionStats"
, assertionStats );
}
};
}
CATCH_REGISTER_LISTENER( MyListener )
MyListener::~MyListener() =
default
;
TEST_CASE(
"1: Hidden testcase"
,
"[.hidden]"
) {
}
TEST_CASE(
"2: Testcase with sections"
,
"[tag-A][tag-B]"
) {
int
i = 42;
REQUIRE( i == 42 );
SECTION(
"Section 1"
) {
INFO(
"Section 1"
);
i = 7;
SECTION(
"Section 1.1"
) {
INFO(
"Section 1.1"
);
REQUIRE( i == 42 );
}
}
SECTION(
"Section 2"
) {
INFO(
"Section 2"
);
REQUIRE( i == 42 );
}
WARN(
"At end of test case"
);
}
struct
Fixture {
int
fortytwo()
const
{
return
42;
}
};
TEST_CASE_METHOD( Fixture,
"3: Testcase with class-based fixture"
,
"[tag-C][tag-D]"
) {
REQUIRE( fortytwo() == 42 );
}