2010-12-31 23:07:47 +01:00
|
|
|
/*
|
|
|
|
* Created by Phil on 31/12/2010.
|
|
|
|
* Copyright 2010 Two Blue Cubes Ltd. All rights reserved.
|
|
|
|
*
|
|
|
|
* Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
* file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
*/
|
2012-09-17 07:42:29 +02:00
|
|
|
#ifndef TWOBLUECUBES_CATCH_INTERFACES_REPORTER_H_INCLUDED
|
|
|
|
#define TWOBLUECUBES_CATCH_INTERFACES_REPORTER_H_INCLUDED
|
2010-12-31 23:07:47 +01:00
|
|
|
|
|
|
|
#include "catch_common.h"
|
2012-02-24 09:59:35 +01:00
|
|
|
#include "catch_totals.hpp"
|
2012-05-04 08:55:11 +02:00
|
|
|
#include "catch_ptr.hpp"
|
2012-08-28 09:20:18 +02:00
|
|
|
#include "catch_config.hpp"
|
2012-11-22 20:17:20 +01:00
|
|
|
#include "catch_test_case_info.h"
|
|
|
|
#include "catch_assertionresult.h"
|
2013-02-02 20:58:04 +01:00
|
|
|
#include "catch_message.h"
|
2012-12-05 09:40:53 +01:00
|
|
|
#include "catch_option.hpp"
|
2010-12-31 23:07:47 +01:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <ostream>
|
2010-12-31 23:21:36 +01:00
|
|
|
#include <map>
|
2010-12-31 23:07:47 +01:00
|
|
|
|
|
|
|
namespace Catch
|
|
|
|
{
|
2012-11-29 21:31:17 +01:00
|
|
|
struct ReporterConfig {
|
2013-05-28 19:39:32 +02:00
|
|
|
explicit ReporterConfig( Ptr<IConfig> const& _fullConfig )
|
|
|
|
: m_stream( &_fullConfig->stream() ), m_fullConfig( _fullConfig ) {}
|
|
|
|
|
|
|
|
ReporterConfig( Ptr<IConfig> const& _fullConfig, std::ostream& _stream )
|
2012-11-27 00:28:00 +01:00
|
|
|
: m_stream( &_stream ), m_fullConfig( _fullConfig ) {}
|
2012-09-20 09:17:52 +02:00
|
|
|
|
2013-05-28 19:39:32 +02:00
|
|
|
std::ostream& stream() const { return *m_stream; }
|
|
|
|
Ptr<IConfig> fullConfig() const { return m_fullConfig; }
|
2012-09-20 09:17:52 +02:00
|
|
|
|
|
|
|
private:
|
2012-11-27 00:28:00 +01:00
|
|
|
std::ostream* m_stream;
|
2013-05-28 19:39:32 +02:00
|
|
|
Ptr<IConfig> m_fullConfig;
|
2012-08-28 09:20:18 +02:00
|
|
|
};
|
2012-11-22 20:17:20 +01:00
|
|
|
|
2012-11-29 21:31:17 +01:00
|
|
|
struct ReporterPreferences {
|
2012-11-29 10:05:51 +01:00
|
|
|
ReporterPreferences()
|
|
|
|
: shouldRedirectStdOut( false )
|
|
|
|
{}
|
|
|
|
|
|
|
|
bool shouldRedirectStdOut;
|
|
|
|
};
|
|
|
|
|
2012-12-02 00:49:57 +01:00
|
|
|
struct TestRunInfo {
|
|
|
|
TestRunInfo( std::string const& _name ) : name( _name ) {}
|
|
|
|
std::string name;
|
|
|
|
};
|
|
|
|
struct GroupInfo {
|
2013-01-13 22:51:44 +01:00
|
|
|
GroupInfo( std::string const& _name,
|
|
|
|
std::size_t _groupIndex,
|
|
|
|
std::size_t _groupsCount )
|
|
|
|
: name( _name ),
|
|
|
|
groupIndex( _groupIndex ),
|
|
|
|
groupsCounts( _groupsCount )
|
|
|
|
{}
|
|
|
|
|
2012-12-02 00:49:57 +01:00
|
|
|
std::string name;
|
2013-01-13 22:51:44 +01:00
|
|
|
std::size_t groupIndex;
|
|
|
|
std::size_t groupsCounts;
|
2012-12-02 00:49:57 +01:00
|
|
|
};
|
|
|
|
|
2012-11-29 21:31:17 +01:00
|
|
|
struct SectionInfo {
|
|
|
|
SectionInfo( std::string const& _name,
|
|
|
|
std::string const& _description,
|
|
|
|
SourceLineInfo const& _lineInfo )
|
|
|
|
: name( _name ),
|
|
|
|
description( _description ),
|
2012-11-30 09:58:46 +01:00
|
|
|
lineInfo( _lineInfo )
|
2012-11-29 21:31:17 +01:00
|
|
|
{}
|
|
|
|
|
|
|
|
std::string name;
|
|
|
|
std::string description;
|
2012-11-30 09:58:46 +01:00
|
|
|
SourceLineInfo lineInfo;
|
2012-11-29 21:31:17 +01:00
|
|
|
};
|
|
|
|
|
2012-12-05 09:40:53 +01:00
|
|
|
struct ThreadedSectionInfo : SectionInfo, SharedImpl<> {
|
2013-01-26 21:17:52 +01:00
|
|
|
ThreadedSectionInfo( SectionInfo const& _sectionInfo, ThreadedSectionInfo* _parent = NULL )
|
2012-12-05 09:40:53 +01:00
|
|
|
: SectionInfo( _sectionInfo ),
|
2013-01-16 00:14:52 +01:00
|
|
|
parent( _parent )
|
2012-12-05 09:40:53 +01:00
|
|
|
{}
|
|
|
|
virtual ~ThreadedSectionInfo();
|
|
|
|
|
|
|
|
std::vector<Ptr<ThreadedSectionInfo> > children;
|
2013-01-26 21:17:52 +01:00
|
|
|
ThreadedSectionInfo* parent;
|
2012-12-05 09:40:53 +01:00
|
|
|
};
|
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
struct AssertionStats {
|
2012-11-29 21:31:17 +01:00
|
|
|
AssertionStats( AssertionResult const& _assertionResult,
|
2013-02-02 20:58:04 +01:00
|
|
|
std::vector<MessageInfo> const& _infoMessages,
|
2012-11-29 21:31:17 +01:00
|
|
|
Totals const& _totals )
|
2012-11-25 22:43:36 +01:00
|
|
|
: assertionResult( _assertionResult ),
|
2013-02-02 20:58:04 +01:00
|
|
|
infoMessages( _infoMessages ),
|
2012-11-25 22:43:36 +01:00
|
|
|
totals( _totals )
|
2013-02-02 20:58:04 +01:00
|
|
|
{
|
|
|
|
if( assertionResult.hasMessage() ) {
|
|
|
|
// Copy message into messages list.
|
|
|
|
// !TBD This should have been done earlier, somewhere
|
|
|
|
MessageBuilder builder( assertionResult.getTestMacroName(), assertionResult.getSourceInfo(), assertionResult.getResultType() );
|
|
|
|
builder << assertionResult.getMessage();
|
|
|
|
infoMessages.push_back( builder.build() );
|
|
|
|
}
|
|
|
|
}
|
2012-11-30 19:54:06 +01:00
|
|
|
virtual ~AssertionStats();
|
2012-11-25 22:43:36 +01:00
|
|
|
|
2012-11-22 20:17:20 +01:00
|
|
|
AssertionResult assertionResult;
|
2013-02-02 20:58:04 +01:00
|
|
|
std::vector<MessageInfo> infoMessages;
|
2012-11-22 20:17:20 +01:00
|
|
|
Totals totals;
|
|
|
|
};
|
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
struct SectionStats {
|
2012-11-30 09:58:46 +01:00
|
|
|
SectionStats( SectionInfo const& _sectionInfo,
|
|
|
|
Counts const& _assertions,
|
|
|
|
bool _missingAssertions )
|
|
|
|
: sectionInfo( _sectionInfo ),
|
|
|
|
assertions( _assertions ),
|
|
|
|
missingAssertions( _missingAssertions )
|
|
|
|
{}
|
2012-11-30 19:54:06 +01:00
|
|
|
virtual ~SectionStats();
|
|
|
|
|
2012-11-29 21:31:17 +01:00
|
|
|
SectionInfo sectionInfo;
|
2012-11-30 09:58:46 +01:00
|
|
|
Counts assertions;
|
|
|
|
bool missingAssertions;
|
2012-11-29 21:31:17 +01:00
|
|
|
};
|
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
struct TestCaseStats {
|
2012-11-29 21:31:17 +01:00
|
|
|
TestCaseStats( TestCaseInfo const& _testInfo,
|
|
|
|
Totals const& _totals,
|
|
|
|
std::string const& _stdOut,
|
|
|
|
std::string const& _stdErr,
|
2012-11-27 00:28:00 +01:00
|
|
|
bool _missingAssertions,
|
2012-11-25 22:43:36 +01:00
|
|
|
bool _aborting )
|
|
|
|
: testInfo( _testInfo ),
|
|
|
|
totals( _totals ),
|
|
|
|
stdOut( _stdOut ),
|
|
|
|
stdErr( _stdErr ),
|
2012-11-27 00:28:00 +01:00
|
|
|
missingAssertions( _missingAssertions ),
|
2012-11-25 22:43:36 +01:00
|
|
|
aborting( _aborting )
|
|
|
|
{}
|
2012-11-30 19:54:06 +01:00
|
|
|
virtual ~TestCaseStats();
|
2012-11-25 22:43:36 +01:00
|
|
|
|
|
|
|
TestCaseInfo testInfo;
|
2012-11-22 20:17:20 +01:00
|
|
|
Totals totals;
|
|
|
|
std::string stdOut;
|
|
|
|
std::string stdErr;
|
2012-11-27 00:28:00 +01:00
|
|
|
bool missingAssertions;
|
2012-11-22 20:17:20 +01:00
|
|
|
bool aborting;
|
|
|
|
};
|
2010-12-31 23:07:47 +01:00
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
struct TestGroupStats {
|
2012-12-02 00:49:57 +01:00
|
|
|
TestGroupStats( GroupInfo const& _groupInfo,
|
2012-11-29 21:31:17 +01:00
|
|
|
Totals const& _totals,
|
2012-11-25 22:43:36 +01:00
|
|
|
bool _aborting )
|
2012-12-02 00:49:57 +01:00
|
|
|
: groupInfo( _groupInfo ),
|
2012-11-25 22:43:36 +01:00
|
|
|
totals( _totals ),
|
|
|
|
aborting( _aborting )
|
|
|
|
{}
|
2013-01-03 10:04:46 +01:00
|
|
|
TestGroupStats( GroupInfo const& _groupInfo )
|
|
|
|
: groupInfo( _groupInfo ),
|
|
|
|
aborting( false )
|
|
|
|
{}
|
2012-11-30 19:54:06 +01:00
|
|
|
virtual ~TestGroupStats();
|
2012-11-25 22:43:36 +01:00
|
|
|
|
2012-12-02 00:49:57 +01:00
|
|
|
GroupInfo groupInfo;
|
2012-11-22 20:17:20 +01:00
|
|
|
Totals totals;
|
|
|
|
bool aborting;
|
|
|
|
};
|
2010-12-31 23:07:47 +01:00
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
struct TestRunStats {
|
2012-12-02 00:49:57 +01:00
|
|
|
TestRunStats( TestRunInfo const& _runInfo,
|
2012-11-29 21:31:17 +01:00
|
|
|
Totals const& _totals,
|
2012-11-25 22:43:36 +01:00
|
|
|
bool _aborting )
|
2012-12-02 00:49:57 +01:00
|
|
|
: runInfo( _runInfo ),
|
2012-11-25 22:43:36 +01:00
|
|
|
totals( _totals ),
|
|
|
|
aborting( _aborting )
|
|
|
|
{}
|
2013-01-03 10:04:46 +01:00
|
|
|
TestRunStats( TestRunStats const& _other )
|
|
|
|
: runInfo( _other.runInfo ),
|
|
|
|
totals( _other.totals ),
|
|
|
|
aborting( _other.aborting )
|
|
|
|
{}
|
2012-11-30 19:54:06 +01:00
|
|
|
virtual ~TestRunStats();
|
2013-01-03 10:04:46 +01:00
|
|
|
|
2012-12-02 00:49:57 +01:00
|
|
|
TestRunInfo runInfo;
|
2012-11-22 20:17:20 +01:00
|
|
|
Totals totals;
|
|
|
|
bool aborting;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct IStreamingReporter : IShared {
|
2012-11-25 22:43:36 +01:00
|
|
|
virtual ~IStreamingReporter();
|
2012-12-05 09:40:53 +01:00
|
|
|
|
2013-01-13 22:51:44 +01:00
|
|
|
// Implementing class must also provide the following static method:
|
2012-12-05 09:40:53 +01:00
|
|
|
// static std::string getDescription();
|
|
|
|
|
2012-11-29 10:05:51 +01:00
|
|
|
virtual ReporterPreferences getPreferences() const = 0;
|
2013-03-12 20:06:40 +01:00
|
|
|
|
|
|
|
virtual void noMatchingTestCases( std::string const& spec ) = 0;
|
2012-11-29 10:05:51 +01:00
|
|
|
|
2012-12-02 00:49:57 +01:00
|
|
|
virtual void testRunStarting( TestRunInfo const& testRunInfo ) = 0;
|
|
|
|
virtual void testGroupStarting( GroupInfo const& groupInfo ) = 0;
|
2012-11-22 20:17:20 +01:00
|
|
|
|
2012-11-29 21:31:17 +01:00
|
|
|
virtual void testCaseStarting( TestCaseInfo const& testInfo ) = 0;
|
2012-11-30 09:58:46 +01:00
|
|
|
virtual void sectionStarting( SectionInfo const& sectionInfo ) = 0;
|
|
|
|
|
2012-11-29 21:31:17 +01:00
|
|
|
virtual void assertionStarting( AssertionInfo const& assertionInfo ) = 0;
|
2012-11-22 20:17:20 +01:00
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
virtual void assertionEnded( AssertionStats const& assertionStats ) = 0;
|
|
|
|
virtual void sectionEnded( SectionStats const& sectionStats ) = 0;
|
|
|
|
virtual void testCaseEnded( TestCaseStats const& testCaseStats ) = 0;
|
|
|
|
virtual void testGroupEnded( TestGroupStats const& testGroupStats ) = 0;
|
|
|
|
virtual void testRunEnded( TestRunStats const& testRunStats ) = 0;
|
2012-11-22 20:17:20 +01:00
|
|
|
};
|
2012-12-05 09:40:53 +01:00
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
struct StreamingReporterBase : SharedImpl<IStreamingReporter> {
|
|
|
|
|
|
|
|
StreamingReporterBase( ReporterConfig const& _config )
|
2013-05-28 19:39:32 +02:00
|
|
|
: m_config( _config.fullConfig() ),
|
2012-12-09 12:20:46 +01:00
|
|
|
stream( _config.stream() )
|
2012-12-05 09:40:53 +01:00
|
|
|
{}
|
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
virtual ~StreamingReporterBase();
|
2013-03-12 20:06:40 +01:00
|
|
|
|
|
|
|
virtual void noMatchingTestCases( std::string const& ) {}
|
2012-12-05 09:40:53 +01:00
|
|
|
|
|
|
|
virtual void testRunStarting( TestRunInfo const& _testRunInfo ) {
|
|
|
|
testRunInfo = _testRunInfo;
|
|
|
|
}
|
|
|
|
virtual void testGroupStarting( GroupInfo const& _groupInfo ) {
|
|
|
|
unusedGroupInfo = _groupInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void testCaseStarting( TestCaseInfo const& _testInfo ) {
|
|
|
|
unusedTestCaseInfo = _testInfo;
|
|
|
|
}
|
|
|
|
virtual void sectionStarting( SectionInfo const& _sectionInfo ) {
|
|
|
|
Ptr<ThreadedSectionInfo> sectionInfo = new ThreadedSectionInfo( _sectionInfo );
|
|
|
|
if( !currentSectionInfo ) {
|
|
|
|
currentSectionInfo = sectionInfo;
|
2013-01-26 21:17:52 +01:00
|
|
|
m_rootSections.push_back( currentSectionInfo );
|
2012-12-05 09:40:53 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
currentSectionInfo->children.push_back( sectionInfo );
|
2013-01-26 21:17:52 +01:00
|
|
|
sectionInfo->parent = currentSectionInfo.get();
|
2012-12-05 09:40:53 +01:00
|
|
|
currentSectionInfo = sectionInfo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
virtual void sectionEnded( SectionStats const& /* _sectionStats */ ) {
|
2012-12-05 09:40:53 +01:00
|
|
|
currentSectionInfo = currentSectionInfo->parent;
|
|
|
|
}
|
2013-01-03 10:04:46 +01:00
|
|
|
virtual void testCaseEnded( TestCaseStats const& /* _testCaseStats */ ) {
|
2012-12-05 09:40:53 +01:00
|
|
|
unusedTestCaseInfo.reset();
|
|
|
|
}
|
2013-01-03 10:04:46 +01:00
|
|
|
virtual void testGroupEnded( TestGroupStats const& /* _testGroupStats */ ) {
|
2012-12-05 09:40:53 +01:00
|
|
|
unusedGroupInfo.reset();
|
|
|
|
}
|
2013-01-03 10:04:46 +01:00
|
|
|
virtual void testRunEnded( TestRunStats const& /* _testRunStats */ ) {
|
2012-12-14 19:28:03 +01:00
|
|
|
currentSectionInfo.reset();
|
|
|
|
unusedTestCaseInfo.reset();
|
|
|
|
unusedGroupInfo.reset();
|
|
|
|
testRunInfo.reset();
|
2012-12-05 09:40:53 +01:00
|
|
|
}
|
|
|
|
|
2013-05-28 19:39:32 +02:00
|
|
|
Ptr<IConfig> m_config;
|
2012-12-05 09:40:53 +01:00
|
|
|
Option<TestRunInfo> testRunInfo;
|
|
|
|
Option<GroupInfo> unusedGroupInfo;
|
|
|
|
Option<TestCaseInfo> unusedTestCaseInfo;
|
|
|
|
Ptr<ThreadedSectionInfo> currentSectionInfo;
|
|
|
|
std::ostream& stream;
|
2013-01-26 21:17:52 +01:00
|
|
|
|
|
|
|
// !TBD: This should really go in the TestCaseStats class
|
|
|
|
std::vector<Ptr<ThreadedSectionInfo> > m_rootSections;
|
2012-12-05 09:40:53 +01:00
|
|
|
};
|
|
|
|
|
2013-01-03 10:04:46 +01:00
|
|
|
struct TestGroupNode : TestGroupStats {
|
|
|
|
TestGroupNode( TestGroupStats const& _stats ) : TestGroupStats( _stats ) {}
|
|
|
|
// TestGroupNode( GroupInfo const& _info ) : TestGroupStats( _stats ) {}
|
|
|
|
~TestGroupNode();
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TestRunNode : TestRunStats {
|
|
|
|
|
|
|
|
TestRunNode( TestRunStats const& _stats ) : TestRunStats( _stats ) {}
|
|
|
|
~TestRunNode();
|
|
|
|
|
|
|
|
std::vector<TestGroupNode> groups;
|
|
|
|
};
|
|
|
|
|
2012-11-30 19:54:06 +01:00
|
|
|
// Deprecated
|
2012-05-09 09:17:51 +02:00
|
|
|
struct IReporter : IShared {
|
2012-08-13 08:46:10 +02:00
|
|
|
virtual ~IReporter();
|
2012-10-16 09:27:21 +02:00
|
|
|
|
|
|
|
virtual bool shouldRedirectStdout() const = 0;
|
|
|
|
|
2012-05-09 09:17:51 +02:00
|
|
|
virtual void StartTesting() = 0;
|
2012-11-30 19:54:06 +01:00
|
|
|
virtual void EndTesting( Totals const& totals ) = 0;
|
2012-11-29 21:31:17 +01:00
|
|
|
virtual void StartGroup( std::string const& groupName ) = 0;
|
2012-11-30 19:54:06 +01:00
|
|
|
virtual void EndGroup( std::string const& groupName, Totals const& totals ) = 0;
|
2012-11-29 21:31:17 +01:00
|
|
|
virtual void StartTestCase( TestCaseInfo const& testInfo ) = 0;
|
|
|
|
virtual void EndTestCase( TestCaseInfo const& testInfo, Totals const& totals, std::string const& stdOut, std::string const& stdErr ) = 0;
|
|
|
|
virtual void StartSection( std::string const& sectionName, std::string const& description ) = 0;
|
2013-04-23 19:58:56 +02:00
|
|
|
virtual void EndSection( std::string const& sectionName, Counts const& assertions ) = 0;
|
2012-11-29 21:31:17 +01:00
|
|
|
virtual void NoAssertionsInSection( std::string const& sectionName ) = 0;
|
|
|
|
virtual void NoAssertionsInTestCase( std::string const& testName ) = 0;
|
2012-06-01 20:40:27 +02:00
|
|
|
virtual void Aborted() = 0;
|
2013-04-23 19:58:56 +02:00
|
|
|
virtual void Result( AssertionResult const& result ) = 0;
|
2010-12-31 23:07:47 +01:00
|
|
|
};
|
2012-11-25 22:43:36 +01:00
|
|
|
|
2010-12-31 23:07:47 +01:00
|
|
|
|
2012-05-09 09:17:51 +02:00
|
|
|
struct IReporterFactory {
|
2012-08-13 08:46:10 +02:00
|
|
|
virtual ~IReporterFactory();
|
2012-11-30 20:15:23 +01:00
|
|
|
virtual IStreamingReporter* create( ReporterConfig const& config ) const = 0;
|
2012-05-09 09:17:51 +02:00
|
|
|
virtual std::string getDescription() const = 0;
|
2010-12-31 23:07:47 +01:00
|
|
|
};
|
|
|
|
|
2012-05-09 09:17:51 +02:00
|
|
|
struct IReporterRegistry {
|
2010-12-31 23:21:36 +01:00
|
|
|
typedef std::map<std::string, IReporterFactory*> FactoryMap;
|
|
|
|
|
2012-08-13 08:46:10 +02:00
|
|
|
virtual ~IReporterRegistry();
|
2013-05-28 19:39:32 +02:00
|
|
|
virtual IStreamingReporter* create( std::string const& name, Ptr<IConfig> const& config ) const = 0;
|
2013-04-23 19:58:56 +02:00
|
|
|
virtual FactoryMap const& getFactories() const = 0;
|
2010-12-31 23:07:47 +01:00
|
|
|
};
|
|
|
|
|
2012-11-29 21:31:17 +01:00
|
|
|
inline std::string trim( std::string const& str ) {
|
2010-12-31 23:07:47 +01:00
|
|
|
std::string::size_type start = str.find_first_not_of( "\n\r\t " );
|
|
|
|
std::string::size_type end = str.find_last_not_of( "\n\r\t " );
|
|
|
|
|
2012-02-28 21:04:25 +01:00
|
|
|
return start != std::string::npos ? str.substr( start, 1+end-start ) : "";
|
2010-12-31 23:07:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-17 07:42:29 +02:00
|
|
|
#endif // TWOBLUECUBES_CATCH_INTERFACES_REPORTER_H_INCLUDED
|