- fixes infinite loop bug (#185 and #166)
This commit is contained in:
Phil Nash 2013-07-25 08:12:03 +01:00
parent 28d3881ff9
commit 8a52a39fdc
3 changed files with 153 additions and 207 deletions

View File

@ -1,6 +1,6 @@
![catch logo](catch-logo-small.png) ![catch logo](catch-logo-small.png)
*v1.0 build 5 (master branch)* *v1.0 build 6 (master branch)*
# New release with significant changes # New release with significant changes

View File

@ -13,7 +13,7 @@
namespace Catch { namespace Catch {
// These numbers are maintained by a script // These numbers are maintained by a script
Version libraryVersion( 1, 0, 5, "master" ); Version libraryVersion( 1, 0, 6, "master" );
} }
#endif // TWOBLUECUBES_CATCH_VERSION_HPP_INCLUDED #endif // TWOBLUECUBES_CATCH_VERSION_HPP_INCLUDED

View File

@ -1,6 +1,6 @@
/* /*
* CATCH v1.0 build 5 (master branch) * CATCH v1.0 build 6 (master branch)
* Generated: 2013-07-05 08:38:07.926803 * Generated: 2013-07-25 08:10:27.710799
* ---------------------------------------------------------- * ----------------------------------------------------------
* This file has been merged from multiple headers. Please don't edit it directly * This file has been merged from multiple headers. Please don't edit it directly
* Copyright (c) 2012 Two Blue Cubes Ltd. All rights reserved. * Copyright (c) 2012 Two Blue Cubes Ltd. All rights reserved.
@ -2248,12 +2248,18 @@ namespace Catch {
nullableValue = new( storage ) T( *_other ); nullableValue = new( storage ) T( *_other );
return *this; return *this;
} }
Option& operator = ( T const& _value ) {
reset();
nullableValue = new( storage ) T( _value );
return *this;
}
void reset() { void reset() {
if( nullableValue ) if( nullableValue )
nullableValue->~T(); nullableValue->~T();
nullableValue = NULL; nullableValue = NULL;
} }
T& operator*() { return *nullableValue; } T& operator*() { return *nullableValue; }
T const& operator*() const { return *nullableValue; } T const& operator*() const { return *nullableValue; }
T* operator->() { return nullableValue; } T* operator->() { return nullableValue; }
@ -4549,214 +4555,143 @@ namespace Catch {
// #included from: internal/catch_runner_impl.hpp // #included from: internal/catch_runner_impl.hpp
#define TWOBLUECUBES_CATCH_RUNNER_IMPL_HPP_INCLUDED #define TWOBLUECUBES_CATCH_RUNNER_IMPL_HPP_INCLUDED
// #included from: catch_running_test.hpp // #included from: catch_test_case_tracker.hpp
#define TWOBLUECUBES_CATCH_RUNNING_TEST_HPP_INCLUDED #define TWOBLUECUBES_CATCH_TEST_CASE_TRACKER_HPP_INCLUDED
// #included from: catch_section_info.hpp
#define TWOBLUECUBES_CATCH_SECTION_INFO_HPP_INCLUDED
#include <map> #include <map>
#include <string> #include <string>
#include <assert.h>
namespace Catch { namespace Catch {
namespace SectionTracking {
class TrackedSection {
typedef std::map<std::string, TrackedSection> TrackedSections;
class RunningSection {
public: public:
enum RunState {
typedef std::vector<RunningSection*> SubSections; NotStarted,
Executing,
enum State { ExecutingChildren,
Root, Completed
Unknown,
Branch,
TestedBranch,
TestedLeaf
}; };
RunningSection( RunningSection* parent, std::string const& name ) TrackedSection( std::string const& name, TrackedSection* parent )
: m_state( Unknown ), : m_name( name ), m_runState( NotStarted ), m_parent( parent )
m_parent( parent ),
m_name( name )
{} {}
RunningSection( std::string const& name ) RunState runState() const { return m_runState; }
: m_state( Root ),
m_parent( NULL ),
m_name( name )
{}
~RunningSection() { void addChild( std::string const& childName ) {
deleteAll( m_subSections ); m_children.insert( std::make_pair( childName, TrackedSection( childName, this ) ) );
}
TrackedSection* getChild( std::string const& childName ) {
return &m_children.find( childName )->second;
} }
std::string getName() const { void enter() {
return m_name; if( m_runState == NotStarted )
m_runState = Executing;
} }
void leave() {
bool shouldRun() const { for( TrackedSections::const_iterator it = m_children.begin(), itEnd = m_children.end();
return m_state < TestedBranch; it != itEnd;
++it )
if( it->second.runState() != Completed ) {
m_runState = ExecutingChildren;
return;
}
m_runState = Completed;
} }
TrackedSection* getParent() {
bool isBranch() const {
return m_state == Branch;
}
const RunningSection* getParent() const {
return m_parent; return m_parent;
} }
bool hasChildren() const {
bool hasUntestedSections() const { return !m_children.empty();
if( m_state == Unknown )
return true;
for( SubSections::const_iterator it = m_subSections.begin();
it != m_subSections.end();
++it)
if( (*it)->hasUntestedSections() )
return true;
return false;
}
// Mutable methods:
RunningSection* getParent() {
return m_parent;
}
RunningSection* findOrAddSubSection( std::string const& name, bool& changed ) {
for( SubSections::const_iterator it = m_subSections.begin();
it != m_subSections.end();
++it)
if( (*it)->getName() == name )
return *it;
RunningSection* subSection = new RunningSection( this, name );
m_subSections.push_back( subSection );
m_state = Branch;
changed = true;
return subSection;
}
bool ran() {
if( m_state >= Branch )
return false;
m_state = TestedLeaf;
return true;
}
void ranToCompletion() {
if( m_state == Branch && !hasUntestedSections() )
m_state = TestedBranch;
} }
private: private:
State m_state;
RunningSection* m_parent;
std::string m_name; std::string m_name;
SubSections m_subSections; RunState m_runState;
TrackedSections m_children;
TrackedSection* m_parent;
}; };
}
namespace Catch {
class RunningTest {
enum RunStatus {
NothingRun,
EncounteredASection,
RanAtLeastOneSection,
RanToCompletionWithSections,
RanToCompletionWithNoSections
};
class TestCaseTracker {
public: public:
explicit RunningTest( TestCase const& info ) TestCaseTracker( std::string const& testCaseName )
: m_info( info ), : m_testCase( testCaseName, NULL ),
m_runStatus( RanAtLeastOneSection ), m_currentSection( &m_testCase ),
m_rootSection( info.getTestCaseInfo().name ), m_completedASectionThisRun( false )
m_currentSection( &m_rootSection ),
m_changed( false )
{} {}
bool wasSectionSeen() const { bool enterSection( std::string const& name ) {
return m_runStatus == RanAtLeastOneSection || if( m_completedASectionThisRun )
m_runStatus == RanToCompletionWithSections; return false;
} if( m_currentSection->runState() == TrackedSection::Executing ) {
m_currentSection->addChild( name );
bool isBranchSection() const { return false;
return m_currentSection && }
m_currentSection->isBranch(); else {
} TrackedSection* child = m_currentSection->getChild( name );
if( child->runState() != TrackedSection::Completed ) {
bool hasSections() const { m_currentSection = child;
return m_runStatus == RanAtLeastOneSection || m_currentSection->enter();
m_runStatus == RanToCompletionWithSections || return true;
m_runStatus == EncounteredASection; }
} return false;
void reset() {
m_runStatus = NothingRun;
m_changed = false;
m_lastSectionToRun = NULL;
}
void ranToCompletion() {
if( m_runStatus != RanAtLeastOneSection && m_runStatus != EncounteredASection )
m_runStatus = RanToCompletionWithNoSections;
m_runStatus = RanToCompletionWithSections;
if( m_lastSectionToRun ) {
m_lastSectionToRun->ranToCompletion();
m_changed = true;
} }
} }
void leaveSection() {
bool addSection( std::string const& name ) { m_currentSection->leave();
if( m_runStatus == NothingRun )
m_runStatus = EncounteredASection;
RunningSection* thisSection = m_currentSection->findOrAddSubSection( name, m_changed );
if( !wasSectionSeen() && thisSection->shouldRun() ) {
m_currentSection = thisSection;
m_lastSectionToRun = NULL;
return true;
}
return false;
}
void endSection( std::string const&, bool stealth ) {
if( m_currentSection->ran() ) {
if( !stealth )
m_runStatus = RanAtLeastOneSection;
m_changed = true;
}
else if( m_runStatus == EncounteredASection ) {
if( !stealth )
m_runStatus = RanAtLeastOneSection;
m_lastSectionToRun = m_currentSection;
}
m_currentSection = m_currentSection->getParent(); m_currentSection = m_currentSection->getParent();
assert( m_currentSection != NULL );
m_completedASectionThisRun = true;
} }
TestCase const& getTestCase() const { bool currentSectionHasChildren() const {
return m_info; return m_currentSection->hasChildren();
}
bool isCompleted() const {
return m_testCase.runState() == TrackedSection::Completed;
} }
bool hasUntestedSections() const { class Guard {
return m_runStatus == RanAtLeastOneSection || public:
( m_rootSection.hasUntestedSections() && m_changed ); Guard( TestCaseTracker& tracker )
} : m_tracker( tracker )
{
m_tracker.enterTestCase();
}
~Guard() {
m_tracker.leaveTestCase();
}
private:
TestCaseTracker& m_tracker;
};
private: private:
RunningTest( RunningTest const& ); void enterTestCase() {
void operator=( RunningTest const& ); m_currentSection = &m_testCase;
m_completedASectionThisRun = false;
m_testCase.enter();
}
void leaveTestCase() {
m_testCase.leave();
}
TestCase const& m_info; TrackedSection m_testCase;
RunStatus m_runStatus; TrackedSection* m_currentSection;
RunningSection m_rootSection; bool m_completedASectionThisRun;
RunningSection* m_currentSection;
RunningSection* m_lastSectionToRun;
bool m_changed;
}; };
}
} // namespace SectionTracking
using SectionTracking::TestCaseTracker;
} // namespace Catch
#include <set> #include <set>
#include <string> #include <string>
@ -4798,7 +4733,7 @@ namespace Catch {
explicit RunContext( Ptr<IConfig const> const& config, Ptr<IStreamingReporter> const& reporter ) explicit RunContext( Ptr<IConfig const> const& config, Ptr<IStreamingReporter> const& reporter )
: m_runInfo( config->name() ), : m_runInfo( config->name() ),
m_context( getCurrentMutableContext() ), m_context( getCurrentMutableContext() ),
m_runningTest( NULL ), m_activeTestCase( NULL ),
m_config( config ), m_config( config ),
m_reporter( reporter ), m_reporter( reporter ),
m_prevRunner( &m_context.getRunner() ), m_prevRunner( &m_context.getRunner() ),
@ -4853,13 +4788,14 @@ namespace Catch {
m_reporter->testCaseStarting( testInfo ); m_reporter->testCaseStarting( testInfo );
m_runningTest = new RunningTest( testCase ); m_activeTestCase = &testCase;
m_testCaseTracker = TestCaseTracker( testInfo.name );
do { do {
do { do {
runCurrentTest( redirectedCout, redirectedCerr ); runCurrentTest( redirectedCout, redirectedCerr );
} }
while( m_runningTest->hasUntestedSections() && !aborting() ); while( !m_testCaseTracker->isCompleted() && !aborting() );
} }
while( getCurrentContext().advanceGeneratorsForCurrentTest() && !aborting() ); while( getCurrentContext().advanceGeneratorsForCurrentTest() && !aborting() );
@ -4880,8 +4816,8 @@ namespace Catch {
missingAssertions, missingAssertions,
aborting() ) ); aborting() ) );
delete m_runningTest; m_activeTestCase = NULL;
m_runningTest = NULL; m_testCaseTracker.reset();
return deltaTotals; return deltaTotals;
} }
@ -4920,7 +4856,7 @@ namespace Catch {
std::ostringstream oss; std::ostringstream oss;
oss << sectionInfo.name << "@" << sectionInfo.lineInfo; oss << sectionInfo.name << "@" << sectionInfo.lineInfo;
if( !m_runningTest->addSection( oss.str() ) ) if( !m_testCaseTracker->enterSection( oss.str() ) )
return false; return false;
m_lastAssertionInfo.lineInfo = sectionInfo.lineInfo; m_lastAssertionInfo.lineInfo = sectionInfo.lineInfo;
@ -4942,13 +4878,13 @@ namespace Catch {
bool missingAssertions = false; bool missingAssertions = false;
if( assertions.total() == 0 && if( assertions.total() == 0 &&
m_config->warnAboutMissingAssertions() && m_config->warnAboutMissingAssertions() &&
!m_runningTest->isBranchSection() ) { !m_testCaseTracker->currentSectionHasChildren() ) {
m_totals.assertions.failed++; m_totals.assertions.failed++;
assertions.failed++; assertions.failed++;
missingAssertions = true; missingAssertions = true;
} }
m_runningTest->endSection( info.name, false ); m_testCaseTracker->leaveSection();
m_reporter->sectionEnded( SectionStats( info, assertions, missingAssertions ) ); m_reporter->sectionEnded( SectionStats( info, assertions, missingAssertions ) );
m_messages.clear(); m_messages.clear();
@ -4967,8 +4903,8 @@ namespace Catch {
} }
virtual std::string getCurrentTestName() const { virtual std::string getCurrentTestName() const {
return m_runningTest return m_activeTestCase
? m_runningTest->getTestCase().getTestCaseInfo().name ? m_activeTestCase->getTestCaseInfo().name
: ""; : "";
} }
@ -5001,19 +4937,21 @@ namespace Catch {
} }
void runCurrentTest( std::string& redirectedCout, std::string& redirectedCerr ) { void runCurrentTest( std::string& redirectedCout, std::string& redirectedCerr ) {
TestCaseInfo const& testCaseInfo = m_activeTestCase->getTestCaseInfo();
SectionInfo testCaseSection( testCaseInfo.name, testCaseInfo.description, testCaseInfo.lineInfo );
m_reporter->sectionStarting( testCaseSection );
try { try {
m_lastAssertionInfo = AssertionInfo( "TEST_CASE", m_runningTest->getTestCase().getTestCaseInfo().lineInfo, "", ResultDisposition::Normal ); m_lastAssertionInfo = AssertionInfo( "TEST_CASE", testCaseInfo.lineInfo, "", ResultDisposition::Normal );
m_runningTest->reset(); TestCaseTracker::Guard guard( *m_testCaseTracker );
if( m_reporter->getPreferences().shouldRedirectStdOut ) { if( m_reporter->getPreferences().shouldRedirectStdOut ) {
StreamRedirect coutRedir( std::cout, redirectedCout ); StreamRedirect coutRedir( std::cout, redirectedCout );
StreamRedirect cerrRedir( std::cerr, redirectedCerr ); StreamRedirect cerrRedir( std::cerr, redirectedCerr );
m_runningTest->getTestCase().invoke(); m_activeTestCase->invoke();
} }
else { else {
m_runningTest->getTestCase().invoke(); m_activeTestCase->invoke();
} }
m_runningTest->ranToCompletion();
} }
catch( TestFailureException& ) { catch( TestFailureException& ) {
// This just means the test was aborted due to failure // This just means the test was aborted due to failure
@ -5023,6 +4961,8 @@ namespace Catch {
exResult << translateActiveException(); exResult << translateActiveException();
actOnCurrentResult( exResult.buildResult( m_lastAssertionInfo ) ); actOnCurrentResult( exResult.buildResult( m_lastAssertionInfo ) );
} }
// If sections ended prematurely due to an exception we stored their
// infos here so we can tear them down outside the unwind process.
for( std::vector<UnfinishedSections>::const_iterator it = m_unfinishedSections.begin(), for( std::vector<UnfinishedSections>::const_iterator it = m_unfinishedSections.begin(),
itEnd = m_unfinishedSections.end(); itEnd = m_unfinishedSections.end();
it != itEnd; it != itEnd;
@ -5030,6 +4970,8 @@ namespace Catch {
sectionEnded( it->info, it->prevAssertions ); sectionEnded( it->info, it->prevAssertions );
m_unfinishedSections.clear(); m_unfinishedSections.clear();
m_messages.clear(); m_messages.clear();
SectionStats testCaseSectionStats( testCaseSection, Counts(), 0 ); // !TBD
m_reporter->sectionEnded( testCaseSectionStats );
} }
private: private:
@ -5044,7 +4986,8 @@ namespace Catch {
TestRunInfo m_runInfo; TestRunInfo m_runInfo;
IMutableContext& m_context; IMutableContext& m_context;
RunningTest* m_runningTest; TestCase const* m_activeTestCase;
Option<TestCaseTracker> m_testCaseTracker;
AssertionResult m_lastResult; AssertionResult m_lastResult;
Ptr<IConfig const> m_config; Ptr<IConfig const> m_config;
@ -5684,8 +5627,8 @@ namespace Catch {
std::map<std::string, IGeneratorsForTest*>::const_iterator it = std::map<std::string, IGeneratorsForTest*>::const_iterator it =
m_generatorsByTestName.find( testName ); m_generatorsByTestName.find( testName );
return it != m_generatorsByTestName.end() return it != m_generatorsByTestName.end()
? it->second ? it->second
: NULL; : NULL;
} }
IGeneratorsForTest& getGeneratorsForCurrentTest() { IGeneratorsForTest& getGeneratorsForCurrentTest() {
@ -6236,7 +6179,7 @@ namespace Catch {
namespace Catch { namespace Catch {
// These numbers are maintained by a script // These numbers are maintained by a script
Version libraryVersion( 1, 0, 5, "master" ); Version libraryVersion( 1, 0, 6, "master" );
} }
// #included from: catch_text.hpp // #included from: catch_text.hpp
@ -7070,7 +7013,7 @@ namespace Catch {
namespace Catch { namespace Catch {
class XmlReporter : public SharedImpl<IReporter> { class XmlReporter : public SharedImpl<IReporter> {
public: public:
XmlReporter( ReporterConfig const& config ) : m_config( config ) {} XmlReporter( ReporterConfig const& config ) : m_config( config ), m_sectionDepth( 0 ) {}
static std::string getDescription() { static std::string getDescription() {
return "Reports test results as an XML document"; return "Reports test results as an XML document";
@ -7110,18 +7053,22 @@ namespace Catch {
} }
virtual void StartSection( const std::string& sectionName, const std::string& description ) { virtual void StartSection( const std::string& sectionName, const std::string& description ) {
m_xml.startElement( "Section" ) if( m_sectionDepth++ > 0 ) {
.writeAttribute( "name", sectionName ) m_xml.startElement( "Section" )
.writeAttribute( "description", description ); .writeAttribute( "name", sectionName )
.writeAttribute( "description", description );
}
} }
virtual void NoAssertionsInSection( const std::string& ) {} virtual void NoAssertionsInSection( const std::string& ) {}
virtual void NoAssertionsInTestCase( const std::string& ) {} virtual void NoAssertionsInTestCase( const std::string& ) {}
virtual void EndSection( const std::string& /*sectionName*/, const Counts& assertions ) { virtual void EndSection( const std::string& /*sectionName*/, const Counts& assertions ) {
m_xml.scopedElement( "OverallResults" ) if( --m_sectionDepth > 0 ) {
.writeAttribute( "successes", assertions.passed ) m_xml.scopedElement( "OverallResults" )
.writeAttribute( "failures", assertions.failed ); .writeAttribute( "successes", assertions.passed )
m_xml.endElement(); .writeAttribute( "failures", assertions.failed );
m_xml.endElement();
}
} }
virtual void StartTestCase( const Catch::TestCaseInfo& testInfo ) { virtual void StartTestCase( const Catch::TestCaseInfo& testInfo ) {
@ -7192,6 +7139,7 @@ namespace Catch {
ReporterConfig m_config; ReporterConfig m_config;
bool m_currentTestSuccess; bool m_currentTestSuccess;
XmlWriter m_xml; XmlWriter m_xml;
int m_sectionDepth;
}; };
} // end namespace Catch } // end namespace Catch
@ -7696,12 +7644,10 @@ namespace Catch {
sections.push_back( section ); sections.push_back( section );
// Sections // Sections
if( !sections.empty() ) { std::vector<ThreadedSectionInfo*>::const_reverse_iterator
typedef std::vector<ThreadedSectionInfo*>::const_reverse_iterator It; it = sections.rbegin(), itEnd = sections.rend();
for( It it = sections.rbegin(), itEnd = sections.rend(); it != itEnd; ++it ) for( ++it; it != itEnd; ++it ) // Skip first section (test case)
printHeaderString( (*it)->name, 2 ); printHeaderString( (*it)->name, 2 );
}
} }
SourceLineInfo lineInfo = currentSectionInfo SourceLineInfo lineInfo = currentSectionInfo
? currentSectionInfo->lineInfo ? currentSectionInfo->lineInfo