mirror of
https://github.com/catchorg/Catch2.git
synced 2024-11-22 21:36:11 +01:00
More complete fix for nested sections
This commit is contained in:
parent
70e7d7f16b
commit
a243a23363
@ -59,12 +59,16 @@ TEST_CASE( "./succeeding/Misc/Sections/nested2", "nested SECTION tests" )
|
|||||||
|
|
||||||
SECTION( "s2", "equal" )
|
SECTION( "s2", "equal" )
|
||||||
{
|
{
|
||||||
REQUIRE( a == b);
|
REQUIRE( a == b );
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION( "s3", "not equal" )
|
SECTION( "s3", "not equal" )
|
||||||
{
|
{
|
||||||
REQUIRE( a != b);
|
REQUIRE( a != b );
|
||||||
|
}
|
||||||
|
SECTION( "s4", "less than" )
|
||||||
|
{
|
||||||
|
REQUIRE( b < a );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -24,6 +24,41 @@
|
|||||||
|
|
||||||
namespace Catch
|
namespace Catch
|
||||||
{
|
{
|
||||||
|
|
||||||
|
class StreamRedirect
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
StreamRedirect
|
||||||
|
(
|
||||||
|
std::ostream& stream,
|
||||||
|
std::string& targetString
|
||||||
|
)
|
||||||
|
: m_stream( stream ),
|
||||||
|
m_prevBuf( stream.rdbuf() ),
|
||||||
|
m_targetString( targetString )
|
||||||
|
{
|
||||||
|
stream.rdbuf( m_oss.rdbuf() );
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
~StreamRedirect
|
||||||
|
()
|
||||||
|
{
|
||||||
|
m_targetString = m_oss.str();
|
||||||
|
m_stream.rdbuf( m_prevBuf );
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::ostream& m_stream;
|
||||||
|
std::streambuf* m_prevBuf;
|
||||||
|
std::ostringstream m_oss;
|
||||||
|
std::string& m_targetString;
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
class TestSpec
|
class TestSpec
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
@ -60,11 +95,126 @@ namespace Catch
|
|||||||
bool m_isWildcarded;
|
bool m_isWildcarded;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
class SectionInfo
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum Status
|
||||||
|
{
|
||||||
|
Root,
|
||||||
|
Unknown,
|
||||||
|
NonLeaf,
|
||||||
|
UntestedLeaf,
|
||||||
|
TestedLeaf
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
SectionInfo
|
||||||
|
(
|
||||||
|
SectionInfo* parent
|
||||||
|
)
|
||||||
|
: m_status( Unknown ),
|
||||||
|
m_parent( parent )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
SectionInfo
|
||||||
|
()
|
||||||
|
: m_status( Root ),
|
||||||
|
m_parent( NULL )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
~SectionInfo
|
||||||
|
()
|
||||||
|
{
|
||||||
|
deleteAllValues( m_subSections );
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
bool shouldRun
|
||||||
|
()
|
||||||
|
const
|
||||||
|
{
|
||||||
|
return m_status != TestedLeaf;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
bool ran
|
||||||
|
()
|
||||||
|
{
|
||||||
|
if( m_status != NonLeaf )
|
||||||
|
{
|
||||||
|
m_status = TestedLeaf;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
SectionInfo* getSubSection
|
||||||
|
(
|
||||||
|
const std::string& name
|
||||||
|
)
|
||||||
|
{
|
||||||
|
std::map<std::string, SectionInfo*>::const_iterator it = m_subSections.find( name );
|
||||||
|
if( it != m_subSections.end() )
|
||||||
|
return it->second;
|
||||||
|
|
||||||
|
SectionInfo* subSection = new SectionInfo( this );
|
||||||
|
m_subSections.insert( std::make_pair( name, subSection ) );
|
||||||
|
m_status = NonLeaf;
|
||||||
|
return subSection;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
SectionInfo* getParent
|
||||||
|
()
|
||||||
|
{
|
||||||
|
return m_parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
bool hasUntestedSections
|
||||||
|
()
|
||||||
|
const
|
||||||
|
{
|
||||||
|
if( m_status == Unknown || m_status == UntestedLeaf )
|
||||||
|
return true;
|
||||||
|
|
||||||
|
std::map<std::string, SectionInfo*>::const_iterator it = m_subSections.begin();
|
||||||
|
std::map<std::string, SectionInfo*>::const_iterator itEnd = m_subSections.end();
|
||||||
|
for(; it != itEnd; ++it )
|
||||||
|
{
|
||||||
|
if( it->second->hasUntestedSections() )
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Status m_status;
|
||||||
|
std::map<std::string, SectionInfo*> m_subSections;
|
||||||
|
SectionInfo* m_parent;
|
||||||
|
};
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////
|
||||||
///////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
class RunningTest
|
class RunningTest
|
||||||
{
|
{
|
||||||
|
enum RunStatus
|
||||||
|
{
|
||||||
|
NothingRun,
|
||||||
|
EncounteredASection,
|
||||||
|
RanAtLeastOneSection,
|
||||||
|
RanToCompletionWithSections,
|
||||||
|
RanToCompletionWithNoSections
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
///////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////
|
||||||
explicit RunningTest
|
explicit RunningTest
|
||||||
@ -72,32 +222,35 @@ namespace Catch
|
|||||||
const TestCaseInfo* info = NULL
|
const TestCaseInfo* info = NULL
|
||||||
)
|
)
|
||||||
: m_info( info ),
|
: m_info( info ),
|
||||||
m_sectionSeen( false ),
|
m_runStatus( RanAtLeastOneSection ),
|
||||||
m_isLeaf( true )
|
m_currentSection( &m_rootSection )
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
|
||||||
size_t sectionsSeenCount
|
|
||||||
()
|
|
||||||
const
|
|
||||||
{
|
|
||||||
return m_sectionsSeen.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////
|
||||||
bool wasSectionSeen
|
bool wasSectionSeen
|
||||||
()
|
()
|
||||||
const
|
const
|
||||||
{
|
{
|
||||||
return m_sectionSeen;
|
return m_runStatus == RanAtLeastOneSection ||
|
||||||
|
m_runStatus == RanToCompletionWithSections;
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////
|
||||||
void resetSectionSeen
|
void reset
|
||||||
()
|
()
|
||||||
{
|
{
|
||||||
m_sectionSeen = false;
|
m_runStatus = NothingRun;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
void ranToCompletion
|
||||||
|
()
|
||||||
|
{
|
||||||
|
m_runStatus = m_runStatus == RanAtLeastOneSection ||
|
||||||
|
m_runStatus == EncounteredASection
|
||||||
|
? RanToCompletionWithSections
|
||||||
|
: RanToCompletionWithNoSections;
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////
|
||||||
@ -106,25 +259,28 @@ namespace Catch
|
|||||||
const std::string& name
|
const std::string& name
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
m_isLeaf = true;
|
if( m_runStatus == NothingRun )
|
||||||
std::string qualifiedSection = m_sectionPath + "\\" + name;
|
m_runStatus = EncounteredASection;
|
||||||
if( m_sectionsSeen.find( qualifiedSection ) != m_sectionsSeen.end() )
|
|
||||||
return false;
|
SectionInfo* thisSection = m_currentSection->getSubSection( name );
|
||||||
m_sectionPath = qualifiedSection;
|
|
||||||
|
if( !wasSectionSeen() && thisSection->shouldRun() )
|
||||||
|
{
|
||||||
|
m_currentSection = thisSection;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////
|
||||||
void endSection
|
void endSection
|
||||||
(
|
(
|
||||||
const std::string& name
|
const std::string&
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
if( m_isLeaf )
|
if( m_currentSection->ran() )
|
||||||
m_sectionsSeen.insert( m_sectionPath );
|
m_runStatus = RanAtLeastOneSection;
|
||||||
m_isLeaf = false;
|
m_currentSection = m_currentSection->getParent();
|
||||||
m_sectionPath = m_sectionPath.substr( 0, m_sectionPath.length()-name.length()-1 );
|
|
||||||
m_sectionSeen = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////
|
||||||
@ -135,52 +291,24 @@ namespace Catch
|
|||||||
return *m_info;
|
return *m_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////
|
||||||
|
bool hasUntestedSections
|
||||||
|
()
|
||||||
|
const
|
||||||
|
{
|
||||||
|
return m_rootSection.hasUntestedSections() ||
|
||||||
|
m_runStatus != RanToCompletionWithSections;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
const TestCaseInfo* m_info;
|
const TestCaseInfo* m_info;
|
||||||
bool m_sectionSeen;
|
RunStatus m_runStatus;
|
||||||
std::string m_sectionPath;
|
SectionInfo m_rootSection;
|
||||||
std::set<std::string> m_sectionsSeen;
|
SectionInfo* m_currentSection;
|
||||||
bool m_isLeaf;
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////
|
||||||
///////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
class StreamRedirect
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
|
||||||
StreamRedirect
|
|
||||||
(
|
|
||||||
std::ostream& stream,
|
|
||||||
std::string& targetString
|
|
||||||
)
|
|
||||||
: m_stream( stream ),
|
|
||||||
m_prevBuf( stream.rdbuf() ),
|
|
||||||
m_targetString( targetString )
|
|
||||||
{
|
|
||||||
stream.rdbuf( m_oss.rdbuf() );
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
|
||||||
~StreamRedirect
|
|
||||||
()
|
|
||||||
{
|
|
||||||
m_targetString = m_oss.str();
|
|
||||||
m_stream.rdbuf( m_prevBuf );
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::ostream& m_stream;
|
|
||||||
std::streambuf* m_prevBuf;
|
|
||||||
std::ostringstream m_oss;
|
|
||||||
std::string& m_targetString;
|
|
||||||
};
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////
|
|
||||||
///////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
class Runner : public IResultCapture, public IRunner
|
class Runner : public IResultCapture, public IRunner
|
||||||
{
|
{
|
||||||
Runner( const Runner& );
|
Runner( const Runner& );
|
||||||
@ -193,7 +321,8 @@ namespace Catch
|
|||||||
(
|
(
|
||||||
const Config& config
|
const Config& config
|
||||||
)
|
)
|
||||||
: m_config( config ),
|
: m_runningTest( NULL ),
|
||||||
|
m_config( config ),
|
||||||
m_successes( 0 ),
|
m_successes( 0 ),
|
||||||
m_failures( 0 ),
|
m_failures( 0 ),
|
||||||
m_reporter( m_config.getReporter() ),
|
m_reporter( m_config.getReporter() ),
|
||||||
@ -263,20 +392,20 @@ namespace Catch
|
|||||||
|
|
||||||
m_reporter->StartTestCase( testInfo );
|
m_reporter->StartTestCase( testInfo );
|
||||||
|
|
||||||
m_runningTest = RunningTest( &testInfo );
|
m_runningTest = new RunningTest( &testInfo );
|
||||||
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
m_runningTest.resetSectionSeen();
|
|
||||||
runCurrentTest( redirectedCout, redirectedCerr );
|
runCurrentTest( redirectedCout, redirectedCerr );
|
||||||
}
|
}
|
||||||
while( m_runningTest.wasSectionSeen() );
|
while( m_runningTest->hasUntestedSections() );
|
||||||
}
|
}
|
||||||
while( Hub::advanceGeneratorsForCurrentTest() );
|
while( Hub::advanceGeneratorsForCurrentTest() );
|
||||||
|
|
||||||
m_runningTest = RunningTest();
|
delete m_runningTest;
|
||||||
|
m_runningTest = NULL;
|
||||||
|
|
||||||
m_reporter->EndTestCase( testInfo, m_successes - prevSuccessCount, m_failures - prevFailureCount, redirectedCout, redirectedCerr );
|
m_reporter->EndTestCase( testInfo, m_successes - prevSuccessCount, m_failures - prevFailureCount, redirectedCout, redirectedCerr );
|
||||||
}
|
}
|
||||||
@ -382,14 +511,13 @@ namespace Catch
|
|||||||
std::size_t& failures
|
std::size_t& failures
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
if( m_runningTest.wasSectionSeen() || !m_runningTest.addSection( name ) )
|
if( !m_runningTest->addSection( name ) )
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
m_reporter->StartSection( name, description );
|
m_reporter->StartSection( name, description );
|
||||||
successes = m_successes;
|
successes = m_successes;
|
||||||
failures = m_failures;
|
failures = m_failures;
|
||||||
|
|
||||||
// !TBD look up whether we should execute this section or not
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -401,7 +529,7 @@ namespace Catch
|
|||||||
std::size_t prevFailures
|
std::size_t prevFailures
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
m_runningTest.endSection( name );
|
m_runningTest->endSection( name );
|
||||||
m_reporter->EndSection( name, m_successes - prevSuccesses, m_failures - prevFailures );
|
m_reporter->EndSection( name, m_successes - prevSuccesses, m_failures - prevFailures );
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -437,7 +565,9 @@ namespace Catch
|
|||||||
()
|
()
|
||||||
const
|
const
|
||||||
{
|
{
|
||||||
return m_runningTest.getTestCaseInfo().getName();
|
return m_runningTest
|
||||||
|
? m_runningTest->getTestCaseInfo().getName()
|
||||||
|
: "";
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
@ -451,9 +581,11 @@ namespace Catch
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
m_runningTest->reset();
|
||||||
StreamRedirect coutRedir( std::cout, redirectedCout );
|
StreamRedirect coutRedir( std::cout, redirectedCout );
|
||||||
StreamRedirect cerrRedir( std::cerr, redirectedCerr );
|
StreamRedirect cerrRedir( std::cerr, redirectedCerr );
|
||||||
m_runningTest.getTestCaseInfo().invoke();
|
m_runningTest->getTestCaseInfo().invoke();
|
||||||
|
m_runningTest->ranToCompletion();
|
||||||
}
|
}
|
||||||
catch( TestFailureException& )
|
catch( TestFailureException& )
|
||||||
{
|
{
|
||||||
@ -473,7 +605,7 @@ namespace Catch
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
RunningTest m_runningTest;
|
RunningTest* m_runningTest;
|
||||||
MutableResultInfo m_currentResult;
|
MutableResultInfo m_currentResult;
|
||||||
|
|
||||||
const Config& m_config;
|
const Config& m_config;
|
||||||
|
Loading…
Reference in New Issue
Block a user