2011-05-24 09:23:02 +02:00
/*
2013-07-02 09:49:29 +02:00
* CATCH v1 .0 build 5 ( master branch )
* Generated : 2013 - 07 - 02 08 : 47 : 24.946227
2012-05-22 23:22:22 +02:00
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
2012-05-09 20:37:51 +02:00
* This file has been merged from multiple headers . Please don ' t edit it directly
* Copyright ( c ) 2012 Two Blue Cubes Ltd . All rights reserved .
2011-05-24 09:23:02 +02:00
*
* 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-05-09 20:37:51 +02:00
# ifndef TWOBLUECUBES_SINGLE_INCLUDE_CATCH_HPP_INCLUDED
# define TWOBLUECUBES_SINGLE_INCLUDE_CATCH_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_HPP_INCLUDED
2012-06-03 00:12:42 +02:00
2012-08-16 19:48:50 +02:00
# ifdef __clang__
2012-11-01 09:27:09 +01:00
# pragma clang diagnostic ignored "-Wglobal-constructors"
2013-03-16 21:21:51 +01:00
# pragma clang diagnostic ignored "-Wvariadic-macros"
2012-08-16 19:48:50 +02:00
2012-08-13 08:46:10 +02:00
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Wpadded"
2012-08-16 19:48:50 +02:00
# endif
2012-08-13 08:46:10 +02:00
2012-08-09 08:47:30 +02:00
// #included from: internal/catch_notimplemented_exception.h
# define TWOBLUECUBES_CATCH_NOTIMPLEMENTED_EXCEPTION_H_INCLUDED
2011-05-24 09:23:02 +02:00
// #included from: catch_common.h
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_COMMON_H_INCLUDED
2011-05-24 09:23:02 +02:00
# define INTERNAL_CATCH_UNIQUE_NAME_LINE2( name, line ) name##line
# define INTERNAL_CATCH_UNIQUE_NAME_LINE( name, line ) INTERNAL_CATCH_UNIQUE_NAME_LINE2( name, line )
# define INTERNAL_CATCH_UNIQUE_NAME( name ) INTERNAL_CATCH_UNIQUE_NAME_LINE( name, __LINE__ )
# define INTERNAL_CATCH_STRINGIFY2( expr ) #expr
# define INTERNAL_CATCH_STRINGIFY( expr ) INTERNAL_CATCH_STRINGIFY2( expr )
# include <sstream>
# include <stdexcept>
# include <algorithm>
2013-04-24 20:10:02 +02:00
// #included from: catch_compiler_capabilities.h
# define TWOBLUECUBES_CATCH_COMPILER_CAPABILITIES_HPP_INCLUDED
// Much of the following code is based on Boost (1.53)
////////////////////////////////////////////////////////////////////////////////
// Borland
# ifdef __BORLANDC__
# if (__BORLANDC__ > 0x582 )
//#define CATCH_CONFIG_SFINAE // Not confirmed
# endif
# endif // __BORLANDC__
////////////////////////////////////////////////////////////////////////////////
// EDG
# ifdef __EDG_VERSION__
# if (__EDG_VERSION__ > 238 )
//#define CATCH_CONFIG_SFINAE // Not confirmed
# endif
# endif // __EDG_VERSION__
////////////////////////////////////////////////////////////////////////////////
// Digital Mars
# ifdef __DMC__
# if (__DMC__ > 0x840 )
//#define CATCH_CONFIG_SFINAE // Not confirmed
# endif
# endif // __DMC__
////////////////////////////////////////////////////////////////////////////////
// GCC
# ifdef __GNUC__
# if __GNUC__ < 3
# if (__GNUC_MINOR__ >= 96 )
//#define CATCH_CONFIG_SFINAE
# endif
# elif __GNUC__ >= 3
// #define CATCH_CONFIG_SFINAE // Taking this out completely for now
# endif // __GNUC__ < 3
# endif // __GNUC__
////////////////////////////////////////////////////////////////////////////////
// Visual C++
# ifdef _MSC_VER
# if (_MSC_VER >= 1310 ) // (VC++ 7.0+)
//#define CATCH_CONFIG_SFINAE // Not confirmed
# endif
# endif // _MSC_VER
// Use variadic macros if the compiler supports them
# if ( defined _MSC_VER && _MSC_VER > 1400 && !defined __EDGE__) || \
( defined __WAVE__ & & __WAVE_HAS_VARIADICS ) | | \
( defined __GNUC__ & & __GNUC__ > = 3 ) | | \
( ! defined __cplusplus & & __STDC_VERSION__ > = 199901L | | __cplusplus > = 201103L )
2013-06-07 20:07:50 +02:00
# ifndef CATCH_CONFIG_NO_VARIADIC_MACROS
# define CATCH_CONFIG_VARIADIC_MACROS
# endif
2013-04-24 20:10:02 +02:00
# endif
2012-05-16 16:09:17 +02:00
namespace Catch {
2013-03-08 10:30:25 +01:00
class NonCopyable {
2013-04-24 20:10:02 +02:00
NonCopyable ( NonCopyable const & ) ;
void operator = ( NonCopyable const & ) ;
2013-03-08 10:30:25 +01:00
protected :
NonCopyable ( ) { }
virtual ~ NonCopyable ( ) ;
} ;
2011-05-24 09:23:02 +02:00
2012-05-31 20:40:26 +02:00
class SafeBool {
public :
typedef void ( SafeBool : : * type ) ( ) const ;
static type makeSafe ( bool value ) {
return value ? & SafeBool : : trueValue : 0 ;
}
private :
void trueValue ( ) const { }
} ;
2011-05-24 09:23:02 +02:00
template < typename ContainerT >
2012-05-09 20:04:00 +02:00
inline void deleteAll ( ContainerT & container ) {
2011-05-24 09:23:02 +02:00
typename ContainerT : : const_iterator it = container . begin ( ) ;
typename ContainerT : : const_iterator itEnd = container . end ( ) ;
for ( ; it ! = itEnd ; + + it )
delete * it ;
}
template < typename AssociativeContainerT >
2012-05-09 20:04:00 +02:00
inline void deleteAllValues ( AssociativeContainerT & container ) {
2011-05-24 09:23:02 +02:00
typename AssociativeContainerT : : const_iterator it = container . begin ( ) ;
typename AssociativeContainerT : : const_iterator itEnd = container . end ( ) ;
for ( ; it ! = itEnd ; + + it )
delete it - > second ;
}
template < typename ContainerT , typename Function >
2012-05-09 20:04:00 +02:00
inline void forEach ( ContainerT & container , Function function ) {
2011-05-24 09:23:02 +02:00
std : : for_each ( container . begin ( ) , container . end ( ) , function ) ;
}
template < typename ContainerT , typename Function >
2013-04-24 20:10:02 +02:00
inline void forEach ( ContainerT const & container , Function function ) {
2011-05-24 09:23:02 +02:00
std : : for_each ( container . begin ( ) , container . end ( ) , function ) ;
}
2013-04-24 20:10:02 +02:00
inline bool startsWith ( std : : string const & s , std : : string const & prefix ) {
2012-08-23 21:08:50 +02:00
return s . size ( ) > = prefix . size ( ) & & s . substr ( 0 , prefix . size ( ) ) = = prefix ;
}
2013-04-24 20:10:02 +02:00
inline bool endsWith ( std : : string const & s , std : : string const & suffix ) {
2012-08-24 09:23:50 +02:00
return s . size ( ) > = suffix . size ( ) & & s . substr ( s . size ( ) - suffix . size ( ) , suffix . size ( ) ) = = suffix ;
}
2013-04-24 20:10:02 +02:00
inline bool contains ( std : : string const & s , std : : string const & infix ) {
2012-08-24 09:23:50 +02:00
return s . find ( infix ) ! = std : : string : : npos ;
}
2013-03-25 10:25:31 +01:00
inline void toLowerInPlace ( std : : string & s ) {
2013-03-12 19:49:37 +01:00
std : : transform ( s . begin ( ) , s . end ( ) , s . begin ( ) , : : tolower ) ;
}
2013-03-25 10:25:31 +01:00
inline std : : string toLower ( std : : string const & s ) {
std : : string lc = s ;
toLowerInPlace ( lc ) ;
return lc ;
}
2012-08-24 09:23:50 +02:00
struct pluralise {
2013-04-24 20:10:02 +02:00
pluralise ( std : : size_t count , std : : string const & label )
2012-08-24 09:23:50 +02:00
: m_count ( count ) ,
m_label ( label )
{ }
2013-04-24 20:10:02 +02:00
friend std : : ostream & operator < < ( std : : ostream & os , pluralise const & pluraliser ) {
2012-08-24 09:23:50 +02:00
os < < pluraliser . m_count < < " " < < pluraliser . m_label ;
if ( pluraliser . m_count ! = 1 )
2012-11-13 23:04:29 +01:00
os < < " s " ;
2012-08-24 09:23:50 +02:00
return os ;
}
std : : size_t m_count ;
std : : string m_label ;
} ;
2012-08-23 21:08:50 +02:00
2012-05-16 16:09:17 +02:00
struct SourceLineInfo {
2012-05-09 20:04:00 +02:00
SourceLineInfo ( ) : line ( 0 ) { }
2013-04-24 20:10:02 +02:00
SourceLineInfo ( std : : string const & _file , std : : size_t _line )
2012-05-11 20:06:43 +02:00
: file ( _file ) ,
line ( _line )
2012-02-15 19:37:21 +01:00
{ }
2013-04-24 20:10:02 +02:00
SourceLineInfo ( SourceLineInfo const & other )
2012-05-09 20:04:00 +02:00
: file ( other . file ) ,
line ( other . line )
{ }
2012-11-01 09:27:09 +01:00
bool empty ( ) const {
return file . empty ( ) ;
}
2012-02-15 19:37:21 +01:00
std : : string file ;
std : : size_t line ;
} ;
2013-04-24 20:10:02 +02:00
inline std : : ostream & operator < < ( std : : ostream & os , SourceLineInfo const & info ) {
2012-02-15 19:37:21 +01:00
# ifndef __GNUG__
2013-04-01 12:27:10 +02:00
os < < info . file < < " ( " < < info . line < < " ) " ;
2012-02-15 19:37:21 +01:00
# else
2013-01-16 10:44:43 +01:00
os < < info . file < < " : " < < info . line ;
2012-02-15 19:37:21 +01:00
# endif
return os ;
}
2013-04-24 20:10:02 +02:00
// This is just here to avoid compiler warnings with macro constants and boolean literals
inline bool isTrue ( bool value ) { return value ; }
inline void throwLogicError ( std : : string const & message , SourceLineInfo const & locationInfo ) {
2011-05-24 09:23:02 +02:00
std : : ostringstream oss ;
2013-01-16 10:44:43 +01:00
oss < < locationInfo < < " : Internal Catch error: ' " < < message < < " ' " ;
2013-04-24 20:10:02 +02:00
if ( isTrue ( true ) )
throw std : : logic_error ( oss . str ( ) ) ;
2011-05-24 09:23:02 +02:00
}
}
2012-08-13 08:46:10 +02:00
# define CATCH_INTERNAL_LINEINFO ::Catch::SourceLineInfo( __FILE__, static_cast<std::size_t>( __LINE__ ) )
# define CATCH_INTERNAL_ERROR( msg ) ::Catch::throwLogicError( msg, CATCH_INTERNAL_LINEINFO );
2011-05-24 09:23:02 +02:00
2012-08-09 08:47:30 +02:00
# include <ostream>
namespace Catch {
class NotImplementedException : public std : : exception
{
public :
2013-04-24 20:10:02 +02:00
NotImplementedException ( SourceLineInfo const & lineInfo ) ;
2012-08-09 08:47:30 +02:00
virtual ~ NotImplementedException ( ) throw ( ) { }
virtual const char * what ( ) const throw ( ) ;
private :
std : : string m_what ;
SourceLineInfo m_lineInfo ;
} ;
} // end namespace Catch
///////////////////////////////////////////////////////////////////////////////
# define CATCH_NOT_IMPLEMENTED throw Catch::NotImplementedException( CATCH_INTERNAL_LINEINFO )
// #included from: internal/catch_context.h
# define TWOBLUECUBES_CATCH_CONTEXT_H_INCLUDED
2012-08-31 09:10:36 +02:00
// #included from: catch_interfaces_generators.h
# define TWOBLUECUBES_CATCH_INTERFACES_GENERATORS_H_INCLUDED
2012-08-09 08:47:30 +02:00
2012-08-31 09:10:36 +02:00
# include <string>
2012-02-25 10:39:45 +01:00
2012-05-16 16:09:17 +02:00
namespace Catch {
2012-08-31 09:10:36 +02:00
struct IGeneratorInfo {
virtual ~ IGeneratorInfo ( ) ;
virtual bool moveNext ( ) = 0 ;
virtual std : : size_t getCurrentIndex ( ) const = 0 ;
} ;
2012-02-25 10:39:45 +01:00
2012-08-31 09:10:36 +02:00
struct IGeneratorsForTest {
virtual ~ IGeneratorsForTest ( ) ;
2012-05-11 09:17:16 +02:00
2013-04-24 20:10:02 +02:00
virtual IGeneratorInfo & getGeneratorInfo ( std : : string const & fileInfo , std : : size_t size ) = 0 ;
2012-08-31 09:10:36 +02:00
virtual bool moveNext ( ) = 0 ;
2012-02-25 10:39:45 +01:00
} ;
2012-08-31 09:10:36 +02:00
IGeneratorsForTest * createGeneratorsForTest ( ) ;
2012-05-16 16:09:17 +02:00
2012-08-31 09:10:36 +02:00
} // end namespace Catch
2012-02-25 10:39:45 +01:00
2012-05-05 20:35:35 +02:00
// #included from: catch_ptr.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_PTR_HPP_INCLUDED
2012-05-05 20:35:35 +02:00
2013-03-13 13:19:30 +01:00
# ifdef __clang__
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Wpadded"
# endif
2012-05-16 16:09:17 +02:00
namespace Catch {
2012-05-05 20:35:35 +02:00
// An intrusive reference counting smart pointer.
// T must implement addRef() and release() methods
// typically implementing the IShared interface
template < typename T >
2012-05-16 16:09:17 +02:00
class Ptr {
2012-05-05 20:35:35 +02:00
public :
Ptr ( ) : m_p ( NULL ) { }
Ptr ( T * p ) : m_p ( p ) {
2012-07-28 21:37:07 +02:00
if ( m_p )
m_p - > addRef ( ) ;
2012-05-05 20:35:35 +02:00
}
2012-12-02 00:57:18 +01:00
Ptr ( Ptr const & other ) : m_p ( other . m_p ) {
2012-07-28 21:37:07 +02:00
if ( m_p )
m_p - > addRef ( ) ;
2012-05-05 20:35:35 +02:00
}
~ Ptr ( ) {
if ( m_p )
m_p - > release ( ) ;
}
2012-12-06 09:44:51 +01:00
void reset ( ) {
if ( m_p )
m_p - > release ( ) ;
m_p = NULL ;
}
2012-05-05 20:35:35 +02:00
Ptr & operator = ( T * p ) {
Ptr temp ( p ) ;
swap ( temp ) ;
return * this ;
}
2012-12-02 00:57:18 +01:00
Ptr & operator = ( Ptr const & other ) {
2012-05-05 20:35:35 +02:00
Ptr temp ( other ) ;
swap ( temp ) ;
return * this ;
}
2012-12-02 00:57:18 +01:00
void swap ( Ptr & other ) { std : : swap ( m_p , other . m_p ) ; }
T * get ( ) { return m_p ; }
const T * get ( ) const { return m_p ; }
T & operator * ( ) const { return * m_p ; }
T * operator - > ( ) const { return m_p ; }
bool operator ! ( ) const { return m_p = = NULL ; }
2012-12-06 09:44:51 +01:00
operator SafeBool : : type ( ) const { return SafeBool : : makeSafe ( m_p ! = NULL ) ; }
2012-05-05 20:35:35 +02:00
private :
T * m_p ;
} ;
struct IShared : NonCopyable {
2012-08-13 08:46:10 +02:00
virtual ~ IShared ( ) ;
2012-12-02 00:57:18 +01:00
virtual void addRef ( ) const = 0 ;
virtual void release ( ) const = 0 ;
2012-05-05 20:35:35 +02:00
} ;
2012-12-02 00:57:18 +01:00
template < typename T = IShared >
2012-05-05 20:35:35 +02:00
struct SharedImpl : T {
SharedImpl ( ) : m_rc ( 0 ) { }
2012-12-02 00:57:18 +01:00
virtual void addRef ( ) const {
2012-05-05 20:35:35 +02:00
+ + m_rc ;
}
2012-12-02 00:57:18 +01:00
virtual void release ( ) const {
2012-05-05 20:35:35 +02:00
if ( - - m_rc = = 0 )
delete this ;
}
2012-12-02 00:57:18 +01:00
mutable unsigned int m_rc ;
2012-05-05 20:35:35 +02:00
} ;
} // end namespace Catch
2013-03-13 13:19:30 +01:00
# ifdef __clang__
# pragma clang diagnostic pop
# endif
2013-06-07 20:07:50 +02:00
# include <memory>
# include <vector>
# include <stdlib.h>
namespace Catch {
class TestCase ;
class Stream ;
struct IResultCapture ;
struct IRunner ;
struct IGeneratorsForTest ;
struct IConfig ;
struct IContext
{
virtual ~ IContext ( ) ;
virtual IResultCapture & getResultCapture ( ) = 0 ;
virtual IRunner & getRunner ( ) = 0 ;
virtual size_t getGeneratorIndex ( std : : string const & fileInfo , size_t totalSize ) = 0 ;
virtual bool advanceGeneratorsForCurrentTest ( ) = 0 ;
virtual Ptr < IConfig const > getConfig ( ) const = 0 ;
} ;
struct IMutableContext : IContext
{
virtual ~ IMutableContext ( ) ;
virtual void setResultCapture ( IResultCapture * resultCapture ) = 0 ;
virtual void setRunner ( IRunner * runner ) = 0 ;
virtual void setConfig ( Ptr < IConfig const > const & config ) = 0 ;
} ;
IContext & getCurrentContext ( ) ;
IMutableContext & getCurrentMutableContext ( ) ;
void cleanUpContext ( ) ;
Stream createStream ( std : : string const & streamName ) ;
}
// #included from: internal/catch_test_registry.hpp
# define TWOBLUECUBES_CATCH_TEST_REGISTRY_HPP_INCLUDED
// #included from: catch_interfaces_testcase.h
# define TWOBLUECUBES_CATCH_INTERFACES_TESTCASE_H_INCLUDED
2012-06-03 00:12:42 +02:00
# include <vector>
namespace Catch {
2012-08-23 21:08:50 +02:00
class TestCaseFilters ;
2012-08-16 19:48:50 +02:00
struct ITestCase : IShared {
2012-06-03 00:12:42 +02:00
virtual void invoke ( ) const = 0 ;
2012-08-16 19:48:50 +02:00
protected :
virtual ~ ITestCase ( ) ;
2012-06-03 00:12:42 +02:00
} ;
2012-12-02 00:57:18 +01:00
class TestCase ;
2012-06-03 00:12:42 +02:00
struct ITestCaseRegistry {
2012-08-13 08:46:10 +02:00
virtual ~ ITestCaseRegistry ( ) ;
2013-04-24 20:10:02 +02:00
virtual std : : vector < TestCase > const & getAllTests ( ) const = 0 ;
virtual std : : vector < TestCase > getMatchingTestCases ( std : : string const & rawTestSpec ) const = 0 ;
2012-06-03 00:12:42 +02:00
} ;
}
2012-05-16 16:09:17 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
template < typename C >
2012-08-16 19:48:50 +02:00
class MethodTestCase : public SharedImpl < ITestCase > {
2012-05-16 16:09:17 +02:00
2012-05-11 09:17:16 +02:00
public :
MethodTestCase ( void ( C : : * method ) ( ) ) : m_method ( method ) { }
2011-05-24 09:23:02 +02:00
2012-05-11 09:17:16 +02:00
virtual void invoke ( ) const {
2011-05-24 09:23:02 +02:00
C obj ;
( obj . * m_method ) ( ) ;
}
private :
2012-08-16 19:48:50 +02:00
virtual ~ MethodTestCase ( ) { }
2011-05-24 09:23:02 +02:00
void ( C : : * m_method ) ( ) ;
} ;
typedef void ( * TestFunction ) ( ) ;
2013-03-16 21:21:51 +01:00
struct NameAndDesc {
NameAndDesc ( const char * _name = " " , const char * _description = " " )
: name ( _name ) , description ( _description )
{ }
const char * name ;
const char * description ;
} ;
2012-05-16 16:09:17 +02:00
struct AutoReg {
2012-05-11 09:17:16 +02:00
AutoReg ( TestFunction function ,
2013-03-16 21:21:51 +01:00
SourceLineInfo const & lineInfo ,
NameAndDesc const & nameAndDesc ) ;
2011-05-24 09:23:02 +02:00
template < typename C >
2012-05-11 09:17:16 +02:00
AutoReg ( void ( C : : * method ) ( ) ,
2013-03-16 21:21:51 +01:00
char const * className ,
NameAndDesc const & nameAndDesc ,
SourceLineInfo const & lineInfo ) {
registerTestCase ( new MethodTestCase < C > ( method ) ,
className ,
nameAndDesc ,
lineInfo ) ;
2011-05-24 09:23:02 +02:00
}
2012-05-11 09:17:16 +02:00
void registerTestCase ( ITestCase * testCase ,
2013-03-16 21:21:51 +01:00
char const * className ,
NameAndDesc const & nameAndDesc ,
SourceLineInfo const & lineInfo ) ;
2011-05-24 09:23:02 +02:00
2012-05-11 09:17:16 +02:00
~ AutoReg ( ) ;
2011-05-24 09:23:02 +02:00
private :
2013-04-24 20:10:02 +02:00
AutoReg ( AutoReg const & ) ;
void operator = ( AutoReg const & ) ;
2011-05-24 09:23:02 +02:00
} ;
} // end namespace Catch
2013-03-16 21:21:51 +01:00
# ifdef CATCH_CONFIG_VARIADIC_MACROS
///////////////////////////////////////////////////////////////////////////////
# define INTERNAL_CATCH_TESTCASE( ... ) \
2013-06-07 20:07:50 +02:00
static void INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) ( ) ; \
namespace { Catch : : AutoReg INTERNAL_CATCH_UNIQUE_NAME ( autoRegistrar ) ( & INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) , CATCH_INTERNAL_LINEINFO , Catch : : NameAndDesc ( __VA_ARGS__ ) ) ; } \
static void INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) ( )
2013-03-16 21:21:51 +01:00
///////////////////////////////////////////////////////////////////////////////
# define INTERNAL_CATCH_METHOD_AS_TEST_CASE( QualifiedMethod, ... ) \
namespace { Catch : : AutoReg INTERNAL_CATCH_UNIQUE_NAME ( autoRegistrar ) ( & QualifiedMethod , " & " # QualifiedMethod , Catch : : NameAndDesc ( __VA_ARGS__ ) , CATCH_INTERNAL_LINEINFO ) ; }
///////////////////////////////////////////////////////////////////////////////
2013-06-07 20:07:50 +02:00
# define INTERNAL_CATCH_TEST_CASE_METHOD( ClassName, ... )\
2013-03-16 21:21:51 +01:00
namespace { \
2013-06-07 20:07:50 +02:00
struct INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) : ClassName { \
2013-03-16 21:21:51 +01:00
void test ( ) ; \
} ; \
2013-06-07 20:07:50 +02:00
Catch : : AutoReg INTERNAL_CATCH_UNIQUE_NAME ( autoRegistrar ) ( & INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) : : test , # ClassName , Catch : : NameAndDesc ( __VA_ARGS__ ) , CATCH_INTERNAL_LINEINFO ) ; \
2013-03-16 21:21:51 +01:00
} \
2013-06-07 20:07:50 +02:00
void INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) : : test ( )
2011-05-24 09:23:02 +02:00
2013-03-16 21:21:51 +01:00
# else
///////////////////////////////////////////////////////////////////////////////
# define INTERNAL_CATCH_TESTCASE( Name, Desc ) \
2013-06-07 20:07:50 +02:00
static void INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) ( ) ; \
namespace { Catch : : AutoReg INTERNAL_CATCH_UNIQUE_NAME ( autoRegistrar ) ( & INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) , CATCH_INTERNAL_LINEINFO , Catch : : NameAndDesc ( Name , Desc ) ) ; } \
static void INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) ( )
2013-03-16 21:21:51 +01:00
///////////////////////////////////////////////////////////////////////////////
# define INTERNAL_CATCH_METHOD_AS_TEST_CASE( QualifiedMethod, Name, Desc ) \
namespace { Catch : : AutoReg INTERNAL_CATCH_UNIQUE_NAME ( autoRegistrar ) ( & QualifiedMethod , " & " # QualifiedMethod , Catch : : NameAndDesc ( Name , Desc ) , CATCH_INTERNAL_LINEINFO ) ; }
///////////////////////////////////////////////////////////////////////////////
2013-06-07 20:07:50 +02:00
# define INTERNAL_CATCH_TEST_CASE_METHOD( ClassName, TestName, Desc )\
2013-03-16 21:21:51 +01:00
namespace { \
2013-06-07 20:07:50 +02:00
struct INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) : ClassName { \
2013-03-16 21:21:51 +01:00
void test ( ) ; \
} ; \
2013-06-07 20:07:50 +02:00
Catch : : AutoReg INTERNAL_CATCH_UNIQUE_NAME ( autoRegistrar ) ( & INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) : : test , # ClassName , Catch : : NameAndDesc ( TestName , Desc ) , CATCH_INTERNAL_LINEINFO ) ; \
2013-03-16 21:21:51 +01:00
} \
2013-06-07 20:07:50 +02:00
void INTERNAL_CATCH_UNIQUE_NAME ( ____C_A_T_C_H____T_E_S_T____ ) : : test ( )
2011-05-24 09:23:02 +02:00
2013-03-16 21:21:51 +01:00
# endif
2011-05-24 09:23:02 +02:00
// #included from: internal/catch_capture.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_CAPTURE_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_expression_decomposer.hpp
# define TWOBLUECUBES_CATCH_EXPRESSION_DECOMPOSER_HPP_INCLUDED
2012-05-16 16:09:17 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_expression_lhs.hpp
# define TWOBLUECUBES_CATCH_EXPRESSION_LHS_HPP_INCLUDED
2012-05-11 09:17:16 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_expressionresult_builder.h
# define TWOBLUECUBES_CATCH_ASSERTIONRESULT_BUILDER_H_INCLUDED
2012-05-09 20:04:00 +02:00
// #included from: catch_tostring.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_TOSTRING_HPP_INCLUDED
2012-05-09 20:04:00 +02:00
2013-04-20 22:07:32 +02:00
// #included from: catch_sfinae.hpp
# define TWOBLUECUBES_CATCH_SFINAE_HPP_INCLUDED
// Try to detect if the current compiler supports SFINAE
namespace Catch {
struct TrueType {
static const bool value = true ;
typedef void Enable ;
char sizer [ 1 ] ;
} ;
struct FalseType {
static const bool value = false ;
typedef void Disable ;
char sizer [ 2 ] ;
} ;
2013-04-22 19:55:12 +02:00
# ifdef CATCH_CONFIG_SFINAE
2013-04-21 00:20:05 +02:00
2013-04-20 22:07:32 +02:00
template < bool > struct NotABooleanExpression ;
template < bool c > struct If : NotABooleanExpression < c > { } ;
template < > struct If < true > : TrueType { } ;
template < > struct If < false > : FalseType { } ;
template < int size > struct SizedIf ;
template < > struct SizedIf < sizeof ( TrueType ) > : TrueType { } ;
template < > struct SizedIf < sizeof ( FalseType ) > : FalseType { } ;
2013-04-22 19:55:12 +02:00
# endif // CATCH_CONFIG_SFINAE
2013-04-20 22:07:32 +02:00
2013-04-21 00:20:05 +02:00
} // end namespace Catch
2012-05-09 20:04:00 +02:00
# include <sstream>
2013-03-04 12:19:15 +01:00
# include <iomanip>
# include <limits>
2012-05-09 20:04:00 +02:00
2012-08-01 09:17:07 +02:00
# ifdef __OBJC__
// #included from: catch_objc_arc.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_OBJC_ARC_HPP_INCLUDED
2012-08-01 09:17:07 +02:00
# import <Foundation / Foundation.h>
# ifdef __has_feature
# define CATCH_ARC_ENABLED __has_feature(objc_arc)
# else
# define CATCH_ARC_ENABLED 0
# endif
void arcSafeRelease ( NSObject * obj ) ;
id performOptionalSelector ( id obj , SEL sel ) ;
# if !CATCH_ARC_ENABLED
inline void arcSafeRelease ( NSObject * obj ) {
[ obj release ] ;
}
inline id performOptionalSelector ( id obj , SEL sel ) {
if ( [ obj respondsToSelector : sel ] )
return [ obj performSelector : sel ] ;
return nil ;
}
# define CATCH_UNSAFE_UNRETAINED
# define CATCH_ARC_STRONG
# else
inline void arcSafeRelease ( NSObject * ) { }
inline id performOptionalSelector ( id obj , SEL sel ) {
2012-08-16 19:48:50 +02:00
# ifdef __clang__
2012-08-01 09:17:07 +02:00
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Warc-performSelector-leaks"
2012-08-16 19:48:50 +02:00
# endif
2012-08-01 09:17:07 +02:00
if ( [ obj respondsToSelector : sel ] )
return [ obj performSelector : sel ] ;
2012-08-16 19:48:50 +02:00
# ifdef __clang__
2012-08-01 09:17:07 +02:00
# pragma clang diagnostic pop
2012-08-16 19:48:50 +02:00
# endif
2012-08-01 09:17:07 +02:00
return nil ;
}
# define CATCH_UNSAFE_UNRETAINED __unsafe_unretained
# define CATCH_ARC_STRONG __strong
# endif
# endif
2012-05-16 16:09:17 +02:00
namespace Catch {
namespace Detail {
2013-04-21 00:20:05 +02:00
// SFINAE is currently disabled by default for all compilers.
// If the non SFINAE version of IsStreamInsertable is ambiguous for you
2013-04-22 19:55:12 +02:00
// and your compiler supports SFINAE, try #defining CATCH_CONFIG_SFINAE
# ifdef CATCH_CONFIG_SFINAE
2012-05-09 20:04:00 +02:00
2013-04-20 22:07:32 +02:00
template < typename T >
class IsStreamInsertableHelper {
template < int N > struct TrueIfSizeable : TrueType { } ;
template < typename T2 >
static TrueIfSizeable < sizeof ( ( * ( std : : ostream * ) 0 ) < < * ( ( T2 const * ) 0 ) ) > dummy ( T2 * ) ;
static FalseType dummy ( . . . ) ;
public :
typedef SizedIf < sizeof ( dummy ( ( T * ) 0 ) ) > type ;
} ;
2012-05-09 20:04:00 +02:00
template < typename T >
2013-04-20 22:07:32 +02:00
struct IsStreamInsertable : IsStreamInsertableHelper < T > : : type { } ;
2013-04-21 00:20:05 +02:00
# else
2013-04-20 22:07:32 +02:00
2013-04-21 00:20:05 +02:00
struct BorgType {
template < typename T > BorgType ( T const & ) ;
} ;
2013-04-20 22:07:32 +02:00
2013-04-21 00:20:05 +02:00
TrueType & testStreamable ( std : : ostream & ) ;
FalseType testStreamable ( FalseType ) ;
2013-04-20 22:07:32 +02:00
2013-04-21 00:20:05 +02:00
FalseType operator < < ( std : : ostream const & , BorgType const & ) ;
2012-05-09 20:04:00 +02:00
template < typename T >
2013-04-21 00:20:05 +02:00
struct IsStreamInsertable {
static std : : ostream & s ;
2013-04-24 20:10:02 +02:00
static T const & t ;
2013-04-21 00:20:05 +02:00
enum { value = sizeof ( testStreamable ( s < < t ) ) = = sizeof ( TrueType ) } ;
} ;
2013-04-20 22:07:32 +02:00
2013-04-21 00:20:05 +02:00
# endif
2012-05-16 16:09:17 +02:00
2013-04-21 00:20:05 +02:00
template < bool C >
struct StringMakerBase {
template < typename T >
static std : : string convert ( T const & ) { return " {?} " ; }
2012-05-09 20:04:00 +02:00
} ;
2013-04-21 00:20:05 +02:00
template < >
struct StringMakerBase < true > {
template < typename T >
static std : : string convert ( T const & _value ) {
std : : ostringstream oss ;
oss < < _value ;
return oss . str ( ) ;
}
} ;
2012-05-09 20:04:00 +02:00
2013-04-21 00:20:05 +02:00
} // end namespace Detail
2013-03-04 12:19:15 +01:00
template < typename T >
2013-04-21 00:20:05 +02:00
struct StringMaker :
Detail : : StringMakerBase < Detail : : IsStreamInsertable < T > : : value > { } ;
2013-04-20 22:07:32 +02:00
2013-03-04 12:19:15 +01:00
template < typename T >
struct StringMaker < T * > {
2013-07-01 20:02:29 +02:00
template < typename U >
static std : : string convert ( U * p ) {
2012-05-09 20:04:00 +02:00
if ( ! p )
return INTERNAL_CATCH_STRINGIFY ( NULL ) ;
std : : ostringstream oss ;
oss < < p ;
return oss . str ( ) ;
}
2013-03-04 12:19:15 +01:00
} ;
2012-05-09 20:04:00 +02:00
2013-03-04 12:19:15 +01:00
template < typename T >
struct StringMaker < std : : vector < T > > {
static std : : string convert ( std : : vector < T > const & v ) {
2012-05-09 20:04:00 +02:00
std : : ostringstream oss ;
2013-03-04 12:19:15 +01:00
oss < < " { " ;
for ( std : : size_t i = 0 ; i < v . size ( ) ; + + i ) {
2013-04-20 22:07:32 +02:00
oss < < toString ( v [ i ] ) ;
2013-03-04 12:19:15 +01:00
if ( i < v . size ( ) - 1 )
oss < < " , " ;
}
oss < < " } " ;
2012-05-09 20:04:00 +02:00
return oss . str ( ) ;
}
2013-03-04 12:19:15 +01:00
} ;
2012-05-09 20:04:00 +02:00
2013-03-04 12:19:15 +01:00
namespace Detail {
template < typename T >
2013-04-24 20:10:02 +02:00
inline std : : string makeString ( T const & value ) {
2013-03-04 12:19:15 +01:00
return StringMaker < T > : : convert ( value ) ;
}
2012-05-09 20:04:00 +02:00
} // end namespace Detail
/// \brief converts any type to a string
///
/// The default template forwards on to ostringstream - except when an
/// ostringstream overload does not exist - in which case it attempts to detect
/// that and writes {?}.
/// Overload (not specialise) this template for custom typs that you don't want
/// to provide an ostream overload for.
template < typename T >
2013-04-24 20:10:02 +02:00
std : : string toString ( T const & value ) {
2013-03-04 12:19:15 +01:00
return StringMaker < T > : : convert ( value ) ;
2012-05-09 20:04:00 +02:00
}
// Built in overloads
2013-04-24 20:10:02 +02:00
inline std : : string toString ( std : : string const & value ) {
2012-05-09 20:04:00 +02:00
return " \" " + value + " \" " ;
}
2013-04-24 20:10:02 +02:00
inline std : : string toString ( std : : wstring const & value ) {
2012-05-09 20:04:00 +02:00
std : : ostringstream oss ;
oss < < " \" " ;
for ( size_t i = 0 ; i < value . size ( ) ; + + i )
oss < < static_cast < char > ( value [ i ] < = 0xff ? value [ i ] : ' ? ' ) ;
oss < < " \" " ;
return oss . str ( ) ;
}
inline std : : string toString ( const char * const value ) {
return value ? Catch : : toString ( std : : string ( value ) ) : std : : string ( " {null string} " ) ;
}
inline std : : string toString ( char * const value ) {
return Catch : : toString ( static_cast < const char * > ( value ) ) ;
}
inline std : : string toString ( int value ) {
std : : ostringstream oss ;
oss < < value ;
return oss . str ( ) ;
}
inline std : : string toString ( unsigned long value ) {
std : : ostringstream oss ;
if ( value > 8192 )
oss < < " 0x " < < std : : hex < < value ;
else
oss < < value ;
return oss . str ( ) ;
}
inline std : : string toString ( unsigned int value ) {
2012-05-11 20:06:43 +02:00
return toString ( static_cast < unsigned long > ( value ) ) ;
2012-05-09 20:04:00 +02:00
}
inline std : : string toString ( const double value ) {
std : : ostringstream oss ;
2013-03-04 12:19:15 +01:00
oss < < std : : setprecision ( std : : numeric_limits < double > : : digits10 + 1 )
< < value ;
2012-05-09 20:04:00 +02:00
return oss . str ( ) ;
}
inline std : : string toString ( bool value ) {
return value ? " true " : " false " ;
}
2012-06-06 09:06:40 +02:00
inline std : : string toString ( char value ) {
return value < ' '
2012-12-14 08:50:08 +01:00
? toString ( static_cast < unsigned int > ( value ) )
2012-06-06 09:06:40 +02:00
: Detail : : makeString ( value ) ;
}
inline std : : string toString ( signed char value ) {
return toString ( static_cast < char > ( value ) ) ;
}
2012-12-14 08:50:08 +01:00
inline std : : string toString ( unsigned char value ) {
return toString ( static_cast < char > ( value ) ) ;
}
2012-05-11 09:17:16 +02:00
# ifdef CATCH_CONFIG_CPP11_NULLPTR
2012-05-25 09:52:05 +02:00
inline std : : string toString ( std : : nullptr_t ) {
2012-05-11 09:17:16 +02:00
return " nullptr " ;
}
# endif
2012-08-01 09:17:07 +02:00
# ifdef __OBJC__
inline std : : string toString ( NSString const * const & nsstring ) {
2013-03-12 19:49:37 +01:00
if ( ! nsstring )
return " nil " ;
2012-08-01 09:17:07 +02:00
return std : : string ( " @ \" " ) + [ nsstring UTF8String ] + " \" " ;
}
inline std : : string toString ( NSString * CATCH_ARC_STRONG const & nsstring ) {
2013-03-12 19:49:37 +01:00
if ( ! nsstring )
return " nil " ;
2012-08-01 09:17:07 +02:00
return std : : string ( " @ \" " ) + [ nsstring UTF8String ] + " \" " ;
}
inline std : : string toString ( NSObject * const & nsObject ) {
return toString ( [ nsObject description ] ) ;
}
# endif
2012-05-09 20:04:00 +02:00
} // end namespace Catch
2012-11-01 09:27:09 +01:00
// #included from: catch_assertionresult.h
# define TWOBLUECUBES_CATCH_ASSERTIONRESULT_H_INCLUDED
2011-05-24 09:23:02 +02:00
# include <string>
// #included from: catch_result_type.h
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_RESULT_TYPE_H_INCLUDED
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
// ResultWas::OfType enum
struct ResultWas { enum OfType {
Unknown = - 1 ,
Ok = 0 ,
Info = 1 ,
Warning = 2 ,
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
FailureBit = 0x10 ,
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
ExpressionFailed = FailureBit | 1 ,
ExplicitFailure = FailureBit | 2 ,
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
Exception = 0x100 | FailureBit ,
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
ThrewException = Exception | 1 ,
DidntThrowException = Exception | 2
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
} ; } ;
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
inline bool isOk ( ResultWas : : OfType resultType ) {
return ( resultType & ResultWas : : FailureBit ) = = 0 ;
}
2013-02-02 21:37:58 +01:00
inline bool isJustInfo ( int flags ) {
return flags = = ResultWas : : Info ;
}
2012-11-13 23:04:29 +01:00
// ResultAction::Value enum
struct ResultAction { enum Value {
None ,
Failed = 1 , // Failure - but no debug break if Debug bit not set
Debug = 2 , // If this bit is set, invoke the debugger
Abort = 4 // Test run should abort
} ; } ;
2012-10-09 12:48:55 +02:00
2012-11-13 23:04:29 +01:00
// ResultDisposition::Flags enum
struct ResultDisposition { enum Flags {
Normal = 0x00 ,
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
ContinueOnFailure = 0x01 , // Failures fail test, but execution continues
NegateResult = 0x02 , // Prefix expressiom with !
SuppressFail = 0x04 // Failures are reported but do not fail the test
} ; } ;
inline ResultDisposition : : Flags operator | ( ResultDisposition : : Flags lhs , ResultDisposition : : Flags rhs ) {
return static_cast < ResultDisposition : : Flags > ( static_cast < int > ( lhs ) | static_cast < int > ( rhs ) ) ;
}
2012-11-17 11:49:24 +01:00
inline bool shouldContinueOnFailure ( int flags ) { return ( flags & ResultDisposition : : ContinueOnFailure ) ! = 0 ; }
inline bool shouldNegate ( int flags ) { return ( flags & ResultDisposition : : NegateResult ) ! = 0 ; }
inline bool shouldSuppressFailure ( int flags ) { return ( flags & ResultDisposition : : SuppressFail ) ! = 0 ; }
2012-11-13 23:04:29 +01:00
} // end namespace Catch
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
struct AssertionInfo
2012-10-09 12:48:55 +02:00
{
2012-11-01 09:27:09 +01:00
AssertionInfo ( ) { }
2013-04-24 20:10:02 +02:00
AssertionInfo ( std : : string const & _macroName ,
SourceLineInfo const & _lineInfo ,
std : : string const & _capturedExpression ,
2012-11-13 23:04:29 +01:00
ResultDisposition : : Flags _resultDisposition ) ;
2012-10-09 12:48:55 +02:00
std : : string macroName ;
SourceLineInfo lineInfo ;
std : : string capturedExpression ;
2012-11-13 23:04:29 +01:00
ResultDisposition : : Flags resultDisposition ;
2012-11-01 09:27:09 +01:00
} ;
struct AssertionResultData
{
AssertionResultData ( ) : resultType ( ResultWas : : Unknown ) { }
2012-10-09 12:48:55 +02:00
std : : string reconstructedExpression ;
std : : string message ;
ResultWas : : OfType resultType ;
} ;
2012-11-01 09:27:09 +01:00
class AssertionResult {
2012-05-16 16:09:17 +02:00
public :
2012-11-01 09:27:09 +01:00
AssertionResult ( ) ;
2013-04-24 20:10:02 +02:00
AssertionResult ( AssertionInfo const & info , AssertionResultData const & data ) ;
2012-11-01 09:27:09 +01:00
~ AssertionResult ( ) ;
2012-08-09 08:47:30 +02:00
2012-11-13 23:04:29 +01:00
bool isOk ( ) const ;
bool succeeded ( ) const ;
2012-08-09 08:47:30 +02:00
ResultWas : : OfType getResultType ( ) const ;
bool hasExpression ( ) const ;
bool hasMessage ( ) const ;
std : : string getExpression ( ) const ;
2013-06-07 20:07:50 +02:00
std : : string getExpressionInMacro ( ) const ;
2012-08-09 08:47:30 +02:00
bool hasExpandedExpression ( ) const ;
std : : string getExpandedExpression ( ) const ;
std : : string getMessage ( ) const ;
2012-11-01 09:27:09 +01:00
SourceLineInfo getSourceInfo ( ) const ;
2012-08-09 08:47:30 +02:00
std : : string getTestMacroName ( ) const ;
2011-05-24 09:23:02 +02:00
protected :
2012-11-01 09:27:09 +01:00
AssertionInfo m_info ;
AssertionResultData m_resultData ;
2011-05-24 09:23:02 +02:00
} ;
} // end namespace Catch
// #included from: catch_evaluate.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_EVALUATE_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2013-03-25 10:25:31 +01:00
# ifdef _MSC_VER
# pragma warning(push)
# pragma warning(disable:4389) // '==' : signed/unsigned mismatch
# endif
2012-05-16 16:09:17 +02:00
namespace Catch {
namespace Internal {
enum Operator {
2011-05-24 09:23:02 +02:00
IsEqualTo ,
IsNotEqualTo ,
IsLessThan ,
IsGreaterThan ,
IsLessThanOrEqualTo ,
IsGreaterThanOrEqualTo
} ;
2012-05-21 19:52:09 +02:00
template < Operator Op > struct OperatorTraits { static const char * getName ( ) { return " *error* " ; } } ;
template < > struct OperatorTraits < IsEqualTo > { static const char * getName ( ) { return " == " ; } } ;
template < > struct OperatorTraits < IsNotEqualTo > { static const char * getName ( ) { return " != " ; } } ;
template < > struct OperatorTraits < IsLessThan > { static const char * getName ( ) { return " < " ; } } ;
template < > struct OperatorTraits < IsGreaterThan > { static const char * getName ( ) { return " > " ; } } ;
template < > struct OperatorTraits < IsLessThanOrEqualTo > { static const char * getName ( ) { return " <= " ; } } ;
template < > struct OperatorTraits < IsGreaterThanOrEqualTo > { static const char * getName ( ) { return " >= " ; } } ;
2011-05-24 09:23:02 +02:00
2013-04-22 19:55:12 +02:00
template < typename T >
2013-04-24 20:10:02 +02:00
inline T & opCast ( T const & t ) { return const_cast < T & > ( t ) ; }
2013-04-22 19:55:12 +02:00
// nullptr_t support based on pull request #154 from Konstantin Baumann
# ifdef CATCH_CONFIG_CPP11_NULLPTR
inline std : : nullptr_t opCast ( std : : nullptr_t ) { return nullptr ; }
# endif // CATCH_CONFIG_CPP11_NULLPTR
2011-05-24 09:23:02 +02:00
// So the compare overloads can be operator agnostic we convey the operator as a template
// enum, which is used to specialise an Evaluator for doing the comparison.
template < typename T1 , typename T2 , Operator Op >
class Evaluator { } ;
template < typename T1 , typename T2 >
2012-05-16 16:09:17 +02:00
struct Evaluator < T1 , T2 , IsEqualTo > {
2013-04-24 20:10:02 +02:00
static bool evaluate ( T1 const & lhs , T2 const & rhs ) {
2013-04-22 19:55:12 +02:00
return opCast ( lhs ) = = opCast ( rhs ) ;
2011-05-24 09:23:02 +02:00
}
} ;
template < typename T1 , typename T2 >
2012-05-16 16:09:17 +02:00
struct Evaluator < T1 , T2 , IsNotEqualTo > {
2013-04-24 20:10:02 +02:00
static bool evaluate ( T1 const & lhs , T2 const & rhs ) {
2013-04-22 19:55:12 +02:00
return opCast ( lhs ) ! = opCast ( rhs ) ;
2011-05-24 09:23:02 +02:00
}
} ;
template < typename T1 , typename T2 >
2012-05-16 16:09:17 +02:00
struct Evaluator < T1 , T2 , IsLessThan > {
2013-04-24 20:10:02 +02:00
static bool evaluate ( T1 const & lhs , T2 const & rhs ) {
2013-04-22 19:55:12 +02:00
return opCast ( lhs ) < opCast ( rhs ) ;
2011-05-24 09:23:02 +02:00
}
} ;
template < typename T1 , typename T2 >
2012-05-16 16:09:17 +02:00
struct Evaluator < T1 , T2 , IsGreaterThan > {
2013-04-24 20:10:02 +02:00
static bool evaluate ( T1 const & lhs , T2 const & rhs ) {
2013-04-22 19:55:12 +02:00
return opCast ( lhs ) > opCast ( rhs ) ;
2011-05-24 09:23:02 +02:00
}
} ;
template < typename T1 , typename T2 >
2012-05-16 16:09:17 +02:00
struct Evaluator < T1 , T2 , IsGreaterThanOrEqualTo > {
2013-04-24 20:10:02 +02:00
static bool evaluate ( T1 const & lhs , T2 const & rhs ) {
2013-04-22 19:55:12 +02:00
return opCast ( lhs ) > = opCast ( rhs ) ;
2011-05-24 09:23:02 +02:00
}
} ;
template < typename T1 , typename T2 >
2012-05-16 16:09:17 +02:00
struct Evaluator < T1 , T2 , IsLessThanOrEqualTo > {
2013-04-24 20:10:02 +02:00
static bool evaluate ( T1 const & lhs , T2 const & rhs ) {
2013-04-22 19:55:12 +02:00
return opCast ( lhs ) < = opCast ( rhs ) ;
2011-05-24 09:23:02 +02:00
}
} ;
template < Operator Op , typename T1 , typename T2 >
2013-04-24 20:10:02 +02:00
bool applyEvaluator ( T1 const & lhs , T2 const & rhs ) {
2011-05-24 09:23:02 +02:00
return Evaluator < T1 , T2 , Op > : : evaluate ( lhs , rhs ) ;
}
2012-11-13 23:04:29 +01:00
// This level of indirection allows us to specialise for integer types
// to avoid signed/ unsigned warnings
2011-05-24 09:23:02 +02:00
// "base" overload
template < Operator Op , typename T1 , typename T2 >
2013-04-24 20:10:02 +02:00
bool compare ( T1 const & lhs , T2 const & rhs ) {
2011-05-24 09:23:02 +02:00
return Evaluator < T1 , T2 , Op > : : evaluate ( lhs , rhs ) ;
}
// unsigned X to int
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( unsigned int lhs , int rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( lhs , static_cast < unsigned int > ( rhs ) ) ;
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( unsigned long lhs , int rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( lhs , static_cast < unsigned int > ( rhs ) ) ;
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( unsigned char lhs , int rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( lhs , static_cast < unsigned int > ( rhs ) ) ;
}
// unsigned X to long
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( unsigned int lhs , long rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( lhs , static_cast < unsigned long > ( rhs ) ) ;
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( unsigned long lhs , long rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( lhs , static_cast < unsigned long > ( rhs ) ) ;
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( unsigned char lhs , long rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( lhs , static_cast < unsigned long > ( rhs ) ) ;
}
// int to unsigned X
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( int lhs , unsigned int rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( static_cast < unsigned int > ( lhs ) , rhs ) ;
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( int lhs , unsigned long rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( static_cast < unsigned int > ( lhs ) , rhs ) ;
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( int lhs , unsigned char rhs ) {
2011-05-24 09:23:02 +02:00
return applyEvaluator < Op > ( static_cast < unsigned int > ( lhs ) , rhs ) ;
}
// long to unsigned X
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( long lhs , unsigned int rhs ) {
2012-04-27 19:42:40 +02:00
return applyEvaluator < Op > ( static_cast < unsigned long > ( lhs ) , rhs ) ;
2011-05-24 09:23:02 +02:00
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( long lhs , unsigned long rhs ) {
2012-04-27 19:42:40 +02:00
return applyEvaluator < Op > ( static_cast < unsigned long > ( lhs ) , rhs ) ;
2011-05-24 09:23:02 +02:00
}
2012-05-16 16:09:17 +02:00
template < Operator Op > bool compare ( long lhs , unsigned char rhs ) {
2012-04-27 19:42:40 +02:00
return applyEvaluator < Op > ( static_cast < unsigned long > ( lhs ) , rhs ) ;
2011-05-24 09:23:02 +02:00
}
2012-05-25 09:52:05 +02:00
// pointer to long (when comparing against NULL)
2012-11-13 23:04:29 +01:00
template < Operator Op , typename T > bool compare ( long lhs , T * rhs ) {
2011-05-24 09:23:02 +02:00
return Evaluator < T * , T * , Op > : : evaluate ( reinterpret_cast < T * > ( lhs ) , rhs ) ;
2012-05-07 20:46:19 +02:00
}
2012-11-13 23:04:29 +01:00
template < Operator Op , typename T > bool compare ( T * lhs , long rhs ) {
2012-05-07 20:46:19 +02:00
return Evaluator < T * , T * , Op > : : evaluate ( lhs , reinterpret_cast < T * > ( rhs ) ) ;
2011-05-24 09:23:02 +02:00
}
2012-05-25 09:52:05 +02:00
// pointer to int (when comparing against NULL)
2012-11-13 23:04:29 +01:00
template < Operator Op , typename T > bool compare ( int lhs , T * rhs ) {
2012-05-25 09:52:05 +02:00
return Evaluator < T * , T * , Op > : : evaluate ( reinterpret_cast < T * > ( lhs ) , rhs ) ;
}
2012-11-13 23:04:29 +01:00
template < Operator Op , typename T > bool compare ( T * lhs , int rhs ) {
2012-05-25 09:52:05 +02:00
return Evaluator < T * , T * , Op > : : evaluate ( lhs , reinterpret_cast < T * > ( rhs ) ) ;
}
2013-04-22 19:55:12 +02:00
# ifdef CATCH_CONFIG_CPP11_NULLPTR
// pointer to nullptr_t (when comparing against nullptr)
template < Operator Op , typename T > bool compare ( std : : nullptr_t , T * rhs ) {
return Evaluator < T * , T * , Op > : : evaluate ( NULL , rhs ) ;
}
template < Operator Op , typename T > bool compare ( T * lhs , std : : nullptr_t ) {
return Evaluator < T * , T * , Op > : : evaluate ( lhs , NULL ) ;
}
# endif // CATCH_CONFIG_CPP11_NULLPTR
2011-05-24 09:23:02 +02:00
} // end of namespace Internal
} // end of namespace Catch
2013-03-25 10:25:31 +01:00
# ifdef _MSC_VER
# pragma warning(pop)
# endif
2012-05-16 16:09:17 +02:00
namespace Catch {
2011-12-28 11:37:31 +01:00
2013-06-29 13:11:32 +02:00
struct STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison ;
2012-11-01 09:27:09 +01:00
// Wraps the (stringised versions of) the lhs, operator and rhs of an expression - as well as
// the result of evaluating it. This is used to build an AssertionResult object
class ExpressionResultBuilder {
2011-05-24 09:23:02 +02:00
public :
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder ( ResultWas : : OfType resultType = ResultWas : : Unknown ) ;
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder ( ExpressionResultBuilder const & other ) ;
ExpressionResultBuilder & operator = ( ExpressionResultBuilder const & other ) ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder & setResultType ( ResultWas : : OfType result ) ;
ExpressionResultBuilder & setResultType ( bool result ) ;
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & setLhs ( std : : string const & lhs ) ;
ExpressionResultBuilder & setRhs ( std : : string const & rhs ) ;
ExpressionResultBuilder & setOp ( std : : string const & op ) ;
2012-03-04 22:22:36 +01:00
2012-11-13 23:04:29 +01:00
ExpressionResultBuilder & endExpression ( ResultDisposition : : Flags resultDisposition ) ;
2012-10-09 12:48:55 +02:00
2012-11-01 09:27:09 +01:00
template < typename T >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & operator < < ( T const & value ) {
2012-11-01 09:27:09 +01:00
m_stream < < value ;
return * this ;
}
2012-10-09 12:48:55 +02:00
2013-04-24 20:10:02 +02:00
std : : string reconstructExpression ( AssertionInfo const & info ) const ;
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
AssertionResult buildResult ( AssertionInfo const & info ) const ;
2011-05-24 09:23:02 +02:00
2013-06-29 13:11:32 +02:00
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator & & ( RhsT const & ) ;
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator | | ( RhsT const & ) ;
2012-10-09 12:48:55 +02:00
private :
2012-11-01 09:27:09 +01:00
AssertionResultData m_data ;
struct ExprComponents {
ExprComponents ( ) : shouldNegate ( false ) { }
bool shouldNegate ;
std : : string lhs , rhs , op ;
} m_exprComponents ;
std : : ostringstream m_stream ;
2011-05-24 09:23:02 +02:00
} ;
2012-05-09 20:04:00 +02:00
} // end namespace Catch
2012-05-16 16:09:17 +02:00
namespace Catch {
2012-05-11 09:17:16 +02:00
2012-11-01 09:27:09 +01:00
// Wraps the LHS of an expression and captures the operator and RHS (if any) - wrapping them all
// in an ExpressionResultBuilder object
2012-05-11 09:17:16 +02:00
template < typename T >
2012-11-01 09:27:09 +01:00
class ExpressionLhs {
2013-04-24 20:10:02 +02:00
void operator = ( ExpressionLhs const & ) ;
2012-05-11 09:17:16 +02:00
public :
2012-11-16 10:47:33 +01:00
ExpressionLhs ( T lhs ) : m_lhs ( lhs ) { }
2012-05-11 09:17:16 +02:00
template < typename RhsT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & operator = = ( RhsT const & rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsEqualTo > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
template < typename RhsT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & operator ! = ( RhsT const & rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsNotEqualTo > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
template < typename RhsT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & operator < ( RhsT const & rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsLessThan > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
template < typename RhsT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & operator > ( RhsT const & rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsGreaterThan > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
template < typename RhsT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & operator < = ( RhsT const & rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsLessThanOrEqualTo > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
template < typename RhsT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & operator > = ( RhsT const & rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsGreaterThanOrEqualTo > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder & operator = = ( bool rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsEqualTo > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder & operator ! = ( bool rhs ) {
2012-10-09 21:58:33 +02:00
return captureExpression < Internal : : IsNotEqualTo > ( rhs ) ;
2012-05-11 09:17:16 +02:00
}
2012-11-13 23:04:29 +01:00
ExpressionResultBuilder & endExpression ( ResultDisposition : : Flags resultDisposition ) {
2012-11-01 09:27:09 +01:00
bool value = m_lhs ? true : false ;
return m_result
. setLhs ( Catch : : toString ( value ) )
. setResultType ( value )
2012-11-13 23:04:29 +01:00
. endExpression ( resultDisposition ) ;
2012-05-11 09:17:16 +02:00
}
2012-11-01 09:27:09 +01:00
// Only simple binary expressions are allowed on the LHS.
// If more complex compositions are required then place the sub expression in parentheses
2013-04-24 20:10:02 +02:00
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator + ( RhsT const & ) ;
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator - ( RhsT const & ) ;
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator / ( RhsT const & ) ;
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator * ( RhsT const & ) ;
2013-06-29 13:11:32 +02:00
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator & & ( RhsT const & ) ;
template < typename RhsT > STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison & operator | | ( RhsT const & ) ;
2012-05-11 09:17:16 +02:00
2012-10-09 21:58:33 +02:00
private :
template < Internal : : Operator Op , typename RhsT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & captureExpression ( RhsT const & rhs ) {
2012-10-09 21:58:33 +02:00
return m_result
2012-11-01 09:27:09 +01:00
. setResultType ( Internal : : compare < Op > ( m_lhs , rhs ) )
. setLhs ( Catch : : toString ( m_lhs ) )
2012-10-09 21:58:33 +02:00
. setRhs ( Catch : : toString ( rhs ) )
. setOp ( Internal : : OperatorTraits < Op > : : getName ( ) ) ;
}
2012-05-11 09:17:16 +02:00
private :
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder m_result ;
2012-05-11 09:17:16 +02:00
T m_lhs ;
} ;
} // end namespace Catch
2012-05-16 16:09:17 +02:00
namespace Catch {
2012-11-01 09:27:09 +01:00
// Captures the LHS of the expression and wraps it in an Expression Lhs object
class ExpressionDecomposer {
2012-05-16 16:09:17 +02:00
public :
template < typename T >
2013-04-24 20:10:02 +02:00
ExpressionLhs < T const & > operator - > * ( T const & operand ) {
return ExpressionLhs < T const & > ( operand ) ;
2012-05-16 16:09:17 +02:00
}
2012-11-01 09:27:09 +01:00
ExpressionLhs < bool > operator - > * ( bool value ) {
return ExpressionLhs < bool > ( value ) ;
2012-05-16 16:09:17 +02:00
}
} ;
} // end namespace Catch
2013-02-02 21:37:58 +01:00
// #included from: catch_message.h
# define TWOBLUECUBES_CATCH_MESSAGE_H_INCLUDED
2012-05-09 20:04:00 +02:00
# include <string>
2012-09-29 21:54:03 +02:00
2012-05-16 16:09:17 +02:00
namespace Catch {
2013-02-02 21:37:58 +01:00
struct MessageInfo {
MessageInfo ( std : : string const & _macroName ,
SourceLineInfo const & _lineInfo ,
ResultWas : : OfType _type ) ;
2012-08-31 09:10:36 +02:00
2013-02-02 21:37:58 +01:00
std : : string macroName ;
SourceLineInfo lineInfo ;
ResultWas : : OfType type ;
std : : string message ;
unsigned int sequence ;
bool operator = = ( MessageInfo const & other ) const {
return sequence = = other . sequence ;
}
bool operator < ( MessageInfo const & other ) const {
return sequence < other . sequence ;
}
private :
static unsigned int globalCount ;
} ;
2013-06-28 18:09:57 +02:00
struct MessageBuilder {
MessageBuilder ( std : : string const & macroName ,
SourceLineInfo const & lineInfo ,
ResultWas : : OfType type )
: m_info ( macroName , lineInfo , type )
{ }
2013-02-02 21:37:58 +01:00
template < typename T >
2013-06-28 18:09:57 +02:00
MessageBuilder & operator < < ( T const & value ) {
m_stream < < value ;
2013-02-02 21:37:58 +01:00
return * this ;
}
2013-06-28 18:09:57 +02:00
MessageInfo m_info ;
std : : ostringstream m_stream ;
2013-02-02 21:37:58 +01:00
} ;
2013-06-28 18:09:57 +02:00
class ScopedMessage {
2013-02-02 21:37:58 +01:00
public :
2013-06-28 18:09:57 +02:00
ScopedMessage ( MessageBuilder const & builder ) ;
~ ScopedMessage ( ) ;
MessageInfo m_info ;
2013-02-02 21:37:58 +01:00
} ;
} // end namespace Catch
2012-05-09 20:04:00 +02:00
// #included from: catch_interfaces_capture.h
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_INTERFACES_CAPTURE_H_INCLUDED
2012-05-09 20:04:00 +02:00
# include <string>
2012-08-31 09:10:36 +02:00
// #included from: catch_totals.hpp
# define TWOBLUECUBES_CATCH_TOTALS_HPP_INCLUDED
2012-05-09 20:04:00 +02:00
2012-09-29 21:54:03 +02:00
# include <cstddef>
2012-05-16 16:09:17 +02:00
namespace Catch {
2012-08-31 09:10:36 +02:00
struct Counts {
2013-02-02 21:37:58 +01:00
Counts ( ) : passed ( 0 ) , failed ( 0 ) { }
2012-08-31 09:10:36 +02:00
2013-04-24 20:10:02 +02:00
Counts operator - ( Counts const & other ) const {
2012-08-31 09:10:36 +02:00
Counts diff ;
diff . passed = passed - other . passed ;
diff . failed = failed - other . failed ;
return diff ;
}
2013-04-24 20:10:02 +02:00
Counts & operator + = ( Counts const & other ) {
2012-08-31 09:10:36 +02:00
passed + = other . passed ;
failed + = other . failed ;
return * this ;
}
std : : size_t total ( ) const {
return passed + failed ;
}
std : : size_t passed ;
std : : size_t failed ;
} ;
struct Totals {
2013-04-24 20:10:02 +02:00
Totals operator - ( Totals const & other ) const {
2012-08-31 09:10:36 +02:00
Totals diff ;
diff . assertions = assertions - other . assertions ;
diff . testCases = testCases - other . testCases ;
return diff ;
}
2013-04-24 20:10:02 +02:00
Totals delta ( Totals const & prevTotals ) const {
2012-08-31 09:10:36 +02:00
Totals diff = * this - prevTotals ;
if ( diff . assertions . failed > 0 )
+ + diff . testCases . failed ;
else
+ + diff . testCases . passed ;
return diff ;
}
2013-04-24 20:10:02 +02:00
Totals & operator + = ( Totals const & other ) {
2012-08-31 09:10:36 +02:00
assertions + = other . assertions ;
testCases + = other . testCases ;
return * this ;
}
Counts assertions ;
Counts testCases ;
} ;
}
namespace Catch {
2012-12-02 00:57:18 +01:00
class TestCase ;
2012-11-01 09:27:09 +01:00
class ExpressionResultBuilder ;
class AssertionResult ;
struct AssertionInfo ;
2012-12-02 00:57:18 +01:00
struct SectionInfo ;
2013-06-28 18:09:57 +02:00
struct MessageInfo ;
2013-02-02 21:37:58 +01:00
class ScopedMessageBuilder ;
2012-08-31 09:10:36 +02:00
struct IResultCapture {
2012-05-09 20:04:00 +02:00
2012-08-13 08:46:10 +02:00
virtual ~ IResultCapture ( ) ;
2012-05-09 20:04:00 +02:00
2013-02-02 21:37:58 +01:00
virtual void assertionEnded ( AssertionResult const & result ) = 0 ;
2012-12-02 00:57:18 +01:00
virtual bool sectionStarted ( SectionInfo const & sectionInfo ,
2012-05-16 16:09:17 +02:00
Counts & assertions ) = 0 ;
2012-12-02 00:57:18 +01:00
virtual void sectionEnded ( SectionInfo const & name , Counts const & assertions ) = 0 ;
2013-06-28 18:09:57 +02:00
virtual void pushScopedMessage ( MessageInfo const & message ) = 0 ;
virtual void popScopedMessage ( MessageInfo const & message ) = 0 ;
2013-02-02 21:37:58 +01:00
2012-05-16 16:09:17 +02:00
virtual bool shouldDebugBreak ( ) const = 0 ;
2012-12-02 00:57:18 +01:00
virtual ResultAction : : Value acceptExpression ( ExpressionResultBuilder const & assertionResult , AssertionInfo const & assertionInfo ) = 0 ;
2012-05-16 16:09:17 +02:00
virtual std : : string getCurrentTestName ( ) const = 0 ;
2012-11-01 09:27:09 +01:00
virtual const AssertionResult * getLastResult ( ) const = 0 ;
2012-05-09 20:04:00 +02:00
} ;
}
// #included from: catch_debugger.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_DEBUGGER_HPP_INCLUDED
2012-05-09 20:04:00 +02:00
# include <iostream>
# if defined(__MAC_OS_X_VERSION_MIN_REQUIRED)
# define CATCH_PLATFORM_MAC
# elif defined(__IPHONE_OS_VERSION_MIN_REQUIRED)
# define CATCH_PLATFORM_IPHONE
# elif defined(__WIN32__) || defined(_WIN32) || defined(_MSC_VER)
# define CATCH_PLATFORM_WINDOWS
# endif
# ifdef CATCH_PLATFORM_MAC
# include <assert.h>
# include <stdbool.h>
# include <sys/types.h>
# include <unistd.h>
# include <sys/sysctl.h>
2012-05-16 16:09:17 +02:00
namespace Catch {
2012-05-09 20:04:00 +02:00
// The following function is taken directly from the following technical note:
// http://developer.apple.com/library/mac/#qa/qa2004/qa1361.html
// Returns true if the current process is being debugged (either
// running under the debugger or has a debugger attached post facto).
2012-05-16 16:09:17 +02:00
inline bool isDebuggerActive ( ) {
2012-05-09 20:04:00 +02:00
int junk ;
int mib [ 4 ] ;
struct kinfo_proc info ;
size_t size ;
// Initialize the flags so that, if sysctl fails for some bizarre
// reason, we get a predictable result.
info . kp_proc . p_flag = 0 ;
// Initialize mib, which tells sysctl the info we want, in this case
// we're looking for information about a specific process ID.
mib [ 0 ] = CTL_KERN ;
mib [ 1 ] = KERN_PROC ;
mib [ 2 ] = KERN_PROC_PID ;
mib [ 3 ] = getpid ( ) ;
// Call sysctl.
size = sizeof ( info ) ;
junk = sysctl ( mib , sizeof ( mib ) / sizeof ( * mib ) , & info , & size , NULL , 0 ) ;
assert ( junk = = 0 ) ;
// We're being debugged if the P_TRACED flag is set.
return ( ( info . kp_proc . p_flag & P_TRACED ) ! = 0 ) ;
}
}
// The following code snippet taken from:
// http://cocoawithlove.com/2008/03/break-into-debugger.html
# ifdef DEBUG
# if defined(__ppc64__) || defined(__ppc__)
# define BreakIntoDebugger() \
2012-05-16 16:09:17 +02:00
if ( Catch : : isDebuggerActive ( ) ) { \
2012-05-09 20:04:00 +02:00
__asm__ ( " li r0, 20 \n sc \n nop \n li r0, 37 \n li r4, 2 \n sc \n nop \n " \
: : : " memory " , " r0 " , " r3 " , " r4 " ) ; \
}
# else
# define BreakIntoDebugger() if( Catch::isDebuggerActive() ) {__asm__("int $3\n" : : );}
# endif
# else
inline void BreakIntoDebugger ( ) { }
# endif
# elif defined(_MSC_VER)
extern " C " __declspec ( dllimport ) int __stdcall IsDebuggerPresent ( ) ;
# define BreakIntoDebugger() if (IsDebuggerPresent() ) { __debugbreak(); }
2012-05-16 16:09:17 +02:00
inline bool isDebuggerActive ( ) {
2012-05-09 20:04:00 +02:00
return IsDebuggerPresent ( ) ! = 0 ;
}
2012-05-11 09:17:16 +02:00
# elif defined(__MINGW32__)
extern " C " __declspec ( dllimport ) int __stdcall IsDebuggerPresent ( ) ;
extern " C " __declspec ( dllimport ) void __stdcall DebugBreak ( ) ;
# define BreakIntoDebugger() if (IsDebuggerPresent() ) { DebugBreak(); }
2012-05-16 16:09:17 +02:00
inline bool isDebuggerActive ( ) {
2012-05-11 09:17:16 +02:00
return IsDebuggerPresent ( ) ! = 0 ;
}
2012-05-09 20:04:00 +02:00
# else
2013-03-08 10:30:25 +01:00
inline void BreakIntoDebugger ( ) { }
inline bool isDebuggerActive ( ) { return false ; }
2012-05-09 20:04:00 +02:00
# endif
# ifdef CATCH_PLATFORM_WINDOWS
extern " C " __declspec ( dllimport ) void __stdcall OutputDebugStringA ( const char * ) ;
2013-04-24 20:10:02 +02:00
inline void writeToDebugConsole ( std : : string const & text ) {
2012-05-09 20:04:00 +02:00
: : OutputDebugStringA ( text . c_str ( ) ) ;
}
# else
2013-04-24 20:10:02 +02:00
inline void writeToDebugConsole ( std : : string const & text ) {
2012-05-09 20:04:00 +02:00
// !TBD: Need a version for Mac/ XCode and other IDEs
std : : cout < < text ;
}
# endif // CATCH_PLATFORM_WINDOWS
2012-11-01 09:27:09 +01:00
// #included from: catch_interfaces_registry_hub.h
# define TWOBLUECUBES_CATCH_INTERFACES_REGISTRY_HUB_H_INCLUDED
2012-05-09 20:04:00 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_interfaces_reporter.h
# define TWOBLUECUBES_CATCH_INTERFACES_REPORTER_H_INCLUDED
2012-05-09 20:04:00 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_config.hpp
# define TWOBLUECUBES_CATCH_CONFIG_HPP_INCLUDED
2012-05-09 20:04:00 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_test_spec.h
# define TWOBLUECUBES_CATCH_TEST_SPEC_H_INCLUDED
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_test_case_info.h
# define TWOBLUECUBES_CATCH_TEST_CASE_INFO_H_INCLUDED
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
# include <string>
# include <set>
2011-06-29 20:25:14 +02:00
2013-03-13 13:19:30 +01:00
# ifdef __clang__
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Wpadded"
# endif
2012-11-01 09:27:09 +01:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
struct ITestCase ;
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
struct TestCaseInfo {
2013-04-24 20:10:02 +02:00
TestCaseInfo ( std : : string const & _name ,
std : : string const & _className ,
std : : string const & _description ,
std : : set < std : : string > const & _tags ,
2012-12-02 00:57:18 +01:00
bool _isHidden ,
2013-04-24 20:10:02 +02:00
SourceLineInfo const & _lineInfo ) ;
2012-12-02 00:57:18 +01:00
2013-04-24 20:10:02 +02:00
TestCaseInfo ( TestCaseInfo const & other ) ;
2012-12-02 00:57:18 +01:00
std : : string name ;
std : : string className ;
std : : string description ;
std : : set < std : : string > tags ;
2013-03-29 14:44:33 +01:00
std : : string tagsAsString ;
2012-12-02 00:57:18 +01:00
SourceLineInfo lineInfo ;
2012-12-06 09:44:51 +01:00
bool isHidden ;
2012-12-02 00:57:18 +01:00
} ;
class TestCase : protected TestCaseInfo {
2012-11-01 09:27:09 +01:00
public :
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
TestCase ( ITestCase * testCase , TestCaseInfo const & info ) ;
TestCase ( TestCase const & other ) ;
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
TestCase withName ( std : : string const & _newName ) const ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
void invoke ( ) const ;
2012-11-13 23:04:29 +01:00
2013-04-24 20:10:02 +02:00
TestCaseInfo const & getTestCaseInfo ( ) const ;
2012-12-02 00:57:18 +01:00
2012-11-01 09:27:09 +01:00
bool isHidden ( ) const ;
2013-04-24 20:10:02 +02:00
bool hasTag ( std : : string const & tag ) const ;
bool matchesTags ( std : : string const & tagPattern ) const ;
std : : set < std : : string > const & getTags ( ) const ;
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
void swap ( TestCase & other ) ;
2013-04-24 20:10:02 +02:00
bool operator = = ( TestCase const & other ) const ;
bool operator < ( TestCase const & other ) const ;
TestCase & operator = ( TestCase const & other ) ;
2012-02-10 09:30:13 +01:00
2012-11-01 09:27:09 +01:00
private :
2012-12-02 00:57:18 +01:00
Ptr < ITestCase > test ;
2012-11-01 09:27:09 +01:00
} ;
2012-12-02 00:57:18 +01:00
TestCase makeTestCase ( ITestCase * testCase ,
2013-04-24 20:10:02 +02:00
std : : string const & className ,
std : : string const & name ,
std : : string const & description ,
SourceLineInfo const & lineInfo ) ;
2012-11-01 09:27:09 +01:00
}
2012-02-10 09:30:13 +01:00
2013-03-13 13:19:30 +01:00
# ifdef __clang__
# pragma clang diagnostic pop
# endif
2012-11-01 09:27:09 +01:00
// #included from: catch_tags.hpp
# define TWOBLUECUBES_CATCH_TAGS_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
# include <string>
# include <set>
# include <map>
# include <vector>
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
# ifdef __clang__
# pragma clang diagnostic ignored "-Wpadded"
# endif
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
namespace Catch {
class TagParser {
public :
virtual ~ TagParser ( ) ;
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
void parse ( std : : string const & str ) {
2012-11-01 09:27:09 +01:00
std : : size_t pos = 0 ;
while ( pos < str . size ( ) ) {
char c = str [ pos ] ;
if ( c = = ' [ ' ) {
std : : size_t end = str . find_first_of ( ' ] ' , pos ) ;
if ( end ! = std : : string : : npos ) {
acceptTag ( str . substr ( pos + 1 , end - pos - 1 ) ) ;
pos = end + 1 ;
}
else {
acceptChar ( c ) ;
pos + + ;
}
}
else {
acceptChar ( c ) ;
pos + + ;
}
}
endParse ( ) ;
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
protected :
2013-04-24 20:10:02 +02:00
virtual void acceptTag ( std : : string const & tag ) = 0 ;
2012-11-01 09:27:09 +01:00
virtual void acceptChar ( char c ) = 0 ;
virtual void endParse ( ) { }
2012-03-04 22:22:36 +01:00
2012-11-01 09:27:09 +01:00
private :
} ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
class TagExtracter : public TagParser {
public :
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
TagExtracter ( std : : set < std : : string > & tags )
: m_tags ( tags )
{ }
virtual ~ TagExtracter ( ) ;
2012-05-16 16:09:17 +02:00
2012-11-01 09:27:09 +01:00
void parse ( std : : string & description ) {
TagParser : : parse ( description ) ;
description = m_remainder ;
}
private :
2013-04-24 20:10:02 +02:00
virtual void acceptTag ( std : : string const & tag ) {
2013-03-25 10:25:31 +01:00
m_tags . insert ( toLower ( tag ) ) ;
2012-11-01 09:27:09 +01:00
}
virtual void acceptChar ( char c ) {
m_remainder + = c ;
}
2013-04-24 20:10:02 +02:00
TagExtracter & operator = ( TagExtracter const & ) ;
2012-11-01 09:28:26 +01:00
2012-11-01 09:27:09 +01:00
std : : set < std : : string > & m_tags ;
std : : string m_remainder ;
} ;
class Tag {
2011-05-24 09:23:02 +02:00
public :
2012-11-01 09:27:09 +01:00
Tag ( )
2012-11-13 23:04:29 +01:00
: m_isNegated ( false )
2012-11-01 09:27:09 +01:00
{ }
2013-04-24 20:10:02 +02:00
Tag ( std : : string const & name , bool isNegated )
2011-05-24 09:23:02 +02:00
: m_name ( name ) ,
2012-11-13 23:04:29 +01:00
m_isNegated ( isNegated )
2012-05-16 16:09:17 +02:00
{ }
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
std : : string getName ( ) const {
return m_name ;
}
bool isNegated ( ) const {
return m_isNegated ;
2011-05-24 09:23:02 +02:00
}
2012-11-01 09:27:09 +01:00
bool operator ! ( ) const {
return m_name . empty ( ) ;
2011-05-24 09:23:02 +02:00
}
private :
std : : string m_name ;
2012-11-01 09:27:09 +01:00
bool m_isNegated ;
2011-05-24 09:23:02 +02:00
} ;
2012-11-01 09:27:09 +01:00
class TagSet {
typedef std : : map < std : : string , Tag > TagMap ;
public :
2013-04-24 20:10:02 +02:00
void add ( Tag const & tag ) {
2013-03-25 10:25:31 +01:00
m_tags . insert ( std : : make_pair ( toLower ( tag . getName ( ) ) , tag ) ) ;
2012-11-01 09:27:09 +01:00
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
bool empty ( ) const {
return m_tags . empty ( ) ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
bool matches ( std : : set < std : : string > const & tags ) const {
2012-11-01 09:27:09 +01:00
TagMap : : const_iterator it = m_tags . begin ( ) ;
TagMap : : const_iterator itEnd = m_tags . end ( ) ;
for ( ; it ! = itEnd ; + + it ) {
bool found = tags . find ( it - > first ) ! = tags . end ( ) ;
if ( found = = it - > second . isNegated ( ) )
return false ;
2011-05-24 09:23:02 +02:00
}
2012-11-01 09:27:09 +01:00
return true ;
2011-05-24 09:23:02 +02:00
}
2012-11-01 09:27:09 +01:00
private :
TagMap m_tags ;
} ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
class TagExpression {
public :
2013-04-24 20:10:02 +02:00
bool matches ( std : : set < std : : string > const & tags ) const {
2012-11-01 09:27:09 +01:00
std : : vector < TagSet > : : const_iterator it = m_tagSets . begin ( ) ;
std : : vector < TagSet > : : const_iterator itEnd = m_tagSets . end ( ) ;
for ( ; it ! = itEnd ; + + it )
if ( it - > matches ( tags ) )
return true ;
return false ;
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
private :
friend class TagExpressionParser ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
std : : vector < TagSet > m_tagSets ;
} ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
class TagExpressionParser : public TagParser {
public :
TagExpressionParser ( TagExpression & exp )
: m_isNegated ( false ) ,
m_exp ( exp )
{ }
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
~ TagExpressionParser ( ) ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
private :
2013-04-24 20:10:02 +02:00
virtual void acceptTag ( std : : string const & tag ) {
2012-11-01 09:27:09 +01:00
m_currentTagSet . add ( Tag ( tag , m_isNegated ) ) ;
m_isNegated = false ;
}
virtual void acceptChar ( char c ) {
switch ( c ) {
case ' ~ ' :
m_isNegated = true ;
break ;
case ' , ' :
m_exp . m_tagSets . push_back ( m_currentTagSet ) ;
break ;
}
}
virtual void endParse ( ) {
if ( ! m_currentTagSet . empty ( ) )
m_exp . m_tagSets . push_back ( m_currentTagSet ) ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
TagExpressionParser & operator = ( TagExpressionParser const & ) ;
2012-11-01 09:28:26 +01:00
2012-11-01 09:27:09 +01:00
bool m_isNegated ;
TagSet m_currentTagSet ;
TagExpression & m_exp ;
} ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
} // end namespace Catch
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
# include <string>
# include <vector>
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
struct IfFilterMatches { enum DoWhat {
AutoDetectBehaviour ,
IncludeTests ,
ExcludeTests
} ; } ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
class TestCaseFilter {
enum WildcardPosition {
NoWildcard = 0 ,
WildcardAtStart = 1 ,
WildcardAtEnd = 2 ,
WildcardAtBothEnds = WildcardAtStart | WildcardAtEnd
} ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
public :
2013-04-24 20:10:02 +02:00
TestCaseFilter ( std : : string const & testSpec , IfFilterMatches : : DoWhat matchBehaviour = IfFilterMatches : : AutoDetectBehaviour )
2013-03-25 10:25:31 +01:00
: m_stringToMatch ( toLower ( testSpec ) ) ,
2012-11-01 09:27:09 +01:00
m_filterType ( matchBehaviour ) ,
m_wildcardPosition ( NoWildcard )
{
if ( m_filterType = = IfFilterMatches : : AutoDetectBehaviour ) {
if ( startsWith ( m_stringToMatch , " exclude: " ) ) {
m_stringToMatch = m_stringToMatch . substr ( 8 ) ;
m_filterType = IfFilterMatches : : ExcludeTests ;
}
else if ( startsWith ( m_stringToMatch , " ~ " ) ) {
m_stringToMatch = m_stringToMatch . substr ( 1 ) ;
m_filterType = IfFilterMatches : : ExcludeTests ;
}
else {
m_filterType = IfFilterMatches : : IncludeTests ;
}
}
2011-05-24 09:23:02 +02:00
2013-03-21 10:00:24 +01:00
if ( startsWith ( m_stringToMatch , " * " ) ) {
2012-11-01 09:27:09 +01:00
m_stringToMatch = m_stringToMatch . substr ( 1 ) ;
m_wildcardPosition = ( WildcardPosition ) ( m_wildcardPosition | WildcardAtStart ) ;
}
2013-03-21 10:00:24 +01:00
if ( endsWith ( m_stringToMatch , " * " ) ) {
2012-11-01 09:27:09 +01:00
m_stringToMatch = m_stringToMatch . substr ( 0 , m_stringToMatch . size ( ) - 1 ) ;
m_wildcardPosition = ( WildcardPosition ) ( m_wildcardPosition | WildcardAtEnd ) ;
}
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
IfFilterMatches : : DoWhat getFilterType ( ) const {
return m_filterType ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
bool shouldInclude ( TestCase const & testCase ) const {
2012-11-01 09:27:09 +01:00
return isMatch ( testCase ) = = ( m_filterType = = IfFilterMatches : : IncludeTests ) ;
}
private :
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
# ifdef __clang__
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Wunreachable-code"
# endif
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
bool isMatch ( TestCase const & testCase ) const {
2013-03-12 19:49:37 +01:00
std : : string name = testCase . getTestCaseInfo ( ) . name ;
2013-03-25 10:25:31 +01:00
toLowerInPlace ( name ) ;
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
switch ( m_wildcardPosition ) {
case NoWildcard :
return m_stringToMatch = = name ;
case WildcardAtStart :
return endsWith ( name , m_stringToMatch ) ;
case WildcardAtEnd :
return startsWith ( name , m_stringToMatch ) ;
case WildcardAtBothEnds :
return contains ( name , m_stringToMatch ) ;
}
throw std : : logic_error ( " Unhandled wildcard type " ) ;
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
# ifdef __clang__
# pragma clang diagnostic pop
# endif
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
std : : string m_stringToMatch ;
IfFilterMatches : : DoWhat m_filterType ;
WildcardPosition m_wildcardPosition ;
} ;
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
class TestCaseFilters {
public :
2013-04-24 20:10:02 +02:00
TestCaseFilters ( std : : string const & name ) : m_name ( name ) { }
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
std : : string getName ( ) const {
return m_name ;
}
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
void addFilter ( TestCaseFilter const & filter ) {
2012-11-01 09:27:09 +01:00
if ( filter . getFilterType ( ) = = IfFilterMatches : : ExcludeTests )
m_exclusionFilters . push_back ( filter ) ;
else
m_inclusionFilters . push_back ( filter ) ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
void addTags ( std : : string const & tagPattern ) {
2012-11-01 09:27:09 +01:00
TagExpression exp ;
TagExpressionParser ( exp ) . parse ( tagPattern ) ;
2012-05-16 16:09:17 +02:00
2012-11-01 09:27:09 +01:00
m_tagExpressions . push_back ( exp ) ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
bool shouldInclude ( TestCase const & testCase ) const {
2012-11-01 09:27:09 +01:00
if ( ! m_tagExpressions . empty ( ) ) {
std : : vector < TagExpression > : : const_iterator it = m_tagExpressions . begin ( ) ;
std : : vector < TagExpression > : : const_iterator itEnd = m_tagExpressions . end ( ) ;
for ( ; it ! = itEnd ; + + it )
if ( it - > matches ( testCase . getTags ( ) ) )
break ;
if ( it = = itEnd )
return false ;
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
if ( ! m_inclusionFilters . empty ( ) ) {
std : : vector < TestCaseFilter > : : const_iterator it = m_inclusionFilters . begin ( ) ;
std : : vector < TestCaseFilter > : : const_iterator itEnd = m_inclusionFilters . end ( ) ;
for ( ; it ! = itEnd ; + + it )
if ( it - > shouldInclude ( testCase ) )
break ;
if ( it = = itEnd )
return false ;
}
else if ( m_exclusionFilters . empty ( ) & & m_tagExpressions . empty ( ) ) {
return ! testCase . isHidden ( ) ;
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
std : : vector < TestCaseFilter > : : const_iterator it = m_exclusionFilters . begin ( ) ;
std : : vector < TestCaseFilter > : : const_iterator itEnd = m_exclusionFilters . end ( ) ;
for ( ; it ! = itEnd ; + + it )
if ( ! it - > shouldInclude ( testCase ) )
return false ;
return true ;
}
2012-08-31 09:10:36 +02:00
private :
2012-11-01 09:27:09 +01:00
std : : vector < TagExpression > m_tagExpressions ;
std : : vector < TestCaseFilter > m_inclusionFilters ;
std : : vector < TestCaseFilter > m_exclusionFilters ;
2012-08-31 09:10:36 +02:00
std : : string m_name ;
2011-05-24 09:23:02 +02:00
} ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_interfaces_config.h
# define TWOBLUECUBES_CATCH_INTERFACES_CONFIG_H_INCLUDED
2012-09-26 19:38:26 +02:00
2013-06-07 20:07:50 +02:00
# include <iostream>
# include <string>
2012-09-26 19:38:26 +02:00
namespace Catch {
2013-06-07 20:07:50 +02:00
struct IConfig : IShared {
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
virtual ~ IConfig ( ) ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
virtual bool allowThrows ( ) const = 0 ;
2013-06-07 20:07:50 +02:00
virtual std : : ostream & stream ( ) const = 0 ;
virtual std : : string name ( ) const = 0 ;
virtual bool includeSuccessfulResults ( ) const = 0 ;
virtual bool shouldDebugBreak ( ) const = 0 ;
virtual bool warnAboutMissingAssertions ( ) const = 0 ;
virtual int abortAfter ( ) const = 0 ;
2012-09-26 19:38:26 +02:00
} ;
2012-11-01 09:27:09 +01:00
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_stream.hpp
# define TWOBLUECUBES_CATCH_STREAM_HPP_INCLUDED
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_streambuf.h
# define TWOBLUECUBES_CATCH_STREAMBUF_H_INCLUDED
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
# include <streambuf>
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
namespace Catch {
2012-10-31 19:03:08 +01:00
2012-11-01 09:27:09 +01:00
class StreamBufBase : public std : : streambuf {
public :
2013-07-02 09:49:29 +02:00
virtual ~ StreamBufBase ( ) throw ( ) ;
2012-09-26 19:38:26 +02:00
} ;
2012-11-01 09:27:09 +01:00
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
# include <stdexcept>
# include <cstdio>
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
namespace Catch {
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
template < typename WriterF , size_t bufferSize = 256 >
class StreamBufImpl : public StreamBufBase {
char data [ bufferSize ] ;
WriterF m_writer ;
public :
StreamBufImpl ( ) {
setp ( data , data + sizeof ( data ) ) ;
2012-09-26 19:38:26 +02:00
}
2013-07-02 09:49:29 +02:00
~ StreamBufImpl ( ) throw ( ) {
2012-11-01 09:27:09 +01:00
sync ( ) ;
2012-09-26 19:38:26 +02:00
}
private :
2013-03-08 10:30:25 +01:00
int overflow ( int c ) {
2012-11-01 09:27:09 +01:00
sync ( ) ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
if ( c ! = EOF ) {
if ( pbase ( ) = = epptr ( ) )
m_writer ( std : : string ( 1 , static_cast < char > ( c ) ) ) ;
else
sputc ( static_cast < char > ( c ) ) ;
}
return 0 ;
2012-09-26 19:38:26 +02:00
}
2013-03-08 10:30:25 +01:00
int sync ( ) {
2012-11-01 09:27:09 +01:00
if ( pbase ( ) ! = pptr ( ) ) {
m_writer ( std : : string ( pbase ( ) , static_cast < std : : string : : size_type > ( pptr ( ) - pbase ( ) ) ) ) ;
setp ( pbase ( ) , epptr ( ) ) ;
2012-09-26 19:38:26 +02:00
}
2012-11-01 09:27:09 +01:00
return 0 ;
2012-09-26 19:38:26 +02:00
}
} ;
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
struct OutputDebugWriter {
2012-09-26 19:38:26 +02:00
2013-04-24 20:10:02 +02:00
void operator ( ) ( std : : string const & str ) {
2012-11-01 09:27:09 +01:00
writeToDebugConsole ( str ) ;
}
2012-09-26 19:38:26 +02:00
} ;
2012-11-01 09:27:09 +01:00
class Stream {
2012-09-26 19:38:26 +02:00
public :
2012-11-01 09:27:09 +01:00
Stream ( )
: streamBuf ( NULL ) , isOwned ( false )
2012-09-26 19:38:26 +02:00
{ }
2012-11-01 09:27:09 +01:00
Stream ( std : : streambuf * _streamBuf , bool _isOwned )
: streamBuf ( _streamBuf ) , isOwned ( _isOwned )
{ }
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
void release ( ) {
if ( isOwned ) {
delete streamBuf ;
streamBuf = NULL ;
isOwned = false ;
2012-09-26 19:38:26 +02:00
}
}
2012-11-01 09:27:09 +01:00
std : : streambuf * streamBuf ;
2012-10-31 19:03:08 +01:00
2012-11-01 09:27:09 +01:00
private :
bool isOwned ;
2012-09-26 19:38:26 +02:00
} ;
2012-11-01 09:27:09 +01:00
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
# include <memory>
2012-08-31 09:10:36 +02:00
# include <vector>
2012-11-01 09:27:09 +01:00
# include <string>
# include <iostream>
2011-05-24 09:23:02 +02:00
2013-01-26 21:18:30 +01:00
# ifndef CATCH_CONFIG_CONSOLE_WIDTH
# define CATCH_CONFIG_CONSOLE_WIDTH 80
# endif
2012-11-01 09:27:09 +01:00
2012-10-28 13:07:17 +01:00
namespace Catch {
2012-11-01 09:27:09 +01:00
struct ConfigData {
2013-06-07 20:07:50 +02:00
struct Verbosity { enum Level {
NoOutput = 0 ,
Quiet ,
Normal
} ; } ;
2012-11-01 09:27:09 +01:00
struct WarnAbout { enum What {
Nothing = 0x00 ,
NoAssertions = 0x01
} ; } ;
ConfigData ( )
2013-06-07 20:07:50 +02:00
: listTests ( false ) ,
listTags ( false ) ,
listReporters ( false ) ,
showSuccessfulTests ( false ) ,
2012-11-01 09:27:09 +01:00
shouldDebugBreak ( false ) ,
2013-06-07 20:07:50 +02:00
noThrow ( false ) ,
showHelp ( false ) ,
abortAfter ( - 1 ) ,
verbosity ( Verbosity : : Normal ) ,
2012-11-01 09:27:09 +01:00
warnings ( WarnAbout : : Nothing )
{ }
2013-06-07 20:07:50 +02:00
bool listTests ;
bool listTags ;
bool listReporters ;
bool showSuccessfulTests ;
2012-11-01 09:27:09 +01:00
bool shouldDebugBreak ;
2013-06-07 20:07:50 +02:00
bool noThrow ;
bool showHelp ;
int abortAfter ;
Verbosity : : Level verbosity ;
2012-11-01 09:27:09 +01:00
WarnAbout : : What warnings ;
2013-06-07 20:07:50 +02:00
std : : string reporterName ;
std : : string outputFilename ;
std : : string name ;
std : : string processName ;
std : : vector < std : : string > testsOrTags ;
2012-11-01 09:27:09 +01:00
} ;
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
class Config : public SharedImpl < IConfig > {
2012-11-01 09:27:09 +01:00
private :
2013-04-24 20:10:02 +02:00
Config ( Config const & other ) ;
Config & operator = ( Config const & other ) ;
2012-11-01 09:27:09 +01:00
virtual void dummy ( ) ;
2011-05-24 09:23:02 +02:00
public :
2012-09-07 18:52:35 +02:00
2012-11-01 09:27:09 +01:00
Config ( )
: m_os ( std : : cout . rdbuf ( ) )
{ }
2013-04-24 20:10:02 +02:00
Config ( ConfigData const & data )
2012-11-01 09:27:09 +01:00
: m_data ( data ) ,
m_os ( std : : cout . rdbuf ( ) )
2013-06-07 20:07:50 +02:00
{
if ( ! data . testsOrTags . empty ( ) ) {
std : : string groupName ;
for ( std : : size_t i = 0 ; i < data . testsOrTags . size ( ) ; + + i ) {
if ( i ! = 0 )
groupName + = " " ;
groupName + = data . testsOrTags [ i ] ;
}
TestCaseFilters filters ( groupName ) ;
for ( std : : size_t i = 0 ; i < data . testsOrTags . size ( ) ; + + i ) {
std : : string filter = data . testsOrTags [ i ] ;
if ( startsWith ( filter , " [ " ) | | startsWith ( filter , " ~[ " ) )
filters . addTags ( filter ) ;
else
filters . addFilter ( TestCaseFilter ( filter ) ) ;
}
m_filterSets . push_back ( filters ) ;
}
}
2012-11-01 09:27:09 +01:00
virtual ~ Config ( ) {
m_os . rdbuf ( std : : cout . rdbuf ( ) ) ;
m_stream . release ( ) ;
2011-05-24 09:23:02 +02:00
}
2013-04-24 20:10:02 +02:00
void setFilename ( std : : string const & filename ) {
2012-11-01 09:27:09 +01:00
m_data . outputFilename = filename ;
2012-08-31 09:10:36 +02:00
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
std : : string const & getFilename ( ) const {
2012-11-01 09:27:09 +01:00
return m_data . outputFilename ;
}
2011-06-07 15:13:57 +02:00
2013-06-07 20:07:50 +02:00
bool listTests ( ) const { return m_data . listTests ; }
bool listTags ( ) const { return m_data . listTags ; }
bool listReporters ( ) const { return m_data . listReporters ; }
2011-06-07 15:13:57 +02:00
2013-06-07 20:07:50 +02:00
std : : string getProcessName ( ) const {
return m_data . processName ;
2012-11-01 09:27:09 +01:00
}
bool shouldDebugBreak ( ) const {
return m_data . shouldDebugBreak ;
}
void setStreamBuf ( std : : streambuf * buf ) {
m_os . rdbuf ( buf ? buf : std : : cout . rdbuf ( ) ) ;
}
2013-04-24 20:10:02 +02:00
void useStream ( std : : string const & streamName ) {
2012-11-01 09:27:09 +01:00
Stream stream = createStream ( streamName ) ;
setStreamBuf ( stream . streamBuf ) ;
m_stream . release ( ) ;
m_stream = stream ;
}
2013-06-07 20:07:50 +02:00
std : : string getReporterName ( ) const { return m_data . reporterName ; }
2013-04-24 20:10:02 +02:00
void addTestSpec ( std : : string const & testSpec ) {
2012-11-01 09:27:09 +01:00
TestCaseFilters filters ( testSpec ) ;
filters . addFilter ( TestCaseFilter ( testSpec ) ) ;
2013-06-07 20:07:50 +02:00
m_filterSets . push_back ( filters ) ;
2012-11-01 09:27:09 +01:00
}
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
int abortAfter ( ) const {
return m_data . abortAfter ;
2012-11-01 09:27:09 +01:00
}
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
std : : vector < TestCaseFilters > const & filters ( ) const {
return m_filterSets ;
2012-11-01 09:27:09 +01:00
}
2012-08-31 09:10:36 +02:00
2013-06-07 20:07:50 +02:00
bool showHelp ( ) const { return m_data . showHelp ; }
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
// IConfig interface
virtual bool allowThrows ( ) const { return ! m_data . noThrow ; }
virtual std : : ostream & stream ( ) const { return m_os ; }
virtual std : : string name ( ) const { return m_data . name . empty ( ) ? m_data . processName : m_data . name ; }
virtual bool includeSuccessfulResults ( ) const { return m_data . showSuccessfulTests ; }
virtual bool warnAboutMissingAssertions ( ) const { return m_data . warnings & ConfigData : : WarnAbout : : NoAssertions ; }
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
private :
ConfigData m_data ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
Stream m_stream ;
mutable std : : ostream m_os ;
2013-06-07 20:07:50 +02:00
std : : vector < TestCaseFilters > m_filterSets ;
2012-10-28 13:07:17 +01:00
} ;
2011-05-24 09:23:02 +02:00
2012-10-28 13:07:17 +01:00
} // end namespace Catch
2012-08-31 09:10:36 +02:00
2012-12-06 09:44:51 +01:00
// #included from: catch_option.hpp
# define TWOBLUECUBES_CATCH_OPTION_HPP_INCLUDED
namespace Catch {
// An optional type
template < typename T >
class Option {
public :
Option ( ) : nullableValue ( NULL ) { }
Option ( T const & _value )
: nullableValue ( new ( storage ) T ( _value ) )
{ }
Option ( Option const & _other )
: nullableValue ( _other ? new ( storage ) T ( * _other ) : NULL )
{ }
~ Option ( ) {
reset ( ) ;
}
Option & operator = ( Option const & _other ) {
reset ( ) ;
if ( _other )
nullableValue = new ( storage ) T ( * _other ) ;
return * this ;
}
void reset ( ) {
if ( nullableValue )
nullableValue - > ~ T ( ) ;
nullableValue = NULL ;
}
T & operator * ( ) { return * nullableValue ; }
2013-04-24 20:10:02 +02:00
T const & operator * ( ) const { return * nullableValue ; }
2012-12-06 09:44:51 +01:00
T * operator - > ( ) { return nullableValue ; }
const T * operator - > ( ) const { return nullableValue ; }
2013-06-07 20:07:50 +02:00
T valueOr ( T const & defaultValue ) const {
return nullableValue ? * nullableValue : defaultValue ;
}
2012-12-06 09:44:51 +01:00
bool some ( ) const { return nullableValue ! = NULL ; }
bool none ( ) const { return nullableValue = = NULL ; }
bool operator ! ( ) const { return nullableValue = = NULL ; }
operator SafeBool : : type ( ) const {
return SafeBool : : makeSafe ( some ( ) ) ;
}
private :
T * nullableValue ;
char storage [ sizeof ( T ) ] ;
2012-11-01 09:27:09 +01:00
} ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
} // end namespace Catch
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
# include <string>
# include <ostream>
# include <map>
namespace Catch
{
2012-12-02 00:57:18 +01:00
struct ReporterConfig {
2013-06-07 20:07:50 +02:00
explicit ReporterConfig ( Ptr < IConfig > const & _fullConfig )
: m_stream ( & _fullConfig - > stream ( ) ) , m_fullConfig ( _fullConfig ) { }
ReporterConfig ( Ptr < IConfig > const & _fullConfig , std : : ostream & _stream )
2012-12-02 00:57:18 +01:00
: m_stream ( & _stream ) , m_fullConfig ( _fullConfig ) { }
2013-06-07 20:07:50 +02:00
std : : ostream & stream ( ) const { return * m_stream ; }
Ptr < IConfig > fullConfig ( ) const { return m_fullConfig ; }
2012-12-02 00:57:18 +01:00
private :
std : : ostream * m_stream ;
2013-06-07 20:07:50 +02:00
Ptr < IConfig > m_fullConfig ;
2012-12-02 00:57:18 +01:00
} ;
struct ReporterPreferences {
ReporterPreferences ( )
: shouldRedirectStdOut ( false )
2012-10-28 13:07:17 +01:00
{ }
2012-09-26 19:38:26 +02:00
2012-12-02 00:57:18 +01:00
bool shouldRedirectStdOut ;
} ;
struct TestRunInfo {
TestRunInfo ( std : : string const & _name ) : name ( _name ) { }
std : : string name ;
} ;
struct GroupInfo {
2013-01-16 10:44:43 +01:00
GroupInfo ( std : : string const & _name ,
std : : size_t _groupIndex ,
std : : size_t _groupsCount )
2012-11-01 09:27:09 +01:00
: name ( _name ) ,
2013-01-16 10:44:43 +01:00
groupIndex ( _groupIndex ) ,
groupsCounts ( _groupsCount )
2012-11-01 09:27:09 +01:00
{ }
2012-12-02 00:57:18 +01:00
std : : string name ;
2013-01-16 10:44:43 +01:00
std : : size_t groupIndex ;
std : : size_t groupsCounts ;
2012-12-02 00:57:18 +01:00
} ;
struct SectionInfo {
SectionInfo ( std : : string const & _name ,
std : : string const & _description ,
SourceLineInfo const & _lineInfo )
: name ( _name ) ,
description ( _description ) ,
lineInfo ( _lineInfo )
2012-11-01 09:27:09 +01:00
{ }
std : : string name ;
2012-12-02 00:57:18 +01:00
std : : string description ;
SourceLineInfo lineInfo ;
} ;
2012-09-15 18:53:27 +02:00
2012-12-06 09:44:51 +01:00
struct ThreadedSectionInfo : SectionInfo , SharedImpl < > {
2013-01-26 21:18:30 +01:00
ThreadedSectionInfo ( SectionInfo const & _sectionInfo , ThreadedSectionInfo * _parent = NULL )
2012-12-06 09:44:51 +01:00
: SectionInfo ( _sectionInfo ) ,
2013-01-16 10:44:43 +01:00
parent ( _parent )
2012-12-06 09:44:51 +01:00
{ }
virtual ~ ThreadedSectionInfo ( ) ;
std : : vector < Ptr < ThreadedSectionInfo > > children ;
2013-01-26 21:18:30 +01:00
ThreadedSectionInfo * parent ;
2012-08-31 09:10:36 +02:00
} ;
2013-01-16 10:44:43 +01:00
struct AssertionStats {
2012-12-02 00:57:18 +01:00
AssertionStats ( AssertionResult const & _assertionResult ,
2013-02-02 21:37:58 +01:00
std : : vector < MessageInfo > const & _infoMessages ,
2012-12-02 00:57:18 +01:00
Totals const & _totals )
: assertionResult ( _assertionResult ) ,
2013-02-02 21:37:58 +01:00
infoMessages ( _infoMessages ) ,
2012-12-02 00:57:18 +01:00
totals ( _totals )
2013-02-02 21:37:58 +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 ( ) ;
2013-06-28 18:09:57 +02:00
builder . m_info . message = builder . m_stream . str ( ) ;
infoMessages . push_back ( builder . m_info ) ;
2013-02-02 21:37:58 +01:00
}
}
2012-12-02 00:57:18 +01:00
virtual ~ AssertionStats ( ) ;
2012-08-31 09:10:36 +02:00
2012-12-02 00:57:18 +01:00
AssertionResult assertionResult ;
2013-02-02 21:37:58 +01:00
std : : vector < MessageInfo > infoMessages ;
2012-12-02 00:57:18 +01:00
Totals totals ;
2012-08-31 09:10:36 +02:00
} ;
2013-01-16 10:44:43 +01:00
struct SectionStats {
2012-12-02 00:57:18 +01:00
SectionStats ( SectionInfo const & _sectionInfo ,
Counts const & _assertions ,
bool _missingAssertions )
: sectionInfo ( _sectionInfo ) ,
assertions ( _assertions ) ,
missingAssertions ( _missingAssertions )
{ }
virtual ~ SectionStats ( ) ;
2012-08-31 09:10:36 +02:00
2012-12-02 00:57:18 +01:00
SectionInfo sectionInfo ;
Counts assertions ;
bool missingAssertions ;
} ;
2013-01-16 10:44:43 +01:00
struct TestCaseStats {
2012-12-02 00:57:18 +01:00
TestCaseStats ( TestCaseInfo const & _testInfo ,
Totals const & _totals ,
std : : string const & _stdOut ,
std : : string const & _stdErr ,
bool _missingAssertions ,
bool _aborting )
: testInfo ( _testInfo ) ,
totals ( _totals ) ,
stdOut ( _stdOut ) ,
stdErr ( _stdErr ) ,
missingAssertions ( _missingAssertions ) ,
aborting ( _aborting )
{ }
virtual ~ TestCaseStats ( ) ;
TestCaseInfo testInfo ;
Totals totals ;
std : : string stdOut ;
std : : string stdErr ;
bool missingAssertions ;
bool aborting ;
} ;
2013-01-16 10:44:43 +01:00
struct TestGroupStats {
2012-12-02 00:57:18 +01:00
TestGroupStats ( GroupInfo const & _groupInfo ,
Totals const & _totals ,
bool _aborting )
: groupInfo ( _groupInfo ) ,
totals ( _totals ) ,
aborting ( _aborting )
{ }
2013-01-16 10:44:43 +01:00
TestGroupStats ( GroupInfo const & _groupInfo )
: groupInfo ( _groupInfo ) ,
aborting ( false )
{ }
2012-12-02 00:57:18 +01:00
virtual ~ TestGroupStats ( ) ;
GroupInfo groupInfo ;
Totals totals ;
bool aborting ;
} ;
2013-01-16 10:44:43 +01:00
struct TestRunStats {
2012-12-02 00:57:18 +01:00
TestRunStats ( TestRunInfo const & _runInfo ,
Totals const & _totals ,
bool _aborting )
: runInfo ( _runInfo ) ,
totals ( _totals ) ,
aborting ( _aborting )
{ }
2013-01-16 10:44:43 +01:00
TestRunStats ( TestRunStats const & _other )
: runInfo ( _other . runInfo ) ,
totals ( _other . totals ) ,
aborting ( _other . aborting )
{ }
2012-12-02 00:57:18 +01:00
virtual ~ TestRunStats ( ) ;
2012-08-31 09:10:36 +02:00
2012-12-02 00:57:18 +01:00
TestRunInfo runInfo ;
Totals totals ;
bool aborting ;
} ;
2012-08-31 09:10:36 +02:00
2012-12-02 00:57:18 +01:00
struct IStreamingReporter : IShared {
virtual ~ IStreamingReporter ( ) ;
2012-08-31 09:10:36 +02:00
2013-01-16 10:44:43 +01:00
// Implementing class must also provide the following static method:
2012-12-06 09:44:51 +01:00
// static std::string getDescription();
2012-09-26 19:38:26 +02:00
2012-12-02 00:57:18 +01:00
virtual ReporterPreferences getPreferences ( ) const = 0 ;
2012-09-29 21:54:03 +02:00
2013-03-13 13:19:30 +01:00
virtual void noMatchingTestCases ( std : : string const & spec ) = 0 ;
2012-12-02 00:57:18 +01:00
virtual void testRunStarting ( TestRunInfo const & testRunInfo ) = 0 ;
virtual void testGroupStarting ( GroupInfo const & groupInfo ) = 0 ;
virtual void testCaseStarting ( TestCaseInfo const & testInfo ) = 0 ;
virtual void sectionStarting ( SectionInfo const & sectionInfo ) = 0 ;
virtual void assertionStarting ( AssertionInfo const & assertionInfo ) = 0 ;
2012-08-31 09:10:36 +02:00
2013-06-28 18:09:57 +02:00
virtual bool assertionEnded ( AssertionStats const & assertionStats ) = 0 ;
2013-01-16 10:44:43 +01:00
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-12-02 00:57:18 +01:00
} ;
2012-12-06 09:44:51 +01:00
2013-01-16 10:44:43 +01:00
struct StreamingReporterBase : SharedImpl < IStreamingReporter > {
StreamingReporterBase ( ReporterConfig const & _config )
2013-06-07 20:07:50 +02:00
: m_config ( _config . fullConfig ( ) ) ,
2012-12-10 09:54:57 +01:00
stream ( _config . stream ( ) )
2012-12-06 09:44:51 +01:00
{ }
2013-01-16 10:44:43 +01:00
virtual ~ StreamingReporterBase ( ) ;
2012-12-06 09:44:51 +01:00
2013-03-13 13:19:30 +01:00
virtual void noMatchingTestCases ( std : : string const & ) { }
2012-12-06 09:44:51 +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:18:30 +01:00
m_rootSections . push_back ( currentSectionInfo ) ;
2012-12-06 09:44:51 +01:00
}
else {
currentSectionInfo - > children . push_back ( sectionInfo ) ;
2013-01-26 21:18:30 +01:00
sectionInfo - > parent = currentSectionInfo . get ( ) ;
2012-12-06 09:44:51 +01:00
currentSectionInfo = sectionInfo ;
}
}
2013-01-16 10:44:43 +01:00
virtual void sectionEnded ( SectionStats const & /* _sectionStats */ ) {
2012-12-06 09:44:51 +01:00
currentSectionInfo = currentSectionInfo - > parent ;
}
2013-01-16 10:44:43 +01:00
virtual void testCaseEnded ( TestCaseStats const & /* _testCaseStats */ ) {
2012-12-06 09:44:51 +01:00
unusedTestCaseInfo . reset ( ) ;
}
2013-01-16 10:44:43 +01:00
virtual void testGroupEnded ( TestGroupStats const & /* _testGroupStats */ ) {
2012-12-06 09:44:51 +01:00
unusedGroupInfo . reset ( ) ;
}
2013-01-16 10:44:43 +01:00
virtual void testRunEnded ( TestRunStats const & /* _testRunStats */ ) {
currentSectionInfo . reset ( ) ;
unusedTestCaseInfo . reset ( ) ;
unusedGroupInfo . reset ( ) ;
testRunInfo . reset ( ) ;
2012-12-06 09:44:51 +01:00
}
2013-06-07 20:07:50 +02:00
Ptr < IConfig > m_config ;
2012-12-06 09:44:51 +01:00
Option < TestRunInfo > testRunInfo ;
Option < GroupInfo > unusedGroupInfo ;
Option < TestCaseInfo > unusedTestCaseInfo ;
Ptr < ThreadedSectionInfo > currentSectionInfo ;
2013-01-16 10:44:43 +01:00
std : : ostream & stream ;
2013-01-26 21:18:30 +01:00
// !TBD: This should really go in the TestCaseStats class
std : : vector < Ptr < ThreadedSectionInfo > > m_rootSections ;
2013-01-16 10:44:43 +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-12-02 00:57:18 +01:00
// Deprecated
2012-10-28 13:07:17 +01:00
struct IReporter : IShared {
virtual ~ IReporter ( ) ;
2012-08-31 09:10:36 +02:00
2012-10-28 13:07:17 +01:00
virtual bool shouldRedirectStdout ( ) const = 0 ;
2012-09-29 21:54:03 +02:00
2012-10-28 13:07:17 +01:00
virtual void StartTesting ( ) = 0 ;
2012-12-02 00:57:18 +01:00
virtual void EndTesting ( Totals const & totals ) = 0 ;
virtual void StartGroup ( std : : string const & groupName ) = 0 ;
virtual void EndGroup ( std : : string const & groupName , Totals const & totals ) = 0 ;
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-24 20:10:02 +02:00
virtual void EndSection ( std : : string const & sectionName , Counts const & assertions ) = 0 ;
2012-12-02 00:57:18 +01:00
virtual void NoAssertionsInSection ( std : : string const & sectionName ) = 0 ;
virtual void NoAssertionsInTestCase ( std : : string const & testName ) = 0 ;
virtual void Aborted ( ) = 0 ;
2013-04-24 20:10:02 +02:00
virtual void Result ( AssertionResult const & result ) = 0 ;
2012-11-01 09:27:09 +01:00
} ;
2012-09-29 21:54:03 +02:00
2012-11-01 09:27:09 +01:00
struct IReporterFactory {
virtual ~ IReporterFactory ( ) ;
2012-12-02 00:57:18 +01:00
virtual IStreamingReporter * create ( ReporterConfig const & config ) const = 0 ;
2012-11-01 09:27:09 +01:00
virtual std : : string getDescription ( ) const = 0 ;
} ;
struct IReporterRegistry {
typedef std : : map < std : : string , IReporterFactory * > FactoryMap ;
virtual ~ IReporterRegistry ( ) ;
2013-06-07 20:07:50 +02:00
virtual IStreamingReporter * create ( std : : string const & name , Ptr < IConfig > const & config ) const = 0 ;
2013-04-24 20:10:02 +02:00
virtual FactoryMap const & getFactories ( ) const = 0 ;
2012-09-29 21:54:03 +02:00
} ;
2012-11-01 09:27:09 +01:00
2012-12-02 00:57:18 +01:00
inline std : : string trim ( std : : string const & str ) {
2012-11-01 09:27:09 +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 " ) ;
return start ! = std : : string : : npos ? str . substr ( start , 1 + end - start ) : " " ;
}
2012-09-29 21:54:03 +02:00
}
2012-11-01 09:27:09 +01:00
# include <vector>
2012-09-26 19:38:26 +02:00
namespace Catch {
2012-12-02 00:57:18 +01:00
class TestCase ;
2012-11-01 09:27:09 +01:00
struct ITestCaseRegistry ;
struct IExceptionTranslatorRegistry ;
struct IExceptionTranslator ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
struct IRegistryHub {
virtual ~ IRegistryHub ( ) ;
2012-09-26 19:38:26 +02:00
2013-04-24 20:10:02 +02:00
virtual IReporterRegistry const & getReporterRegistry ( ) const = 0 ;
virtual ITestCaseRegistry const & getTestCaseRegistry ( ) const = 0 ;
2012-11-01 09:27:09 +01:00
virtual IExceptionTranslatorRegistry & getExceptionTranslatorRegistry ( ) = 0 ;
} ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
struct IMutableRegistryHub {
virtual ~ IMutableRegistryHub ( ) ;
2013-04-24 20:10:02 +02:00
virtual void registerReporter ( std : : string const & name , IReporterFactory * factory ) = 0 ;
virtual void registerTest ( TestCase const & testInfo ) = 0 ;
2012-11-01 09:27:09 +01:00
virtual void registerTranslator ( const IExceptionTranslator * translator ) = 0 ;
} ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
IRegistryHub & getRegistryHub ( ) ;
IMutableRegistryHub & getMutableRegistryHub ( ) ;
void cleanUp ( ) ;
std : : string translateActiveException ( ) ;
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
# include <ostream>
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
namespace Catch {
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
inline IResultCapture & getResultCapture ( ) {
return getCurrentContext ( ) . getResultCapture ( ) ;
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
template < typename MatcherT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder expressionResultBuilderFromMatcher ( MatcherT const & matcher ,
std : : string const & matcherCallAsString ) {
2012-11-01 09:27:09 +01:00
std : : string matcherAsString = matcher . toString ( ) ;
if ( matcherAsString = = " {?} " )
matcherAsString = matcherCallAsString ;
return ExpressionResultBuilder ( )
. setRhs ( matcherAsString )
. setOp ( " matches " ) ;
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
template < typename MatcherT , typename ArgT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder expressionResultBuilderFromMatcher ( MatcherT const & matcher ,
ArgT const & arg ,
std : : string const & matcherCallAsString ) {
2012-11-01 09:27:09 +01:00
return expressionResultBuilderFromMatcher ( matcher , matcherCallAsString )
. setLhs ( Catch : : toString ( arg ) )
. setResultType ( matcher . match ( arg ) ) ;
}
2012-09-26 19:38:26 +02:00
2012-11-01 09:27:09 +01:00
template < typename MatcherT , typename ArgT >
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder expressionResultBuilderFromMatcher ( MatcherT const & matcher ,
2012-11-01 09:27:09 +01:00
ArgT * arg ,
2013-04-24 20:10:02 +02:00
std : : string const & matcherCallAsString ) {
2012-11-01 09:27:09 +01:00
return expressionResultBuilderFromMatcher ( matcher , matcherCallAsString )
. setLhs ( Catch : : toString ( arg ) )
. setResultType ( matcher . match ( arg ) ) ;
}
struct TestFailureException { } ;
} // end namespace Catch
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
# define INTERNAL_CATCH_ASSERTIONINFO_NAME INTERNAL_CATCH_UNIQUE_NAME( __assertionInfo )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_ACCEPT_EXPR( evaluatedExpr, resultDisposition, originalExpr ) \
2012-11-01 09:27:09 +01:00
if ( Catch : : ResultAction : : Value internal_catch_action = Catch : : getResultCapture ( ) . acceptExpression ( evaluatedExpr , INTERNAL_CATCH_ASSERTIONINFO_NAME ) ) { \
if ( internal_catch_action & Catch : : ResultAction : : Debug ) BreakIntoDebugger ( ) ; \
if ( internal_catch_action & Catch : : ResultAction : : Abort ) throw Catch : : TestFailureException ( ) ; \
2012-11-13 23:04:29 +01:00
if ( ! Catch : : shouldContinueOnFailure ( resultDisposition ) ) throw Catch : : TestFailureException ( ) ; \
2013-03-11 19:38:29 +01:00
Catch : : isTrue ( false & & originalExpr ) ; \
2012-11-01 09:27:09 +01:00
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_ACCEPT_INFO( expr, macroName, resultDisposition ) \
Catch : : AssertionInfo INTERNAL_CATCH_ASSERTIONINFO_NAME ( macroName , CATCH_INTERNAL_LINEINFO , expr , resultDisposition ) ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_TEST( expr, resultDisposition, macroName ) \
2012-11-01 09:27:09 +01:00
do { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_INFO ( # expr , macroName , resultDisposition ) ; \
2012-11-01 09:27:09 +01:00
try { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( ( Catch : : ExpressionDecomposer ( ) - > * expr ) . endExpression ( resultDisposition ) , resultDisposition , expr ) ; \
2012-11-01 09:27:09 +01:00
} catch ( Catch : : TestFailureException & ) { \
throw ; \
} catch ( . . . ) { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( Catch : : ExpressionResultBuilder ( Catch : : ResultWas : : ThrewException ) < < Catch : : translateActiveException ( ) , \
resultDisposition | Catch : : ResultDisposition : : ContinueOnFailure , expr ) ; \
2012-11-01 09:27:09 +01:00
} \
} while ( Catch : : isTrue ( false ) )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_IF( expr, resultDisposition, macroName ) \
INTERNAL_CATCH_TEST ( expr , resultDisposition , macroName ) ; \
if ( Catch : : getResultCapture ( ) . getLastResult ( ) - > succeeded ( ) )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_ELSE( expr, resultDisposition, macroName ) \
INTERNAL_CATCH_TEST ( expr , resultDisposition , macroName ) ; \
if ( ! Catch : : getResultCapture ( ) . getLastResult ( ) - > succeeded ( ) )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_NO_THROW( expr, resultDisposition, macroName ) \
2012-11-01 09:27:09 +01:00
do { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_INFO ( # expr , macroName , resultDisposition ) ; \
2012-11-01 09:27:09 +01:00
try { \
expr ; \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( Catch : : ExpressionResultBuilder ( Catch : : ResultWas : : Ok ) , resultDisposition , false ) ; \
2012-11-01 09:27:09 +01:00
} \
catch ( . . . ) { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( Catch : : ExpressionResultBuilder ( Catch : : ResultWas : : ThrewException ) < < Catch : : translateActiveException ( ) , resultDisposition , false ) ; \
2012-11-01 09:27:09 +01:00
} \
} while ( Catch : : isTrue ( false ) )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_THROWS_IMPL( expr, exceptionType, resultDisposition ) \
2012-11-01 09:27:09 +01:00
try { \
if ( Catch : : getCurrentContext ( ) . getConfig ( ) - > allowThrows ( ) ) { \
expr ; \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( Catch : : ExpressionResultBuilder ( Catch : : ResultWas : : DidntThrowException ) , resultDisposition , false ) ; \
2012-11-01 09:27:09 +01:00
} \
} \
catch ( Catch : : TestFailureException & ) { \
throw ; \
} \
catch ( exceptionType ) { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( Catch : : ExpressionResultBuilder ( Catch : : ResultWas : : Ok ) , resultDisposition , false ) ; \
2012-11-01 09:27:09 +01:00
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_THROWS( expr, exceptionType, resultDisposition, macroName ) \
2012-11-01 09:27:09 +01:00
do { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_INFO ( # expr , macroName , resultDisposition ) ; \
INTERNAL_CATCH_THROWS_IMPL ( expr , exceptionType , resultDisposition ) \
2012-11-01 09:27:09 +01:00
} while ( Catch : : isTrue ( false ) )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CATCH_THROWS_AS( expr, exceptionType, resultDisposition, macroName ) \
2012-11-01 09:27:09 +01:00
do { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_INFO ( # expr , macroName , resultDisposition ) ; \
INTERNAL_CATCH_THROWS_IMPL ( expr , exceptionType , resultDisposition ) \
2012-11-01 09:27:09 +01:00
catch ( . . . ) { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( ( Catch : : ExpressionResultBuilder ( Catch : : ResultWas : : ThrewException ) < < Catch : : translateActiveException ( ) ) , \
resultDisposition | Catch : : ResultDisposition : : ContinueOnFailure , false ) ; \
2012-11-01 09:27:09 +01:00
} \
} while ( Catch : : isTrue ( false ) )
2012-08-31 09:10:36 +02:00
2013-02-02 21:37:58 +01:00
///////////////////////////////////////////////////////////////////////////////
# define INTERNAL_CATCH_MSG( log, messageType, resultDisposition, macroName ) \
2012-11-13 23:04:29 +01:00
do { \
INTERNAL_CATCH_ACCEPT_INFO ( " " , macroName , resultDisposition ) ; \
2013-02-02 21:37:58 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( Catch : : ExpressionResultBuilder ( messageType ) < < log , resultDisposition , true ) \
2012-11-13 23:04:29 +01:00
} while ( Catch : : isTrue ( false ) )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2013-06-28 18:09:57 +02:00
# define INTERNAL_CATCH_INFO( log, macroName ) \
Catch : : ScopedMessage INTERNAL_CATCH_UNIQUE_NAME ( scopedMessage ) = Catch : : MessageBuilder ( macroName , CATCH_INTERNAL_LINEINFO , Catch : : ResultWas : : Info ) < < log ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
///////////////////////////////////////////////////////////////////////////////
2012-11-13 23:04:29 +01:00
# define INTERNAL_CHECK_THAT( arg, matcher, resultDisposition, macroName ) \
2012-11-01 09:27:09 +01:00
do { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_INFO ( # arg " " # matcher , macroName , resultDisposition ) ; \
2012-11-01 09:27:09 +01:00
try { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( ( Catch : : expressionResultBuilderFromMatcher ( : : Catch : : Matchers : : matcher , arg , # matcher ) ) , resultDisposition , false ) ; \
2012-11-01 09:27:09 +01:00
} catch ( Catch : : TestFailureException & ) { \
throw ; \
} catch ( . . . ) { \
2012-11-13 23:04:29 +01:00
INTERNAL_CATCH_ACCEPT_EXPR ( ( Catch : : ExpressionResultBuilder ( Catch : : ResultWas : : ThrewException ) < < Catch : : translateActiveException ( ) ) , \
resultDisposition | Catch : : ResultDisposition : : ContinueOnFailure , false ) ; \
2012-11-01 09:27:09 +01:00
} \
} while ( Catch : : isTrue ( false ) )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
// #included from: internal/catch_section.hpp
# define TWOBLUECUBES_CATCH_SECTION_HPP_INCLUDED
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
# include <string>
namespace Catch {
class Section {
public :
2013-04-24 20:10:02 +02:00
Section ( SourceLineInfo const & lineInfo ,
std : : string const & name ,
std : : string const & description = " " )
2012-12-02 00:57:18 +01:00
: m_info ( name , description , lineInfo ) ,
m_sectionIncluded ( getCurrentContext ( ) . getResultCapture ( ) . sectionStarted ( m_info , m_assertions ) )
2012-08-31 09:10:36 +02:00
{ }
2012-11-01 09:27:09 +01:00
~ Section ( ) {
if ( m_sectionIncluded )
2012-12-02 00:57:18 +01:00
getCurrentContext ( ) . getResultCapture ( ) . sectionEnded ( m_info , m_assertions ) ;
2012-08-31 09:10:36 +02:00
}
2012-11-01 09:27:09 +01:00
// This indicates whether the section should be executed or not
operator bool ( ) {
return m_sectionIncluded ;
2012-08-31 09:10:36 +02:00
}
2012-11-01 09:27:09 +01:00
private :
2012-12-02 00:57:18 +01:00
SectionInfo m_info ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
std : : string m_name ;
Counts m_assertions ;
bool m_sectionIncluded ;
} ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
} // end namespace Catch
2012-08-31 09:10:36 +02:00
2013-03-16 21:21:51 +01:00
# ifdef CATCH_CONFIG_VARIADIC_MACROS
# define INTERNAL_CATCH_SECTION( ... ) \
if ( Catch : : Section INTERNAL_CATCH_UNIQUE_NAME ( catch_internal_Section ) = Catch : : Section ( CATCH_INTERNAL_LINEINFO , __VA_ARGS__ ) )
# else
# define INTERNAL_CATCH_SECTION( name, desc ) \
if ( Catch : : Section INTERNAL_CATCH_UNIQUE_NAME ( catch_internal_Section ) = Catch : : Section ( CATCH_INTERNAL_LINEINFO , name , desc ) )
# endif
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
// #included from: internal/catch_generators.hpp
# define TWOBLUECUBES_CATCH_GENERATORS_HPP_INCLUDED
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
# include <iterator>
# include <vector>
# include <string>
# include <stdlib.h>
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
namespace Catch {
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
template < typename T >
struct IGenerator {
virtual ~ IGenerator ( ) { }
virtual T getValue ( std : : size_t index ) const = 0 ;
virtual std : : size_t size ( ) const = 0 ;
} ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
template < typename T >
class BetweenGenerator : public IGenerator < T > {
public :
BetweenGenerator ( T from , T to ) : m_from ( from ) , m_to ( to ) { }
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
virtual T getValue ( std : : size_t index ) const {
2013-01-16 10:44:43 +01:00
return m_from + static_cast < int > ( index ) ;
2012-11-01 09:27:09 +01:00
}
2012-09-07 18:52:35 +02:00
2012-11-01 09:27:09 +01:00
virtual std : : size_t size ( ) const {
return static_cast < std : : size_t > ( 1 + m_to - m_from ) ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
private :
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
T m_from ;
T m_to ;
} ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
template < typename T >
class ValuesGenerator : public IGenerator < T > {
public :
ValuesGenerator ( ) { }
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
void add ( T value ) {
m_values . push_back ( value ) ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
virtual T getValue ( std : : size_t index ) const {
return m_values [ index ] ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
virtual std : : size_t size ( ) const {
return m_values . size ( ) ;
}
private :
std : : vector < T > m_values ;
} ;
template < typename T >
class CompositeGenerator {
public :
CompositeGenerator ( ) : m_totalSize ( 0 ) { }
2013-03-08 10:30:25 +01:00
// *** Move semantics, similar to auto_ptr ***
2012-11-01 09:27:09 +01:00
CompositeGenerator ( CompositeGenerator & other )
: m_fileInfo ( other . m_fileInfo ) ,
m_totalSize ( 0 )
{
2013-03-08 10:30:25 +01:00
move ( other ) ;
2012-11-01 09:27:09 +01:00
}
CompositeGenerator & setFileInfo ( const char * fileInfo ) {
m_fileInfo = fileInfo ;
return * this ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
~ CompositeGenerator ( ) {
deleteAll ( m_composed ) ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
operator T ( ) const {
size_t overallIndex = getCurrentContext ( ) . getGeneratorIndex ( m_fileInfo , m_totalSize ) ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
typename std : : vector < const IGenerator < T > * > : : const_iterator it = m_composed . begin ( ) ;
typename std : : vector < const IGenerator < T > * > : : const_iterator itEnd = m_composed . end ( ) ;
for ( size_t index = 0 ; it ! = itEnd ; + + it )
{
const IGenerator < T > * generator = * it ;
if ( overallIndex > = index & & overallIndex < index + generator - > size ( ) )
{
return generator - > getValue ( overallIndex - index ) ;
}
index + = generator - > size ( ) ;
}
CATCH_INTERNAL_ERROR ( " Indexed past end of generated range " ) ;
2013-03-08 10:30:25 +01:00
return T ( ) ; // Suppress spurious "not all control paths return a value" warning in Visual Studio - if you know how to fix this please do so
2012-11-01 09:27:09 +01:00
}
2012-09-24 09:30:13 +02:00
2012-11-01 09:27:09 +01:00
void add ( const IGenerator < T > * generator ) {
m_totalSize + = generator - > size ( ) ;
m_composed . push_back ( generator ) ;
}
2012-09-24 09:30:13 +02:00
2012-11-01 09:27:09 +01:00
CompositeGenerator & then ( CompositeGenerator & other ) {
move ( other ) ;
return * this ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
CompositeGenerator & then ( T value ) {
ValuesGenerator < T > * valuesGen = new ValuesGenerator < T > ( ) ;
valuesGen - > add ( value ) ;
add ( valuesGen ) ;
return * this ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
private :
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
void move ( CompositeGenerator & other ) {
std : : copy ( other . m_composed . begin ( ) , other . m_composed . end ( ) , std : : back_inserter ( m_composed ) ) ;
m_totalSize + = other . m_totalSize ;
other . m_composed . clear ( ) ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
std : : vector < const IGenerator < T > * > m_composed ;
std : : string m_fileInfo ;
size_t m_totalSize ;
} ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
namespace Generators
{
template < typename T >
CompositeGenerator < T > between ( T from , T to ) {
CompositeGenerator < T > generators ;
generators . add ( new BetweenGenerator < T > ( from , to ) ) ;
return generators ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
template < typename T >
CompositeGenerator < T > values ( T val1 , T val2 ) {
CompositeGenerator < T > generators ;
ValuesGenerator < T > * valuesGen = new ValuesGenerator < T > ( ) ;
valuesGen - > add ( val1 ) ;
valuesGen - > add ( val2 ) ;
generators . add ( valuesGen ) ;
return generators ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
template < typename T >
CompositeGenerator < T > values ( T val1 , T val2 , T val3 ) {
CompositeGenerator < T > generators ;
ValuesGenerator < T > * valuesGen = new ValuesGenerator < T > ( ) ;
valuesGen - > add ( val1 ) ;
valuesGen - > add ( val2 ) ;
valuesGen - > add ( val3 ) ;
generators . add ( valuesGen ) ;
return generators ;
2012-08-31 09:10:36 +02:00
}
2012-11-01 09:27:09 +01:00
template < typename T >
CompositeGenerator < T > values ( T val1 , T val2 , T val3 , T val4 ) {
CompositeGenerator < T > generators ;
ValuesGenerator < T > * valuesGen = new ValuesGenerator < T > ( ) ;
valuesGen - > add ( val1 ) ;
valuesGen - > add ( val2 ) ;
valuesGen - > add ( val3 ) ;
valuesGen - > add ( val4 ) ;
generators . add ( valuesGen ) ;
return generators ;
}
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
} // end namespace Generators
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
using namespace Generators ;
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
} // end namespace Catch
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
# define INTERNAL_CATCH_LINESTR2( line ) #line
# define INTERNAL_CATCH_LINESTR( line ) INTERNAL_CATCH_LINESTR2( line )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
# define INTERNAL_CATCH_GENERATE( expr ) expr.setFileInfo( __FILE__ "(" INTERNAL_CATCH_LINESTR( __LINE__ ) ")" )
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
// #included from: internal/catch_interfaces_exception.h
# define TWOBLUECUBES_CATCH_INTERFACES_EXCEPTION_H_INCLUDED
2012-08-31 09:10:36 +02:00
2012-11-01 09:27:09 +01:00
# include <string>
2012-08-31 09:10:36 +02:00
namespace Catch {
typedef std : : string ( * exceptionTranslateFunction ) ( ) ;
struct IExceptionTranslator {
virtual ~ IExceptionTranslator ( ) ;
virtual std : : string translate ( ) const = 0 ;
} ;
struct IExceptionTranslatorRegistry {
virtual ~ IExceptionTranslatorRegistry ( ) ;
virtual std : : string translateActiveException ( ) const = 0 ;
} ;
class ExceptionTranslatorRegistrar {
template < typename T >
class ExceptionTranslator : public IExceptionTranslator {
public :
ExceptionTranslator ( std : : string ( * translateFunction ) ( T & ) )
: m_translateFunction ( translateFunction )
{ }
virtual std : : string translate ( ) const {
try {
throw ;
}
catch ( T & ex ) {
return m_translateFunction ( ex ) ;
}
}
protected :
std : : string ( * m_translateFunction ) ( T & ) ;
} ;
public :
template < typename T >
ExceptionTranslatorRegistrar ( std : : string ( * translateFunction ) ( T & ) ) {
getMutableRegistryHub ( ) . registerTranslator
( new ExceptionTranslator < T > ( translateFunction ) ) ;
}
} ;
}
///////////////////////////////////////////////////////////////////////////////
# define INTERNAL_CATCH_TRANSLATE_EXCEPTION( signature ) \
static std : : string INTERNAL_CATCH_UNIQUE_NAME ( catch_internal_ExceptionTranslator ) ( signature ) ; \
namespace { Catch : : ExceptionTranslatorRegistrar INTERNAL_CATCH_UNIQUE_NAME ( catch_internal_ExceptionRegistrar ) ( & INTERNAL_CATCH_UNIQUE_NAME ( catch_internal_ExceptionTranslator ) ) ; } \
static std : : string INTERNAL_CATCH_UNIQUE_NAME ( catch_internal_ExceptionTranslator ) ( signature )
// #included from: internal/catch_approx.hpp
# define TWOBLUECUBES_CATCH_APPROX_HPP_INCLUDED
# include <cmath>
# include <limits>
namespace Catch {
namespace Detail {
class Approx {
public :
explicit Approx ( double value )
: m_epsilon ( std : : numeric_limits < float > : : epsilon ( ) * 100 ) ,
m_scale ( 1.0 ) ,
m_value ( value )
{ }
2013-04-24 20:10:02 +02:00
Approx ( Approx const & other )
2012-08-31 09:10:36 +02:00
: m_epsilon ( other . m_epsilon ) ,
m_scale ( other . m_scale ) ,
m_value ( other . m_value )
{ }
static Approx custom ( ) {
return Approx ( 0 ) ;
}
Approx operator ( ) ( double value ) {
Approx approx ( value ) ;
approx . epsilon ( m_epsilon ) ;
approx . scale ( m_scale ) ;
return approx ;
}
2013-04-24 20:10:02 +02:00
friend bool operator = = ( double lhs , Approx const & rhs ) {
2012-08-31 09:10:36 +02:00
// Thanks to Richard Harris for his help refining this formula
return fabs ( lhs - rhs . m_value ) < rhs . m_epsilon * ( rhs . m_scale + ( std : : max ) ( fabs ( lhs ) , fabs ( rhs . m_value ) ) ) ;
}
2013-04-24 20:10:02 +02:00
friend bool operator = = ( Approx const & lhs , double rhs ) {
2012-08-31 09:10:36 +02:00
return operator = = ( rhs , lhs ) ;
}
2013-04-24 20:10:02 +02:00
friend bool operator ! = ( double lhs , Approx const & rhs ) {
2012-08-31 09:10:36 +02:00
return ! operator = = ( lhs , rhs ) ;
2012-05-16 16:09:17 +02:00
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
friend bool operator ! = ( Approx const & lhs , double rhs ) {
2012-05-16 16:09:17 +02:00
return ! operator = = ( rhs , lhs ) ;
}
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
Approx & epsilon ( double newEpsilon ) {
m_epsilon = newEpsilon ;
return * this ;
}
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
Approx & scale ( double newScale ) {
m_scale = newScale ;
return * this ;
}
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
std : : string toString ( ) const {
std : : ostringstream oss ;
2012-10-09 12:48:55 +02:00
oss < < " Approx( " < < m_value < < " ) " ;
2012-05-16 16:09:17 +02:00
return oss . str ( ) ;
}
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
private :
double m_epsilon ;
double m_scale ;
double m_value ;
} ;
}
template < >
2013-04-24 20:10:02 +02:00
inline std : : string toString < Detail : : Approx > ( Detail : : Approx const & value ) {
2012-05-16 16:09:17 +02:00
return value . toString ( ) ;
}
2011-05-24 09:23:02 +02:00
} // end namespace Catch
2012-05-21 19:52:09 +02:00
// #included from: internal/catch_matchers.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_MATCHERS_HPP_INCLUDED
2012-05-21 19:52:09 +02:00
namespace Catch {
namespace Matchers {
namespace Impl {
2012-10-12 08:58:17 +02:00
template < typename ExpressionT >
struct Matcher : SharedImpl < IShared >
{
2013-02-19 09:46:09 +01:00
typedef ExpressionT ExpressionType ;
2012-10-12 08:58:17 +02:00
virtual ~ Matcher ( ) { }
virtual Ptr < Matcher > clone ( ) const = 0 ;
2012-12-06 09:44:51 +01:00
virtual bool match ( ExpressionT const & expr ) const = 0 ;
2012-10-12 08:58:17 +02:00
virtual std : : string toString ( ) const = 0 ;
} ;
template < typename DerivedT , typename ExpressionT >
struct MatcherImpl : Matcher < ExpressionT > {
virtual Ptr < Matcher < ExpressionT > > clone ( ) const {
2013-04-24 20:10:02 +02:00
return Ptr < Matcher < ExpressionT > > ( new DerivedT ( static_cast < DerivedT const & > ( * this ) ) ) ;
2012-10-12 08:58:17 +02:00
}
} ;
namespace Generic {
2012-05-24 09:29:41 +02:00
2012-10-12 08:58:17 +02:00
template < typename ExpressionT >
class AllOf : public MatcherImpl < AllOf < ExpressionT > , ExpressionT > {
public :
AllOf ( ) { }
2013-04-24 20:10:02 +02:00
AllOf ( AllOf const & other ) : m_matchers ( other . m_matchers ) { }
2012-10-12 08:58:17 +02:00
2013-04-24 20:10:02 +02:00
AllOf & add ( Matcher < ExpressionT > const & matcher ) {
2012-10-12 08:58:17 +02:00
m_matchers . push_back ( matcher . clone ( ) ) ;
return * this ;
}
2013-04-24 20:10:02 +02:00
virtual bool match ( ExpressionT const & expr ) const
2012-05-24 09:29:41 +02:00
{
2012-10-12 08:58:17 +02:00
for ( std : : size_t i = 0 ; i < m_matchers . size ( ) ; + + i )
if ( ! m_matchers [ i ] - > match ( expr ) )
return false ;
return true ;
}
virtual std : : string toString ( ) const {
std : : ostringstream oss ;
oss < < " ( " ;
for ( std : : size_t i = 0 ; i < m_matchers . size ( ) ; + + i ) {
if ( i ! = 0 )
oss < < " and " ;
oss < < m_matchers [ i ] - > toString ( ) ;
}
oss < < " ) " ;
return oss . str ( ) ;
2012-05-24 09:29:41 +02:00
}
2012-10-12 08:58:17 +02:00
private :
std : : vector < Ptr < Matcher < ExpressionT > > > m_matchers ;
} ;
template < typename ExpressionT >
class AnyOf : public MatcherImpl < AnyOf < ExpressionT > , ExpressionT > {
public :
AnyOf ( ) { }
2013-04-24 20:10:02 +02:00
AnyOf ( AnyOf const & other ) : m_matchers ( other . m_matchers ) { }
2012-10-12 08:58:17 +02:00
2013-04-24 20:10:02 +02:00
AnyOf & add ( Matcher < ExpressionT > const & matcher ) {
2012-10-12 08:58:17 +02:00
m_matchers . push_back ( matcher . clone ( ) ) ;
return * this ;
}
2013-04-24 20:10:02 +02:00
virtual bool match ( ExpressionT const & expr ) const
2012-05-24 09:29:41 +02:00
{
2012-10-12 08:58:17 +02:00
for ( std : : size_t i = 0 ; i < m_matchers . size ( ) ; + + i )
if ( m_matchers [ i ] - > match ( expr ) )
return true ;
return false ;
}
virtual std : : string toString ( ) const {
std : : ostringstream oss ;
oss < < " ( " ;
for ( std : : size_t i = 0 ; i < m_matchers . size ( ) ; + + i ) {
if ( i ! = 0 )
oss < < " or " ;
oss < < m_matchers [ i ] - > toString ( ) ;
}
oss < < " ) " ;
return oss . str ( ) ;
2012-05-24 09:29:41 +02:00
}
2012-10-12 08:58:17 +02:00
private :
std : : vector < Ptr < Matcher < ExpressionT > > > m_matchers ;
} ;
}
namespace StdString {
2013-04-24 20:10:02 +02:00
inline std : : string makeString ( std : : string const & str ) { return str ; }
2013-04-23 09:02:48 +02:00
inline std : : string makeString ( const char * str ) { return str ? std : : string ( str ) : std : : string ( ) ; }
2012-10-12 08:58:17 +02:00
struct Equals : MatcherImpl < Equals , std : : string > {
2013-04-24 20:10:02 +02:00
Equals ( std : : string const & str ) : m_str ( str ) { }
Equals ( Equals const & other ) : m_str ( other . m_str ) { }
2012-10-12 08:58:17 +02:00
virtual ~ Equals ( ) ;
2013-04-24 20:10:02 +02:00
virtual bool match ( std : : string const & expr ) const {
2012-10-12 08:58:17 +02:00
return m_str = = expr ;
}
virtual std : : string toString ( ) const {
return " equals: \" " + m_str + " \" " ;
}
2012-08-31 09:10:36 +02:00
std : : string m_str ;
} ;
2012-03-04 22:22:36 +01:00
2012-10-12 08:58:17 +02:00
struct Contains : MatcherImpl < Contains , std : : string > {
2013-04-24 20:10:02 +02:00
Contains ( std : : string const & substr ) : m_substr ( substr ) { }
Contains ( Contains const & other ) : m_substr ( other . m_substr ) { }
2012-03-04 22:22:36 +01:00
2012-10-12 08:58:17 +02:00
virtual ~ Contains ( ) ;
2012-03-04 22:22:36 +01:00
2013-04-24 20:10:02 +02:00
virtual bool match ( std : : string const & expr ) const {
2012-10-12 08:58:17 +02:00
return expr . find ( m_substr ) ! = std : : string : : npos ;
2012-08-31 09:10:36 +02:00
}
2012-10-12 08:58:17 +02:00
virtual std : : string toString ( ) const {
return " contains: \" " + m_substr + " \" " ;
}
2012-08-31 09:10:36 +02:00
std : : string m_substr ;
} ;
2012-03-04 22:22:36 +01:00
2012-10-12 08:58:17 +02:00
struct StartsWith : MatcherImpl < StartsWith , std : : string > {
2013-04-24 20:10:02 +02:00
StartsWith ( std : : string const & substr ) : m_substr ( substr ) { }
StartsWith ( StartsWith const & other ) : m_substr ( other . m_substr ) { }
2012-03-04 22:22:36 +01:00
2012-10-12 08:58:17 +02:00
virtual ~ StartsWith ( ) ;
2012-03-04 22:22:36 +01:00
2013-04-24 20:10:02 +02:00
virtual bool match ( std : : string const & expr ) const {
2012-10-12 08:58:17 +02:00
return expr . find ( m_substr ) = = 0 ;
}
virtual std : : string toString ( ) const {
return " starts with: \" " + m_substr + " \" " ;
2012-08-31 09:10:36 +02:00
}
2012-10-12 08:58:17 +02:00
2012-08-31 09:10:36 +02:00
std : : string m_substr ;
} ;
2012-03-04 22:22:36 +01:00
2012-10-12 08:58:17 +02:00
struct EndsWith : MatcherImpl < EndsWith , std : : string > {
2013-04-24 20:10:02 +02:00
EndsWith ( std : : string const & substr ) : m_substr ( substr ) { }
EndsWith ( EndsWith const & other ) : m_substr ( other . m_substr ) { }
2012-03-14 21:06:14 +01:00
2012-10-12 08:58:17 +02:00
virtual ~ EndsWith ( ) ;
2012-03-14 21:06:14 +01:00
2013-04-24 20:10:02 +02:00
virtual bool match ( std : : string const & expr ) const {
2012-10-12 08:58:17 +02:00
return expr . find ( m_substr ) = = expr . size ( ) - m_substr . size ( ) ;
2012-08-31 09:10:36 +02:00
}
2012-10-12 08:58:17 +02:00
virtual std : : string toString ( ) const {
return " ends with: \" " + m_substr + " \" " ;
}
2012-08-31 09:10:36 +02:00
std : : string m_substr ;
} ;
} // namespace StdString
} // namespace Impl
2012-03-14 21:06:14 +01:00
2012-10-12 08:58:17 +02:00
// The following functions create the actual matcher objects.
// This allows the types to be inferred
template < typename ExpressionT >
2013-04-24 20:10:02 +02:00
inline Impl : : Generic : : AllOf < ExpressionT > AllOf ( Impl : : Matcher < ExpressionT > const & m1 ,
Impl : : Matcher < ExpressionT > const & m2 ) {
2012-10-12 08:58:17 +02:00
return Impl : : Generic : : AllOf < ExpressionT > ( ) . add ( m1 ) . add ( m2 ) ;
}
template < typename ExpressionT >
2013-04-24 20:10:02 +02:00
inline Impl : : Generic : : AllOf < ExpressionT > AllOf ( Impl : : Matcher < ExpressionT > const & m1 ,
Impl : : Matcher < ExpressionT > const & m2 ,
Impl : : Matcher < ExpressionT > const & m3 ) {
2012-10-12 08:58:17 +02:00
return Impl : : Generic : : AllOf < ExpressionT > ( ) . add ( m1 ) . add ( m2 ) . add ( m3 ) ;
}
template < typename ExpressionT >
2013-04-24 20:10:02 +02:00
inline Impl : : Generic : : AnyOf < ExpressionT > AnyOf ( Impl : : Matcher < ExpressionT > const & m1 ,
Impl : : Matcher < ExpressionT > const & m2 ) {
2012-10-12 08:58:17 +02:00
return Impl : : Generic : : AnyOf < ExpressionT > ( ) . add ( m1 ) . add ( m2 ) ;
}
template < typename ExpressionT >
2013-04-24 20:10:02 +02:00
inline Impl : : Generic : : AnyOf < ExpressionT > AnyOf ( Impl : : Matcher < ExpressionT > const & m1 ,
Impl : : Matcher < ExpressionT > const & m2 ,
Impl : : Matcher < ExpressionT > const & m3 ) {
2012-10-12 08:58:17 +02:00
return Impl : : Generic : : AnyOf < ExpressionT > ( ) . add ( m1 ) . add ( m2 ) . add ( m3 ) ;
}
2013-04-24 20:10:02 +02:00
inline Impl : : StdString : : Equals Equals ( std : : string const & str ) {
2013-04-23 09:02:48 +02:00
return Impl : : StdString : : Equals ( str ) ;
}
inline Impl : : StdString : : Equals Equals ( const char * str ) {
return Impl : : StdString : : Equals ( Impl : : StdString : : makeString ( str ) ) ;
}
2013-04-24 20:10:02 +02:00
inline Impl : : StdString : : Contains Contains ( std : : string const & substr ) {
2013-04-23 09:02:48 +02:00
return Impl : : StdString : : Contains ( substr ) ;
}
inline Impl : : StdString : : Contains Contains ( const char * substr ) {
return Impl : : StdString : : Contains ( Impl : : StdString : : makeString ( substr ) ) ;
}
2013-04-24 20:10:02 +02:00
inline Impl : : StdString : : StartsWith StartsWith ( std : : string const & substr ) {
2013-04-23 09:02:48 +02:00
return Impl : : StdString : : StartsWith ( substr ) ;
}
inline Impl : : StdString : : StartsWith StartsWith ( const char * substr ) {
return Impl : : StdString : : StartsWith ( Impl : : StdString : : makeString ( substr ) ) ;
}
2013-04-24 20:10:02 +02:00
inline Impl : : StdString : : EndsWith EndsWith ( std : : string const & substr ) {
2013-04-23 09:02:48 +02:00
return Impl : : StdString : : EndsWith ( substr ) ;
}
inline Impl : : StdString : : EndsWith EndsWith ( const char * substr ) {
return Impl : : StdString : : EndsWith ( Impl : : StdString : : makeString ( substr ) ) ;
}
2012-03-04 22:22:36 +01:00
2012-08-31 09:10:36 +02:00
} // namespace Matchers
2012-03-04 22:22:36 +01:00
2012-08-31 09:10:36 +02:00
using namespace Matchers ;
2012-03-04 22:22:36 +01:00
} // namespace Catch
2011-08-12 19:53:28 +02:00
2012-08-31 09:10:36 +02:00
// These files are included here so the single_include script doesn't put them
// in the conditionally compiled sections
// #included from: internal/catch_interfaces_runner.h
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_INTERFACES_RUNNER_H_INCLUDED
2011-08-12 19:53:28 +02:00
2012-08-31 09:10:36 +02:00
# include <string>
2011-08-12 19:53:28 +02:00
2012-08-31 09:10:36 +02:00
namespace Catch {
2012-12-02 00:57:18 +01:00
class TestCase ;
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
struct IRunner {
virtual ~ IRunner ( ) ;
} ;
}
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
# ifdef __OBJC__
// #included from: internal/catch_objc.hpp
# define TWOBLUECUBES_CATCH_OBJC_HPP_INCLUDED
2011-08-12 19:53:28 +02:00
2012-08-31 09:10:36 +02:00
# import <objc / runtime.h>
2011-08-12 19:53:28 +02:00
2012-08-31 09:10:36 +02:00
# include <string>
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
// NB. Any general catch headers included here must be included
// in catch.hpp first to make sure they are included by the single
// header for non obj-usage
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
///////////////////////////////////////////////////////////////////////////////
// This protocol is really only here for (self) documenting purposes, since
// all its methods are optional.
@ protocol OcFixture
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
@ optional
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
- ( void ) setUp ;
- ( void ) tearDown ;
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
@ end
2012-08-24 09:23:50 +02:00
2012-08-31 09:10:36 +02:00
namespace Catch {
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
class OcMethod : public SharedImpl < ITestCase > {
2012-08-13 08:46:10 +02:00
2012-08-31 09:10:36 +02:00
public :
OcMethod ( Class cls , SEL sel ) : m_cls ( cls ) , m_sel ( sel ) { }
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
virtual void invoke ( ) const {
id obj = [ [ m_cls alloc ] init ] ;
2012-08-27 22:48:15 +02:00
2012-08-31 09:10:36 +02:00
performOptionalSelector ( obj , @ selector ( setUp ) ) ;
performOptionalSelector ( obj , m_sel ) ;
performOptionalSelector ( obj , @ selector ( tearDown ) ) ;
2012-08-24 09:23:50 +02:00
2012-08-31 09:10:36 +02:00
arcSafeRelease ( obj ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-31 09:10:36 +02:00
private :
virtual ~ OcMethod ( ) { }
2012-05-16 16:09:17 +02:00
2012-08-31 09:10:36 +02:00
Class m_cls ;
SEL m_sel ;
2012-08-23 21:08:50 +02:00
} ;
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
namespace Detail {
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
inline bool startsWith ( std : : string const & str , std : : string const & sub ) {
2012-08-31 09:10:36 +02:00
return str . length ( ) > sub . length ( ) & & str . substr ( 0 , sub . length ( ) ) = = sub ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
inline std : : string getAnnotation ( Class cls ,
2013-04-24 20:10:02 +02:00
std : : string const & annotationName ,
std : : string const & testCaseName ) {
2012-08-31 09:10:36 +02:00
NSString * selStr = [ [ NSString alloc ] initWithFormat : @ " Catch_%s_%s " , annotationName . c_str ( ) , testCaseName . c_str ( ) ] ;
SEL sel = NSSelectorFromString ( selStr ) ;
arcSafeRelease ( selStr ) ;
id value = performOptionalSelector ( cls , sel ) ;
if ( value )
return [ ( NSString * ) value UTF8String ] ;
return " " ;
2012-08-23 21:08:50 +02:00
}
2012-08-31 09:10:36 +02:00
}
2012-07-05 19:37:58 +02:00
2012-08-31 09:10:36 +02:00
inline size_t registerTestMethods ( ) {
size_t noTestMethods = 0 ;
int noClasses = objc_getClassList ( NULL , 0 ) ;
2012-08-23 21:08:50 +02:00
2012-08-31 09:10:36 +02:00
Class * classes = ( CATCH_UNSAFE_UNRETAINED Class * ) malloc ( sizeof ( Class ) * noClasses ) ;
objc_getClassList ( classes , noClasses ) ;
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
for ( int c = 0 ; c < noClasses ; c + + ) {
Class cls = classes [ c ] ;
{
u_int count ;
Method * methods = class_copyMethodList ( cls , & count ) ;
for ( u_int m = 0 ; m < count ; m + + ) {
SEL selector = method_getName ( methods [ m ] ) ;
std : : string methodName = sel_getName ( selector ) ;
if ( Detail : : startsWith ( methodName , " Catch_TestCase_ " ) ) {
std : : string testCaseName = methodName . substr ( 15 ) ;
std : : string name = Detail : : getAnnotation ( cls , " Name " , testCaseName ) ;
std : : string desc = Detail : : getAnnotation ( cls , " Description " , testCaseName ) ;
2013-02-19 09:46:09 +01:00
const char * className = class_getName ( cls ) ;
2012-07-05 19:37:58 +02:00
2013-02-19 20:47:18 +01:00
getMutableRegistryHub ( ) . registerTest ( makeTestCase ( new OcMethod ( cls , selector ) , className , name . c_str ( ) , desc . c_str ( ) , SourceLineInfo ( ) ) ) ;
2012-08-31 09:10:36 +02:00
noTestMethods + + ;
}
}
free ( methods ) ;
}
}
return noTestMethods ;
}
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
namespace Matchers {
namespace Impl {
namespace NSStringMatchers {
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
template < typename MatcherT >
struct StringHolder : MatcherImpl < MatcherT , NSString * > {
2012-08-31 09:10:36 +02:00
StringHolder ( NSString * substr ) : m_substr ( [ substr copy ] ) { }
2013-02-19 09:46:09 +01:00
StringHolder ( StringHolder const & other ) : m_substr ( [ other . m_substr copy ] ) { }
2012-08-31 09:10:36 +02:00
StringHolder ( ) {
arcSafeRelease ( m_substr ) ;
}
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
NSString * m_substr ;
} ;
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
struct Equals : StringHolder < Equals > {
2012-08-31 09:10:36 +02:00
Equals ( NSString * substr ) : StringHolder ( substr ) { }
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
virtual bool match ( ExpressionType const & str ) const {
2012-08-31 09:10:36 +02:00
return [ str isEqualToString : m_substr ] ;
}
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
virtual std : : string toString ( ) const {
return " equals string: \" " + Catch : : toString ( m_substr ) + " \" " ;
2012-08-31 09:10:36 +02:00
}
} ;
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
struct Contains : StringHolder < Contains > {
2012-08-31 09:10:36 +02:00
Contains ( NSString * substr ) : StringHolder ( substr ) { }
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
virtual bool match ( ExpressionType const & str ) const {
2012-08-31 09:10:36 +02:00
return [ str rangeOfString : m_substr ] . location ! = NSNotFound ;
}
2012-08-13 08:46:10 +02:00
2013-02-19 09:46:09 +01:00
virtual std : : string toString ( ) const {
return " contains string: \" " + Catch : : toString ( m_substr ) + " \" " ;
2012-08-31 09:10:36 +02:00
}
} ;
2012-08-23 21:08:50 +02:00
2013-02-19 09:46:09 +01:00
struct StartsWith : StringHolder < StartsWith > {
2012-08-31 09:10:36 +02:00
StartsWith ( NSString * substr ) : StringHolder ( substr ) { }
2012-08-13 08:46:10 +02:00
2013-02-19 09:46:09 +01:00
virtual bool match ( ExpressionType const & str ) const {
2012-08-31 09:10:36 +02:00
return [ str rangeOfString : m_substr ] . location = = 0 ;
}
2012-08-13 08:46:10 +02:00
2013-02-19 09:46:09 +01:00
virtual std : : string toString ( ) const {
return " starts with: \" " + Catch : : toString ( m_substr ) + " \" " ;
2012-08-31 09:10:36 +02:00
}
} ;
2013-02-19 09:46:09 +01:00
struct EndsWith : StringHolder < EndsWith > {
2012-08-31 09:10:36 +02:00
EndsWith ( NSString * substr ) : StringHolder ( substr ) { }
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
virtual bool match ( ExpressionType const & str ) const {
2012-08-31 09:10:36 +02:00
return [ str rangeOfString : m_substr ] . location = = [ str length ] - [ m_substr length ] ;
}
2011-05-24 09:23:02 +02:00
2013-02-19 09:46:09 +01:00
virtual std : : string toString ( ) const {
return " ends with: \" " + Catch : : toString ( m_substr ) + " \" " ;
2012-08-31 09:10:36 +02:00
}
} ;
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
} // namespace NSStringMatchers
} // namespace Impl
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
inline Impl : : NSStringMatchers : : Equals
Equals ( NSString * substr ) { return Impl : : NSStringMatchers : : Equals ( substr ) ; }
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
inline Impl : : NSStringMatchers : : Contains
Contains ( NSString * substr ) { return Impl : : NSStringMatchers : : Contains ( substr ) ; }
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
inline Impl : : NSStringMatchers : : StartsWith
StartsWith ( NSString * substr ) { return Impl : : NSStringMatchers : : StartsWith ( substr ) ; }
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
inline Impl : : NSStringMatchers : : EndsWith
EndsWith ( NSString * substr ) { return Impl : : NSStringMatchers : : EndsWith ( substr ) ; }
2012-06-01 20:40:27 +02:00
2012-08-31 09:10:36 +02:00
} // namespace Matchers
2012-06-05 21:51:05 +02:00
2012-08-31 09:10:36 +02:00
using namespace Matchers ;
2012-06-05 21:51:05 +02:00
2012-08-31 09:10:36 +02:00
} // namespace Catch
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
///////////////////////////////////////////////////////////////////////////////
# define OC_TEST_CASE( name, desc )\
+ ( NSString * ) INTERNAL_CATCH_UNIQUE_NAME ( Catch_Name_test ) \
{ \
return @ name ; \
} \
+ ( NSString * ) INTERNAL_CATCH_UNIQUE_NAME ( Catch_Description_test ) \
{ \
return @ desc ; \
} \
- ( void ) INTERNAL_CATCH_UNIQUE_NAME ( Catch_TestCase_test )
2012-08-09 08:47:30 +02:00
2012-08-31 09:10:36 +02:00
# endif
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
# if defined( CATCH_CONFIG_MAIN ) || defined( CATCH_CONFIG_RUNNER )
// #included from: internal/catch_impl.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_IMPL_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
// Collect all the implementation files together here
// These are the equivalent of what would usually be cpp files
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
# ifdef __clang__
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Wweak-vtables"
# endif
2012-05-16 16:09:17 +02:00
2012-08-31 09:10:36 +02:00
// #included from: catch_runner.hpp
# define TWOBLUECUBES_CATCH_RUNNER_HPP_INCLUDED
2012-05-16 16:09:17 +02:00
2012-08-31 09:10:36 +02:00
// #included from: internal/catch_commandline.hpp
# define TWOBLUECUBES_CATCH_COMMANDLINE_HPP_INCLUDED
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
// #included from: clara.h
# define TWOBLUECUBES_CLARA_H_INCLUDED
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
// #included from: catch_text.h
# define TWOBLUECUBES_CATCH_TEXT_H_INCLUDED
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
# include <string>
# include <vector>
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
namespace Catch {
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
struct TextAttributes {
TextAttributes ( )
: initialIndent ( std : : string : : npos ) ,
indent ( 0 ) ,
width ( CATCH_CONFIG_CONSOLE_WIDTH - 1 ) ,
tabChar ( ' \t ' )
{ }
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
TextAttributes & setInitialIndent ( std : : size_t _value ) { initialIndent = _value ; return * this ; }
TextAttributes & setIndent ( std : : size_t _value ) { indent = _value ; return * this ; }
TextAttributes & setWidth ( std : : size_t _value ) { width = _value ; return * this ; }
TextAttributes & setTabChar ( char _value ) { tabChar = _value ; return * this ; }
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
std : : size_t initialIndent ; // indent of first line, or npos
std : : size_t indent ; // indent of subsequent lines, or all if initialIndent is npos
std : : size_t width ; // maximum width of text, including indent. Longer text will wrap
char tabChar ; // If this char is seen the indent is changed to current pos
2012-08-23 21:08:50 +02:00
} ;
2013-06-07 20:07:50 +02:00
class Text {
2012-08-23 21:08:50 +02:00
public :
2013-06-07 20:07:50 +02:00
Text ( std : : string const & _str , TextAttributes const & _attr = TextAttributes ( ) ) ;
void spliceLine ( std : : size_t _indent , std : : string & _remainder , std : : size_t _pos ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
typedef std : : vector < std : : string > : : const_iterator const_iterator ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
const_iterator begin ( ) const { return lines . begin ( ) ; }
const_iterator end ( ) const { return lines . end ( ) ; }
std : : string const & last ( ) const { return lines . back ( ) ; }
std : : size_t size ( ) const { return lines . size ( ) ; }
std : : string const & operator [ ] ( std : : size_t _index ) const { return lines [ _index ] ; }
std : : string toString ( ) const ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
friend std : : ostream & operator < < ( std : : ostream & _stream , Text const & _text ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
private :
std : : string str ;
TextAttributes attr ;
std : : vector < std : : string > lines ;
2012-08-23 21:08:50 +02:00
} ;
2013-06-07 20:07:50 +02:00
} // end namespace Catch
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
namespace Clara {
namespace Detail {
template < typename T > struct RemoveConstRef { typedef T type ; } ;
template < typename T > struct RemoveConstRef < T & > { typedef T type ; } ;
template < typename T > struct RemoveConstRef < T const & > { typedef T type ; } ;
template < typename T > struct RemoveConstRef < T const > { typedef T type ; } ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
template < typename T > struct IsBool { static const bool value = false ; } ;
template < > struct IsBool < bool > { static const bool value = true ; } ;
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
template < typename T >
void convertInto ( std : : string const & _source , T & _dest ) {
std : : stringstream ss ;
ss < < _source ;
ss > > _dest ;
if ( ss . fail ( ) )
throw std : : runtime_error ( " Unable to convert " + _source + " to destination type " ) ;
}
inline void convertInto ( std : : string const & _source , std : : string & _dest ) {
_dest = _source ;
}
inline void convertInto ( std : : string const & _source , bool & _dest ) {
std : : string sourceLC = _source ;
std : : transform ( sourceLC . begin ( ) , sourceLC . end ( ) , sourceLC . begin ( ) , : : tolower ) ;
if ( sourceLC = = " 1 " | | sourceLC = = " true " | | sourceLC = = " yes " | | sourceLC = = " on " )
_dest = true ;
else if ( sourceLC = = " 0 " | | sourceLC = = " false " | | sourceLC = = " no " | | sourceLC = = " off " )
_dest = false ;
else
throw std : : runtime_error ( " Expected a boolean value but did recognise: ' " + _source + " ' " ) ;
2012-08-23 21:08:50 +02:00
}
2013-06-07 20:07:50 +02:00
inline void convertInto ( bool _source , bool & _dest ) {
_dest = _source ;
2012-08-23 21:08:50 +02:00
}
2013-06-07 20:07:50 +02:00
template < typename T >
inline void convertInto ( bool , T & ) {
throw std : : runtime_error ( " Invalid conversion " ) ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
template < typename ConfigT >
struct IArgFunction {
virtual ~ IArgFunction ( ) { }
virtual void set ( ConfigT & config , std : : string const & value ) const = 0 ;
virtual void setFlag ( ConfigT & config ) const = 0 ;
virtual bool takesArg ( ) const = 0 ;
virtual IArgFunction * clone ( ) const = 0 ;
} ;
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
template < typename ConfigT >
class BoundArgFunction {
2012-08-27 22:42:55 +02:00
public :
2013-06-07 20:07:50 +02:00
BoundArgFunction ( IArgFunction < ConfigT > * _functionObj ) : functionObj ( _functionObj ) { }
BoundArgFunction ( BoundArgFunction const & other ) : functionObj ( other . functionObj - > clone ( ) ) { }
BoundArgFunction & operator = ( BoundArgFunction const & other ) {
IArgFunction < ConfigT > * newFunctionObj = other . functionObj - > clone ( ) ;
delete functionObj ;
functionObj = newFunctionObj ;
return * this ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
~ BoundArgFunction ( ) { delete functionObj ; }
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
void set ( ConfigT & config , std : : string const & value ) const {
functionObj - > set ( config , value ) ;
}
void setFlag ( ConfigT & config ) const {
functionObj - > setFlag ( config ) ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
bool takesArg ( ) const { return functionObj - > takesArg ( ) ; }
private :
IArgFunction < ConfigT > * functionObj ;
2012-08-27 22:42:55 +02:00
} ;
2013-06-07 20:07:50 +02:00
template < typename C >
struct NullBinder : IArgFunction < C > {
virtual void set ( C & , std : : string const & ) const { }
virtual void setFlag ( C & ) const { }
virtual bool takesArg ( ) const { return true ; }
virtual IArgFunction < C > * clone ( ) const { return new NullBinder ( * this ) ; }
2012-08-27 22:42:55 +02:00
} ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
template < typename C , typename M >
struct BoundDataMember : IArgFunction < C > {
BoundDataMember ( M C : : * _member ) : member ( _member ) { }
virtual void set ( C & p , std : : string const & stringValue ) const {
convertInto ( stringValue , p . * member ) ;
2012-09-24 09:30:13 +02:00
}
2013-06-07 20:07:50 +02:00
virtual void setFlag ( C & p ) const {
convertInto ( true , p . * member ) ;
2012-09-24 09:30:13 +02:00
}
2013-06-07 20:07:50 +02:00
virtual bool takesArg ( ) const { return ! IsBool < M > : : value ; }
virtual IArgFunction < C > * clone ( ) const { return new BoundDataMember ( * this ) ; }
M C : : * member ;
} ;
template < typename C , typename M >
struct BoundUnaryMethod : IArgFunction < C > {
BoundUnaryMethod ( void ( C : : * _member ) ( M ) ) : member ( _member ) { }
virtual void set ( C & p , std : : string const & stringValue ) const {
typename RemoveConstRef < M > : : type value ;
convertInto ( stringValue , value ) ;
( p . * member ) ( value ) ;
}
virtual void setFlag ( C & p ) const {
typename RemoveConstRef < M > : : type value ;
convertInto ( true , value ) ;
( p . * member ) ( value ) ;
}
virtual bool takesArg ( ) const { return ! IsBool < M > : : value ; }
virtual IArgFunction < C > * clone ( ) const { return new BoundUnaryMethod ( * this ) ; }
void ( C : : * member ) ( M ) ;
} ;
template < typename C >
struct BoundNullaryMethod : IArgFunction < C > {
BoundNullaryMethod ( void ( C : : * _member ) ( ) ) : member ( _member ) { }
virtual void set ( C & p , std : : string const & stringValue ) const {
bool value ;
convertInto ( stringValue , value ) ;
if ( value )
( p . * member ) ( ) ;
}
virtual void setFlag ( C & p ) const {
( p . * member ) ( ) ;
}
virtual bool takesArg ( ) const { return false ; }
virtual IArgFunction < C > * clone ( ) const { return new BoundNullaryMethod ( * this ) ; }
void ( C : : * member ) ( ) ;
} ;
2012-09-24 09:30:13 +02:00
2013-06-07 20:07:50 +02:00
template < typename C >
struct BoundUnaryFunction : IArgFunction < C > {
BoundUnaryFunction ( void ( * _function ) ( C & ) ) : function ( _function ) { }
virtual void set ( C & obj , std : : string const & stringValue ) const {
bool value ;
convertInto ( stringValue , value ) ;
if ( value )
function ( obj ) ;
}
virtual void setFlag ( C & p ) const {
function ( p ) ;
}
virtual bool takesArg ( ) const { return false ; }
virtual IArgFunction < C > * clone ( ) const { return new BoundUnaryFunction ( * this ) ; }
void ( * function ) ( C & ) ;
} ;
2012-09-24 09:30:13 +02:00
2013-06-07 20:07:50 +02:00
template < typename C , typename T >
struct BoundBinaryFunction : IArgFunction < C > {
BoundBinaryFunction ( void ( * _function ) ( C & , T ) ) : function ( _function ) { }
virtual void set ( C & obj , std : : string const & stringValue ) const {
typename RemoveConstRef < T > : : type value ;
convertInto ( stringValue , value ) ;
function ( obj , value ) ;
}
virtual void setFlag ( C & obj ) const {
typename RemoveConstRef < T > : : type value ;
convertInto ( true , value ) ;
function ( obj , value ) ;
}
virtual bool takesArg ( ) const { return ! IsBool < T > : : value ; }
virtual IArgFunction < C > * clone ( ) const { return new BoundBinaryFunction ( * this ) ; }
void ( * function ) ( C & , T ) ;
2012-09-24 09:30:13 +02:00
} ;
2013-06-07 20:07:50 +02:00
template < typename C , typename M >
BoundArgFunction < C > makeBoundField ( M C : : * _member ) {
return BoundArgFunction < C > ( new BoundDataMember < C , M > ( _member ) ) ;
}
template < typename C , typename M >
BoundArgFunction < C > makeBoundField ( void ( C : : * _member ) ( M ) ) {
return BoundArgFunction < C > ( new BoundUnaryMethod < C , M > ( _member ) ) ;
}
template < typename C >
BoundArgFunction < C > makeBoundField ( void ( C : : * _member ) ( ) ) {
return BoundArgFunction < C > ( new BoundNullaryMethod < C > ( _member ) ) ;
}
template < typename C >
BoundArgFunction < C > makeBoundField ( void ( * _function ) ( C & ) ) {
return BoundArgFunction < C > ( new BoundUnaryFunction < C > ( _function ) ) ;
}
template < typename C , typename T >
BoundArgFunction < C > makeBoundField ( void ( * _function ) ( C & , T ) ) {
return BoundArgFunction < C > ( new BoundBinaryFunction < C , T > ( _function ) ) ;
}
} // namespace Detail
struct Parser {
Parser ( ) : separators ( " \t =: " ) { }
struct Token {
enum Type { Positional , ShortOpt , LongOpt } ;
Token ( Type _type , std : : string const & _data ) : type ( _type ) , data ( _data ) { }
Type type ;
std : : string data ;
} ;
void parseIntoTokens ( int argc , char const * const * argv , std : : vector < Parser : : Token > & tokens ) const {
for ( int i = 1 ; i < argc ; + + i )
parseIntoTokens ( argv [ i ] , tokens ) ;
}
void parseIntoTokens ( std : : string arg , std : : vector < Parser : : Token > & tokens ) const {
while ( ! arg . empty ( ) ) {
Parser : : Token token ( Parser : : Token : : Positional , arg ) ;
arg = " " ;
if ( token . data [ 0 ] = = ' - ' ) {
if ( token . data . size ( ) > 1 & & token . data [ 1 ] = = ' - ' ) {
token = Parser : : Token ( Parser : : Token : : LongOpt , token . data . substr ( 2 ) ) ;
}
else {
token = Parser : : Token ( Parser : : Token : : ShortOpt , token . data . substr ( 1 ) ) ;
if ( token . data . size ( ) > 1 & & separators . find ( token . data [ 1 ] ) = = std : : string : : npos ) {
arg = " - " + token . data . substr ( 1 ) ;
token . data = token . data . substr ( 0 , 1 ) ;
}
}
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
if ( token . type ! = Parser : : Token : : Positional ) {
std : : size_t pos = token . data . find_first_of ( separators ) ;
if ( pos ! = std : : string : : npos ) {
arg = token . data . substr ( pos + 1 ) ;
token . data = token . data . substr ( 0 , pos ) ;
}
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
tokens . push_back ( token ) ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
}
std : : string separators ;
} ;
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
template < typename ConfigT >
class CommandLine {
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
struct Arg {
Arg ( Detail : : BoundArgFunction < ConfigT > const & _boundField ) : boundField ( _boundField ) , position ( - 1 ) { }
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
bool hasShortName ( std : : string const & shortName ) const {
for ( std : : vector < std : : string > : : const_iterator
it = shortNames . begin ( ) , itEnd = shortNames . end ( ) ;
it ! = itEnd ;
+ + it )
if ( * it = = shortName )
return true ;
return false ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
bool hasLongName ( std : : string const & _longName ) const {
return _longName = = longName ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
bool takesArg ( ) const {
return ! argName . empty ( ) ;
2012-09-09 12:44:30 +02:00
}
2013-06-07 20:07:50 +02:00
bool isFixedPositional ( ) const {
return position ! = - 1 ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
bool isAnyPositional ( ) const {
return position = = - 1 & & shortNames . empty ( ) & & longName . empty ( ) ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
std : : string dbgName ( ) const {
if ( ! longName . empty ( ) )
return " -- " + longName ;
if ( ! shortNames . empty ( ) )
return " - " + shortNames [ 0 ] ;
return " positional args " ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
void validate ( ) const {
if ( boundField . takesArg ( ) & & ! takesArg ( ) )
throw std : : logic_error ( dbgName ( ) + " must specify an arg name " ) ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
std : : string commands ( ) const {
std : : ostringstream oss ;
bool first = true ;
std : : vector < std : : string > : : const_iterator it = shortNames . begin ( ) , itEnd = shortNames . end ( ) ;
for ( ; it ! = itEnd ; + + it ) {
if ( first )
first = false ;
else
oss < < " , " ;
oss < < " - " < < * it ;
}
if ( ! longName . empty ( ) ) {
if ( ! first )
oss < < " , " ;
oss < < " -- " < < longName ;
}
if ( ! argName . empty ( ) )
oss < < " < " < < argName < < " > " ;
return oss . str ( ) ;
2012-09-09 12:44:30 +02:00
}
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
Detail : : BoundArgFunction < ConfigT > boundField ;
std : : vector < std : : string > shortNames ;
std : : string longName ;
std : : string description ;
std : : string argName ;
int position ;
2012-08-27 22:42:55 +02:00
} ;
2013-06-07 20:07:50 +02:00
class ArgBinder {
2012-08-27 22:42:55 +02:00
public :
2013-06-07 20:07:50 +02:00
template < typename F >
ArgBinder ( CommandLine * cl , F f )
: m_cl ( cl ) ,
m_arg ( Detail : : makeBoundField ( f ) )
{ }
ArgBinder ( ArgBinder & other )
: m_cl ( other . m_cl ) ,
m_arg ( other . m_arg )
{
other . m_cl = NULL ;
}
~ ArgBinder ( ) {
if ( m_cl ) {
m_arg . validate ( ) ;
if ( m_arg . isFixedPositional ( ) ) {
m_cl - > m_positionalArgs . insert ( std : : make_pair ( m_arg . position , m_arg ) ) ;
if ( m_arg . position > m_cl - > m_highestSpecifiedArgPosition )
m_cl - > m_highestSpecifiedArgPosition = m_arg . position ;
}
else if ( m_arg . isAnyPositional ( ) ) {
if ( m_cl - > m_arg . get ( ) )
throw std : : logic_error ( " Only one unpositional argument can be added " ) ;
m_cl - > m_arg = std : : auto_ptr < Arg > ( new Arg ( m_arg ) ) ;
}
else
m_cl - > m_options . push_back ( m_arg ) ;
}
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
ArgBinder & shortOpt ( std : : string const & name ) {
m_arg . shortNames . push_back ( name ) ;
return * this ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
ArgBinder & longOpt ( std : : string const & name ) {
m_arg . longName = name ;
return * this ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
ArgBinder & describe ( std : : string const & description ) {
m_arg . description = description ;
return * this ;
2012-09-09 12:44:30 +02:00
}
2013-06-07 20:07:50 +02:00
ArgBinder & argName ( std : : string const & argName ) {
m_arg . argName = argName ;
return * this ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
ArgBinder & position ( int position ) {
m_arg . position = position ;
return * this ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
private :
CommandLine * m_cl ;
Arg m_arg ;
2012-08-27 22:42:55 +02:00
} ;
2013-06-07 20:07:50 +02:00
public :
CommandLine ( )
: m_boundProcessName ( new Detail : : NullBinder < ConfigT > ( ) ) ,
m_highestSpecifiedArgPosition ( 0 )
{ }
CommandLine ( CommandLine const & other )
: m_boundProcessName ( other . m_boundProcessName ) ,
m_options ( other . m_options ) ,
m_positionalArgs ( other . m_positionalArgs ) ,
m_highestSpecifiedArgPosition ( other . m_highestSpecifiedArgPosition )
{
if ( other . m_arg . get ( ) )
m_arg = std : : auto_ptr < Arg > ( new Arg ( * other . m_arg ) ) ;
}
template < typename F >
ArgBinder bind ( F f ) {
ArgBinder binder ( this , f ) ;
return binder ;
}
template < typename F >
void bindProcessName ( F f ) {
m_boundProcessName = Detail : : makeBoundField ( f ) ;
}
void optUsage ( std : : ostream & os , std : : size_t indent = 0 , std : : size_t width = CATCH_CONFIG_CONSOLE_WIDTH ) const {
typename std : : vector < Arg > : : const_iterator itBegin = m_options . begin ( ) , itEnd = m_options . end ( ) , it ;
std : : size_t maxWidth = 0 ;
for ( it = itBegin ; it ! = itEnd ; + + it )
maxWidth = ( std : : max ) ( maxWidth , it - > commands ( ) . size ( ) ) ;
for ( it = itBegin ; it ! = itEnd ; + + it ) {
Catch : : Text usage ( it - > commands ( ) , Catch : : TextAttributes ( )
. setWidth ( maxWidth + indent )
. setIndent ( indent ) ) ;
// !TBD handle longer usage strings
Catch : : Text desc ( it - > description , Catch : : TextAttributes ( )
. setWidth ( width - maxWidth - 3 ) ) ;
2013-06-29 13:11:32 +02:00
for ( std : : size_t i = 0 ; i < ( std : : max ) ( usage . size ( ) , desc . size ( ) ) ; + + i ) {
2013-06-07 20:07:50 +02:00
std : : string usageCol = i < usage . size ( ) ? usage [ i ] : " " ;
os < < usageCol ;
if ( i < desc . size ( ) & & ! desc [ i ] . empty ( ) )
os < < std : : string ( indent + 2 + maxWidth - usageCol . size ( ) , ' ' )
< < desc [ i ] ;
os < < " \n " ;
2012-08-27 22:42:55 +02:00
}
}
2013-06-07 20:07:50 +02:00
}
std : : string optUsage ( ) const {
std : : ostringstream oss ;
optUsage ( oss ) ;
return oss . str ( ) ;
}
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
void argSynopsis ( std : : ostream & os ) const {
for ( int i = 1 ; i < = m_highestSpecifiedArgPosition ; + + i ) {
if ( i > 1 )
os < < " " ;
typename std : : map < int , Arg > : : const_iterator it = m_positionalArgs . find ( i ) ;
if ( it ! = m_positionalArgs . end ( ) )
os < < " < " < < it - > second . argName < < " > " ;
else if ( m_arg . get ( ) )
os < < " < " < < m_arg - > argName < < " > " ;
else
throw std : : logic_error ( " non consecutive positional arguments with no floating args " ) ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
// !TBD No indication of mandatory args
if ( m_arg . get ( ) ) {
if ( m_highestSpecifiedArgPosition > 1 )
os < < " " ;
os < < " [< " < < m_arg - > argName < < " > ...] " ;
2012-09-09 12:44:30 +02:00
}
2013-06-07 20:07:50 +02:00
}
std : : string argSynopsis ( ) const {
std : : ostringstream oss ;
argSynopsis ( oss ) ;
return oss . str ( ) ;
}
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
void usage ( std : : ostream & os , std : : string const & procName ) const {
os < < " usage: \n " < < procName < < " " ;
argSynopsis ( os ) ;
if ( ! m_options . empty ( ) ) {
os < < " [options] \n \n where options are: \n " ;
optUsage ( os , 2 ) ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
os < < " \n " ;
}
std : : string usage ( std : : string const & procName ) const {
std : : ostringstream oss ;
usage ( oss , procName ) ;
return oss . str ( ) ;
}
2012-08-31 09:10:36 +02:00
2013-06-07 20:07:50 +02:00
std : : vector < Parser : : Token > parseInto ( int argc , char const * const * argv , ConfigT & config ) const {
std : : string processName = argv [ 0 ] ;
std : : size_t lastSlash = processName . find_last_of ( " / \\ " ) ;
if ( lastSlash ! = std : : string : : npos )
processName = processName . substr ( lastSlash + 1 ) ;
m_boundProcessName . set ( config , processName ) ;
std : : vector < Parser : : Token > tokens ;
Parser parser ;
parser . parseIntoTokens ( argc , argv , tokens ) ;
return populate ( tokens , config ) ;
}
std : : vector < Parser : : Token > populate ( std : : vector < Parser : : Token > const & tokens , ConfigT & config ) const {
if ( m_options . empty ( ) & & m_positionalArgs . empty ( ) )
throw std : : logic_error ( " No options or arguments specified " ) ;
std : : vector < Parser : : Token > unusedTokens = populateOptions ( tokens , config ) ;
unusedTokens = populateFixedArgs ( unusedTokens , config ) ;
unusedTokens = populateFloatingArgs ( unusedTokens , config ) ;
return unusedTokens ;
}
std : : vector < Parser : : Token > populateOptions ( std : : vector < Parser : : Token > const & tokens , ConfigT & config ) const {
std : : vector < Parser : : Token > unusedTokens ;
for ( std : : size_t i = 0 ; i < tokens . size ( ) ; + + i ) {
Parser : : Token const & token = tokens [ i ] ;
typename std : : vector < Arg > : : const_iterator it = m_options . begin ( ) , itEnd = m_options . end ( ) ;
for ( ; it ! = itEnd ; + + it ) {
Arg const & arg = * it ;
try {
if ( ( token . type = = Parser : : Token : : ShortOpt & & arg . hasShortName ( token . data ) ) | |
( token . type = = Parser : : Token : : LongOpt & & arg . hasLongName ( token . data ) ) ) {
if ( arg . takesArg ( ) ) {
if ( i = = tokens . size ( ) - 1 | | tokens [ i + 1 ] . type ! = Parser : : Token : : Positional )
throw std : : domain_error ( " Expected argument to option " + token . data ) ;
arg . boundField . set ( config , tokens [ + + i ] . data ) ;
}
else {
arg . boundField . setFlag ( config ) ;
}
break ;
}
}
catch ( std : : exception & ex ) {
throw std : : runtime_error ( std : : string ( ex . what ( ) ) + " while parsing: ( " + arg . commands ( ) + " ) " ) ;
}
2012-08-31 09:10:36 +02:00
}
2013-06-07 20:07:50 +02:00
if ( it = = itEnd )
unusedTokens . push_back ( token ) ;
}
return unusedTokens ;
}
std : : vector < Parser : : Token > populateFixedArgs ( std : : vector < Parser : : Token > const & tokens , ConfigT & config ) const {
std : : vector < Parser : : Token > unusedTokens ;
int position = 1 ;
for ( std : : size_t i = 0 ; i < tokens . size ( ) ; + + i ) {
Parser : : Token const & token = tokens [ i ] ;
typename std : : map < int , Arg > : : const_iterator it = m_positionalArgs . find ( position ) ;
if ( it ! = m_positionalArgs . end ( ) )
it - > second . boundField . set ( config , token . data ) ;
else
unusedTokens . push_back ( token ) ;
if ( token . type = = Parser : : Token : : Positional )
position + + ;
}
return unusedTokens ;
}
std : : vector < Parser : : Token > populateFloatingArgs ( std : : vector < Parser : : Token > const & tokens , ConfigT & config ) const {
if ( ! m_arg . get ( ) )
return tokens ;
std : : vector < Parser : : Token > unusedTokens ;
for ( std : : size_t i = 0 ; i < tokens . size ( ) ; + + i ) {
Parser : : Token const & token = tokens [ i ] ;
if ( token . type = = Parser : : Token : : Positional )
m_arg - > boundField . set ( config , token . data ) ;
else
unusedTokens . push_back ( token ) ;
2012-08-31 09:10:36 +02:00
}
2013-06-07 20:07:50 +02:00
return unusedTokens ;
}
private :
Detail : : BoundArgFunction < ConfigT > m_boundProcessName ;
std : : vector < Arg > m_options ;
std : : map < int , Arg > m_positionalArgs ;
std : : auto_ptr < Arg > m_arg ;
int m_highestSpecifiedArgPosition ;
} ;
} // end namespace Clara
namespace Catch {
inline void abortAfterFirst ( ConfigData & config ) { config . abortAfter = 1 ; }
inline void abortAfterX ( ConfigData & config , int x ) {
if ( x < 1 )
throw std : : runtime_error ( " Value after -x or --abortAfter must be greater than zero " ) ;
config . abortAfter = x ;
2012-08-27 22:42:55 +02:00
}
2013-06-07 20:07:50 +02:00
inline void addTestOrTags ( ConfigData & config , std : : string const & _testSpec ) { config . testsOrTags . push_back ( _testSpec ) ; }
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
inline void addWarning ( ConfigData & config , std : : string const & _warning ) {
if ( _warning = = " NoAssertions " )
config . warnings = ( ConfigData : : WarnAbout : : What ) ( config . warnings | ConfigData : : WarnAbout : : NoAssertions ) ;
else
throw std : : runtime_error ( " Unrecognised warning: ' " + _warning + " ' " ) ;
2012-08-27 22:42:55 +02:00
2013-06-07 20:07:50 +02:00
}
inline void setVerbosity ( ConfigData & config , int level ) {
// !TBD: accept strings?
config . verbosity = ( ConfigData : : Verbosity : : Level ) level ;
}
2012-08-31 09:10:36 +02:00
2013-06-07 20:07:50 +02:00
inline Clara : : CommandLine < ConfigData > makeCommandLineParser ( ) {
2012-08-31 09:10:36 +02:00
2013-06-07 20:07:50 +02:00
Clara : : CommandLine < ConfigData > cli ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
cli . bindProcessName ( & ConfigData : : processName ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : showHelp )
. describe ( " display usage information " )
. shortOpt ( " ? " )
. shortOpt ( " h " )
. longOpt ( " help " ) ;
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : listTests )
. describe ( " list all (or matching) test cases " )
. shortOpt ( " l " )
. longOpt ( " list-tests " ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : listTags )
. describe ( " list all (or matching) tags " )
. shortOpt ( " t " )
. longOpt ( " list-tags " ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : listReporters )
. describe ( " list all reporters " )
. longOpt ( " list-reporters " ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : showSuccessfulTests )
. describe ( " include successful tests in output " )
. shortOpt ( " s " )
. longOpt ( " success " ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : shouldDebugBreak )
. describe ( " break into debugger on failure " )
. shortOpt ( " b " )
. longOpt ( " break " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : noThrow )
2013-06-07 22:15:25 +02:00
. describe ( " skip exception tests " )
2013-06-07 20:07:50 +02:00
. shortOpt ( " e " )
. longOpt ( " nothrow " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : outputFilename )
. describe ( " output filename " )
. shortOpt ( " o " )
. longOpt ( " out " )
. argName ( " filename " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : reporterName )
. describe ( " reporter to use - defaults to console " )
. shortOpt ( " r " )
. longOpt ( " reporter " )
2013-06-28 15:11:28 +02:00
// .argName( "name[:filename]" );
. argName ( " name " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & ConfigData : : name )
. describe ( " suite name " )
. shortOpt ( " n " )
. longOpt ( " name " )
. argName ( " name " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & abortAfterFirst )
. describe ( " abort at first failure " )
. shortOpt ( " a " )
. longOpt ( " abort " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & abortAfterX )
. describe ( " abort after x failures " )
. shortOpt ( " x " )
. longOpt ( " abortx " )
. argName ( " number of failures " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & addWarning )
. describe ( " enable warnings " )
. shortOpt ( " w " )
. longOpt ( " warn " )
. argName ( " warning name " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
// cli.bind( &setVerbosity )
// .describe( "level of verbosity (0=no output)" )
// .shortOpt( "v")
// .longOpt( "verbosity" )
// .argName( "level" );
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
cli . bind ( & addTestOrTags )
. describe ( " which test or tests to use " )
. argName ( " test name, pattern or tags " ) ;
2013-03-29 14:44:33 +01:00
2013-06-07 20:07:50 +02:00
return cli ;
}
2012-08-23 21:08:50 +02:00
} // end namespace Catch
// #included from: internal/catch_list.hpp
# define TWOBLUECUBES_CATCH_LIST_HPP_INCLUDED
2013-04-01 12:27:10 +02:00
// #included from: catch_console_colour.hpp
# define TWOBLUECUBES_CATCH_CONSOLE_COLOUR_HPP_INCLUDED
namespace Catch {
2013-04-05 21:58:01 +02:00
namespace Detail {
struct IColourImpl ;
}
struct Colour {
enum Code {
None = 0 ,
White ,
Red ,
Green ,
Blue ,
Cyan ,
Yellow ,
Grey ,
Bright = 0x10 ,
2013-04-01 12:27:10 +02:00
2013-04-05 21:58:01 +02:00
BrightRed = Bright | Red ,
BrightGreen = Bright | Green ,
LightGrey = Bright | Grey ,
BrightWhite = Bright | White ,
2013-04-01 12:27:10 +02:00
2013-04-05 21:58:01 +02:00
// By intention
FileName = LightGrey ,
ResultError = BrightRed ,
ResultSuccess = BrightGreen ,
2013-04-01 12:27:10 +02:00
2013-04-05 21:58:01 +02:00
Error = BrightRed ,
Success = Green ,
2013-04-01 12:27:10 +02:00
2013-04-05 21:58:01 +02:00
OriginalExpression = Cyan ,
ReconstructedExpression = Yellow ,
SecondaryText = LightGrey ,
Headers = White
2013-04-01 12:27:10 +02:00
} ;
2013-04-05 21:58:01 +02:00
// Use constructed object for RAII guard
Colour ( Code _colourCode ) ;
~ Colour ( ) ;
2013-04-01 12:27:10 +02:00
2013-04-05 21:58:01 +02:00
// Use static method for one-shot changes
static void use ( Code _colourCode ) ;
2013-04-01 12:27:10 +02:00
private :
2013-04-05 21:58:01 +02:00
static Detail : : IColourImpl * impl ;
2013-04-01 12:27:10 +02:00
} ;
} // end namespace Catch
2012-08-23 21:08:50 +02:00
# include <limits>
2013-03-29 14:44:33 +01:00
# include <algorithm>
2012-08-23 21:08:50 +02:00
namespace Catch {
2013-04-24 20:10:02 +02:00
inline bool matchesFilters ( std : : vector < TestCaseFilters > const & filters , TestCase const & testCase ) {
2012-08-23 21:08:50 +02:00
std : : vector < TestCaseFilters > : : const_iterator it = filters . begin ( ) ;
std : : vector < TestCaseFilters > : : const_iterator itEnd = filters . end ( ) ;
for ( ; it ! = itEnd ; + + it )
if ( ! it - > shouldInclude ( testCase ) )
return false ;
return true ;
}
2013-06-07 20:07:50 +02:00
inline std : : size_t listTests ( Config const & config ) {
if ( config . filters ( ) . empty ( ) )
2013-03-29 14:44:33 +01:00
std : : cout < < " All available test cases: \n " ;
else
std : : cout < < " Matching test cases: \n " ;
std : : vector < TestCase > const & allTests = getRegistryHub ( ) . getTestCaseRegistry ( ) . getAllTests ( ) ;
std : : vector < TestCase > : : const_iterator it = allTests . begin ( ) , itEnd = allTests . end ( ) ;
// First pass - get max tags
std : : size_t maxTagLen = 0 ;
std : : size_t maxNameLen = 0 ;
for ( ; it ! = itEnd ; + + it ) {
2013-06-07 20:07:50 +02:00
if ( matchesFilters ( config . filters ( ) , * it ) ) {
2013-03-29 14:44:33 +01:00
maxTagLen = ( std : : max ) ( it - > getTestCaseInfo ( ) . tagsAsString . size ( ) , maxTagLen ) ;
maxNameLen = ( std : : max ) ( it - > getTestCaseInfo ( ) . name . size ( ) , maxNameLen ) ;
2012-08-13 08:46:10 +02:00
}
2011-05-24 09:23:02 +02:00
}
2013-03-29 14:44:33 +01:00
// Try to fit everything in. If not shrink tag column first, down to 30
// then shrink name column until it all fits (strings will be wrapped within column)
while ( maxTagLen + maxNameLen > CATCH_CONFIG_CONSOLE_WIDTH - 5 ) {
if ( maxTagLen > 30 )
- - maxTagLen ;
2012-08-23 21:08:50 +02:00
else
2013-03-29 14:44:33 +01:00
- - maxNameLen ;
}
std : : size_t matchedTests = 0 ;
for ( it = allTests . begin ( ) ; it ! = itEnd ; + + it ) {
2013-06-07 20:07:50 +02:00
if ( matchesFilters ( config . filters ( ) , * it ) ) {
2013-03-29 14:44:33 +01:00
matchedTests + + ;
2013-04-20 22:07:32 +02:00
Text nameWrapper ( it - > getTestCaseInfo ( ) . name ,
2013-06-07 20:07:50 +02:00
TextAttributes ( )
. setWidth ( maxNameLen )
. setInitialIndent ( 2 )
. setIndent ( 4 ) ) ;
2013-03-29 14:44:33 +01:00
2013-04-20 22:07:32 +02:00
Text tagsWrapper ( it - > getTestCaseInfo ( ) . tagsAsString ,
2013-06-07 20:07:50 +02:00
TextAttributes ( )
. setWidth ( maxTagLen )
. setInitialIndent ( 0 )
. setIndent ( 2 ) ) ;
2013-03-29 14:44:33 +01:00
2013-06-29 13:11:32 +02:00
for ( std : : size_t i = 0 ; i < ( std : : max ) ( nameWrapper . size ( ) , tagsWrapper . size ( ) ) ; + + i ) {
2013-04-05 21:58:01 +02:00
Colour : : Code colour = Colour : : None ;
2013-04-01 12:27:10 +02:00
if ( it - > getTestCaseInfo ( ) . isHidden )
2013-04-05 21:58:01 +02:00
colour = Colour : : SecondaryText ;
2013-03-29 14:44:33 +01:00
std : : string nameCol ;
2013-04-01 12:27:10 +02:00
if ( i < nameWrapper . size ( ) ) {
2013-03-29 14:44:33 +01:00
nameCol = nameWrapper [ i ] ;
2013-04-01 12:27:10 +02:00
}
else {
2013-03-29 14:44:33 +01:00
nameCol = " ... " ;
2013-04-05 21:58:01 +02:00
colour = Colour : : SecondaryText ;
2013-04-01 12:27:10 +02:00
}
{
2013-04-05 21:58:01 +02:00
Colour colourGuard ( colour ) ;
2013-04-01 12:27:10 +02:00
std : : cout < < nameCol ;
}
2013-03-29 14:44:33 +01:00
if ( i < tagsWrapper . size ( ) & & ! tagsWrapper [ i ] . empty ( ) ) {
2013-04-01 12:27:10 +02:00
if ( i = = 0 ) {
2013-04-05 21:58:01 +02:00
Colour colourGuard ( Colour : : SecondaryText ) ;
2013-04-01 12:27:10 +02:00
std : : cout < < " " < < std : : string ( maxNameLen - nameCol . size ( ) , ' . ' ) < < " " ;
}
else {
std : : cout < < std : : string ( maxNameLen - nameCol . size ( ) , ' ' ) < < " " ;
}
std : : cout < < tagsWrapper [ i ] ;
2013-03-29 14:44:33 +01:00
}
std : : cout < < " \n " ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
}
}
2013-06-07 20:07:50 +02:00
if ( config . filters ( ) . empty ( ) )
std : : cout < < pluralise ( matchedTests , " test case " ) < < " \n " < < std : : endl ;
2013-03-29 14:44:33 +01:00
else
2013-06-07 20:07:50 +02:00
std : : cout < < pluralise ( matchedTests , " matching test case " ) < < " \n " < < std : : endl ;
return matchedTests ;
2013-03-29 14:44:33 +01:00
}
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
inline std : : size_t listTags ( Config const & config ) {
if ( config . filters ( ) . empty ( ) )
2013-03-29 14:44:33 +01:00
std : : cout < < " All available tags: \n " ;
else
std : : cout < < " Matching tags: \n " ;
std : : vector < TestCase > const & allTests = getRegistryHub ( ) . getTestCaseRegistry ( ) . getAllTests ( ) ;
std : : vector < TestCase > : : const_iterator it = allTests . begin ( ) , itEnd = allTests . end ( ) ;
std : : map < std : : string , int > tagCounts ;
std : : size_t maxTagLen = 0 ;
for ( ; it ! = itEnd ; + + it ) {
2013-06-07 20:07:50 +02:00
if ( matchesFilters ( config . filters ( ) , * it ) ) {
2013-03-29 14:44:33 +01:00
for ( std : : set < std : : string > : : const_iterator tagIt = it - > getTestCaseInfo ( ) . tags . begin ( ) ,
tagItEnd = it - > getTestCaseInfo ( ) . tags . end ( ) ;
tagIt ! = tagItEnd ;
+ + tagIt ) {
2013-04-05 21:58:01 +02:00
std : : string tagName = * tagIt ;
2013-03-29 14:44:33 +01:00
maxTagLen = ( std : : max ) ( maxTagLen , tagName . size ( ) ) ;
std : : map < std : : string , int > : : iterator countIt = tagCounts . find ( tagName ) ;
if ( countIt = = tagCounts . end ( ) )
tagCounts . insert ( std : : make_pair ( tagName , 1 ) ) ;
else
countIt - > second + + ;
}
}
2012-08-23 21:08:50 +02:00
}
2013-04-05 21:58:01 +02:00
maxTagLen + = 4 ;
2013-03-29 14:44:33 +01:00
if ( maxTagLen > CATCH_CONFIG_CONSOLE_WIDTH - 10 )
maxTagLen = CATCH_CONFIG_CONSOLE_WIDTH - 10 ;
for ( std : : map < std : : string , int > : : const_iterator countIt = tagCounts . begin ( ) , countItEnd = tagCounts . end ( ) ;
countIt ! = countItEnd ;
+ + countIt ) {
2013-04-20 22:07:32 +02:00
Text wrapper ( " [ " + countIt - > first + " ] " , TextAttributes ( )
. setIndent ( 2 )
. setWidth ( maxTagLen ) ) ;
2013-03-29 14:44:33 +01:00
std : : cout < < wrapper ;
2013-04-01 12:27:10 +02:00
std : : size_t dots = 2 ;
2013-03-29 14:44:33 +01:00
if ( maxTagLen > wrapper . last ( ) . size ( ) )
2013-04-01 12:27:10 +02:00
dots + = maxTagLen - wrapper . last ( ) . size ( ) ;
{
2013-04-05 21:58:01 +02:00
Colour colourGuard ( Colour : : SecondaryText ) ;
2013-04-01 12:27:10 +02:00
std : : cout < < std : : string ( dots , ' . ' ) ;
}
std : : cout < < countIt - > second
2013-03-29 14:44:33 +01:00
< < " \n " ;
}
2013-06-07 20:07:50 +02:00
std : : cout < < pluralise ( tagCounts . size ( ) , " tag " ) < < " \n " < < std : : endl ;
return tagCounts . size ( ) ;
2013-03-29 14:44:33 +01:00
}
2013-06-07 20:07:50 +02:00
inline std : : size_t listReporters ( Config const & /*config*/ ) {
2013-03-29 14:44:33 +01:00
std : : cout < < " Available reports: \n " ;
IReporterRegistry : : FactoryMap const & factories = getRegistryHub ( ) . getReporterRegistry ( ) . getFactories ( ) ;
2013-06-07 20:07:50 +02:00
IReporterRegistry : : FactoryMap : : const_iterator itBegin = factories . begin ( ) , itEnd = factories . end ( ) , it ;
std : : size_t maxNameLen = 0 ;
for ( it = itBegin ; it ! = itEnd ; + + it )
maxNameLen = ( std : : max ) ( maxNameLen , it - > first . size ( ) ) ;
for ( it = itBegin ; it ! = itEnd ; + + it ) {
Text wrapper ( it - > second - > getDescription ( ) , TextAttributes ( )
. setInitialIndent ( 0 )
. setIndent ( 7 + maxNameLen )
. setWidth ( CATCH_CONFIG_CONSOLE_WIDTH - maxNameLen - 8 ) ) ;
std : : cout < < " "
< < it - > first
< < " : "
< < std : : string ( maxNameLen - it - > first . size ( ) + 2 , ' ' )
< < wrapper < < " \n " ;
2013-03-29 14:44:33 +01:00
}
std : : cout < < std : : endl ;
2013-06-07 20:07:50 +02:00
return factories . size ( ) ;
2013-03-29 14:44:33 +01:00
}
2013-06-07 20:07:50 +02:00
inline Option < std : : size_t > list ( Config const & config ) {
Option < std : : size_t > listedCount ;
if ( config . listTests ( ) )
listedCount = listedCount . valueOr ( 0 ) + listTests ( config ) ;
if ( config . listTags ( ) )
listedCount = listedCount . valueOr ( 0 ) + listTags ( config ) ;
if ( config . listReporters ( ) )
listedCount = listedCount . valueOr ( 0 ) + listReporters ( config ) ;
return listedCount ;
2012-08-23 21:08:50 +02:00
}
2012-05-16 16:09:17 +02:00
2012-08-13 08:46:10 +02:00
} // end namespace Catch
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
// #included from: internal/catch_runner_impl.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_RUNNER_IMPL_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_running_test.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_RUNNING_TEST_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_section_info.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_SECTION_INFO_HPP_INCLUDED
2012-08-23 21:08:50 +02:00
# include <map>
2012-08-13 08:46:10 +02:00
# include <string>
2011-05-24 09:23:02 +02:00
2012-08-13 08:46:10 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
class RunningSection {
2012-08-13 08:46:10 +02:00
public :
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
typedef std : : vector < RunningSection * > SubSections ;
enum State {
2012-08-23 21:08:50 +02:00
Root ,
Unknown ,
Branch ,
TestedBranch ,
TestedLeaf
2012-08-13 08:46:10 +02:00
} ;
2013-04-24 20:10:02 +02:00
RunningSection ( RunningSection * parent , std : : string const & name )
2012-12-02 00:57:18 +01:00
: m_state ( Unknown ) ,
m_parent ( parent ) ,
m_name ( name )
2012-08-13 08:46:10 +02:00
{ }
2013-04-24 20:10:02 +02:00
RunningSection ( std : : string const & name )
2012-12-02 00:57:18 +01:00
: m_state ( Root ) ,
m_parent ( NULL ) ,
m_name ( name )
2012-08-13 08:46:10 +02:00
{ }
2012-12-02 00:57:18 +01:00
~ RunningSection ( ) {
deleteAll ( m_subSections ) ;
2011-05-24 09:23:02 +02:00
}
2012-12-02 00:57:18 +01:00
std : : string getName ( ) const {
return m_name ;
2011-05-24 09:23:02 +02:00
}
2012-12-02 00:57:18 +01:00
bool shouldRun ( ) const {
return m_state < TestedBranch ;
2011-05-24 09:23:02 +02:00
}
2012-08-31 09:10:36 +02:00
bool isBranch ( ) const {
2012-12-02 00:57:18 +01:00
return m_state = = Branch ;
2012-08-31 09:10:36 +02:00
}
2012-12-02 00:57:18 +01:00
const RunningSection * getParent ( ) const {
return m_parent ;
2012-08-13 08:46:10 +02:00
}
2012-05-05 20:35:35 +02:00
2012-12-02 00:57:18 +01:00
bool hasUntestedSections ( ) const {
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 ;
2012-08-13 08:46:10 +02:00
}
2012-05-05 20:35:35 +02:00
2012-12-02 00:57:18 +01:00
// Mutable methods:
2012-05-16 16:09:17 +02:00
2012-12-02 00:57:18 +01:00
RunningSection * getParent ( ) {
2012-08-23 21:08:50 +02:00
return m_parent ;
2012-08-13 08:46:10 +02:00
}
2012-05-16 16:09:17 +02:00
2013-04-24 20:10:02 +02:00
RunningSection * findOrAddSubSection ( std : : string const & name , bool & changed ) {
2012-12-02 00:57:18 +01:00
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 ;
}
2012-05-05 20:35:35 +02:00
2012-12-02 00:57:18 +01:00
bool ran ( ) {
if ( m_state > = Branch )
return false ;
m_state = TestedLeaf ;
return true ;
}
2012-05-05 20:35:35 +02:00
2012-12-02 00:57:18 +01:00
void ranToCompletion ( ) {
if ( m_state = = Branch & & ! hasUntestedSections ( ) )
m_state = TestedBranch ;
2012-05-05 20:35:35 +02:00
}
2012-08-23 21:08:50 +02:00
private :
2012-12-02 00:57:18 +01:00
State m_state ;
RunningSection * m_parent ;
std : : string m_name ;
SubSections m_subSections ;
2012-08-23 21:08:50 +02:00
} ;
}
namespace Catch {
class RunningTest {
enum RunStatus {
NothingRun ,
EncounteredASection ,
RanAtLeastOneSection ,
RanToCompletionWithSections ,
RanToCompletionWithNoSections
} ;
public :
2013-04-24 20:10:02 +02:00
explicit RunningTest ( TestCase const & info )
2012-08-23 21:08:50 +02:00
: m_info ( info ) ,
m_runStatus ( RanAtLeastOneSection ) ,
2012-12-02 00:57:18 +01:00
m_rootSection ( info . getTestCaseInfo ( ) . name ) ,
2012-08-23 21:08:50 +02:00
m_currentSection ( & m_rootSection ) ,
m_changed ( false )
{ }
bool wasSectionSeen ( ) const {
return m_runStatus = = RanAtLeastOneSection | |
m_runStatus = = RanToCompletionWithSections ;
2012-08-13 08:46:10 +02:00
}
2012-05-16 16:09:17 +02:00
2012-08-31 09:10:36 +02:00
bool isBranchSection ( ) const {
return m_currentSection & &
m_currentSection - > isBranch ( ) ;
}
bool hasSections ( ) const {
return m_runStatus = = RanAtLeastOneSection | |
m_runStatus = = RanToCompletionWithSections | |
m_runStatus = = EncounteredASection ;
}
2012-08-23 21:08:50 +02:00
void reset ( ) {
m_runStatus = NothingRun ;
m_changed = false ;
m_lastSectionToRun = NULL ;
2012-08-13 08:46:10 +02:00
}
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
void ranToCompletion ( ) {
2012-12-02 00:57:18 +01:00
if ( m_runStatus ! = RanAtLeastOneSection & & m_runStatus ! = EncounteredASection )
2012-08-23 21:08:50 +02:00
m_runStatus = RanToCompletionWithNoSections ;
2012-12-02 00:57:18 +01:00
m_runStatus = RanToCompletionWithSections ;
if ( m_lastSectionToRun ) {
m_lastSectionToRun - > ranToCompletion ( ) ;
m_changed = true ;
2012-08-23 21:08:50 +02:00
}
2012-08-13 08:46:10 +02:00
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
bool addSection ( std : : string const & name ) {
2012-08-23 21:08:50 +02:00
if ( m_runStatus = = NothingRun )
m_runStatus = EncounteredASection ;
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
RunningSection * thisSection = m_currentSection - > findOrAddSubSection ( name , m_changed ) ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
if ( ! wasSectionSeen ( ) & & thisSection - > shouldRun ( ) ) {
m_currentSection = thisSection ;
m_lastSectionToRun = NULL ;
return true ;
2012-08-13 08:46:10 +02:00
}
2012-08-23 21:08:50 +02:00
return false ;
2011-05-24 09:23:02 +02:00
}
2013-04-24 20:10:02 +02:00
void endSection ( std : : string const & , bool stealth ) {
2012-08-23 21:08:50 +02:00
if ( m_currentSection - > ran ( ) ) {
2013-03-04 12:19:15 +01:00
if ( ! stealth )
m_runStatus = RanAtLeastOneSection ;
2012-08-23 21:08:50 +02:00
m_changed = true ;
}
else if ( m_runStatus = = EncounteredASection ) {
2013-03-04 12:19:15 +01:00
if ( ! stealth )
m_runStatus = RanAtLeastOneSection ;
2012-08-23 21:08:50 +02:00
m_lastSectionToRun = m_currentSection ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
m_currentSection = m_currentSection - > getParent ( ) ;
2011-05-24 09:23:02 +02:00
}
2013-04-24 20:10:02 +02:00
TestCase const & getTestCase ( ) const {
2012-12-02 00:57:18 +01:00
return m_info ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
bool hasUntestedSections ( ) const {
return m_runStatus = = RanAtLeastOneSection | |
( m_rootSection . hasUntestedSections ( ) & & m_changed ) ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
private :
2013-03-25 10:25:31 +01:00
RunningTest ( RunningTest const & ) ;
void operator = ( RunningTest const & ) ;
2013-04-24 20:10:02 +02:00
TestCase const & m_info ;
2012-08-23 21:08:50 +02:00
RunStatus m_runStatus ;
2012-12-02 00:57:18 +01:00
RunningSection m_rootSection ;
RunningSection * m_currentSection ;
RunningSection * m_lastSectionToRun ;
2012-08-23 21:08:50 +02:00
bool m_changed ;
2012-08-13 08:46:10 +02:00
} ;
}
2012-08-23 21:08:50 +02:00
# include <set>
# include <string>
2012-08-13 08:46:10 +02:00
namespace Catch {
2012-08-23 21:08:50 +02:00
class StreamRedirect {
2012-08-13 08:46:10 +02:00
public :
2012-08-23 21:08:50 +02:00
StreamRedirect ( std : : ostream & stream , std : : string & targetString )
: m_stream ( stream ) ,
m_prevBuf ( stream . rdbuf ( ) ) ,
m_targetString ( targetString )
{
stream . rdbuf ( m_oss . rdbuf ( ) ) ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
~ StreamRedirect ( ) {
m_targetString + = m_oss . str ( ) ;
m_stream . rdbuf ( m_prevBuf ) ;
2012-08-13 08:46:10 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
private :
std : : ostream & m_stream ;
std : : streambuf * m_prevBuf ;
std : : ostringstream m_oss ;
std : : string & m_targetString ;
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
///////////////////////////////////////////////////////////////////////////
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
class RunContext : public IResultCapture , public IRunner {
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
RunContext ( RunContext const & ) ;
void operator = ( RunContext const & ) ;
2012-06-05 21:51:05 +02:00
2012-08-23 21:08:50 +02:00
public :
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
explicit RunContext ( Ptr < IConfig const > const & config , Ptr < IStreamingReporter > const & reporter )
: m_runInfo ( config - > name ( ) ) ,
2012-12-02 00:57:18 +01:00
m_context ( getCurrentMutableContext ( ) ) ,
2012-08-23 21:08:50 +02:00
m_runningTest ( NULL ) ,
m_config ( config ) ,
m_reporter ( reporter ) ,
m_prevRunner ( & m_context . getRunner ( ) ) ,
m_prevResultCapture ( & m_context . getResultCapture ( ) ) ,
m_prevConfig ( m_context . getConfig ( ) )
{
m_context . setRunner ( this ) ;
2013-06-07 20:07:50 +02:00
m_context . setConfig ( m_config ) ;
2012-08-23 21:08:50 +02:00
m_context . setResultCapture ( this ) ;
2012-12-02 00:57:18 +01:00
m_reporter - > testRunStarting ( m_runInfo ) ;
2011-05-24 09:23:02 +02:00
}
2013-06-07 20:07:50 +02:00
virtual ~ RunContext ( ) {
2013-01-16 10:44:43 +01:00
m_reporter - > testRunEnded ( TestRunStats ( m_runInfo , m_totals , aborting ( ) ) ) ;
2012-08-23 21:08:50 +02:00
m_context . setRunner ( m_prevRunner ) ;
m_context . setConfig ( NULL ) ;
m_context . setResultCapture ( m_prevResultCapture ) ;
m_context . setConfig ( m_prevConfig ) ;
2011-05-24 09:23:02 +02:00
}
2013-01-16 10:44:43 +01:00
void testGroupStarting ( std : : string const & testSpec , std : : size_t groupIndex , std : : size_t groupsCount ) {
m_reporter - > testGroupStarting ( GroupInfo ( testSpec , groupIndex , groupsCount ) ) ;
2012-12-02 00:57:18 +01:00
}
2013-01-16 10:44:43 +01:00
void testGroupEnded ( std : : string const & testSpec , Totals const & totals , std : : size_t groupIndex , std : : size_t groupsCount ) {
m_reporter - > testGroupEnded ( TestGroupStats ( GroupInfo ( testSpec , groupIndex , groupsCount ) , totals , aborting ( ) ) ) ;
2012-12-02 00:57:18 +01:00
}
2013-04-24 20:10:02 +02:00
Totals runMatching ( std : : string const & testSpec , std : : size_t groupIndex , std : : size_t groupsCount ) {
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
std : : vector < TestCase > matchingTests = getRegistryHub ( ) . getTestCaseRegistry ( ) . getMatchingTestCases ( testSpec ) ;
2012-08-23 21:08:50 +02:00
Totals totals ;
2013-01-16 10:44:43 +01:00
testGroupStarting ( testSpec , groupIndex , groupsCount ) ;
2012-08-23 21:08:50 +02:00
2012-12-02 00:57:18 +01:00
std : : vector < TestCase > : : const_iterator it = matchingTests . begin ( ) ;
std : : vector < TestCase > : : const_iterator itEnd = matchingTests . end ( ) ;
2012-08-23 21:08:50 +02:00
for ( ; it ! = itEnd ; + + it )
totals + = runTest ( * it ) ;
2013-01-16 10:44:43 +01:00
testGroupEnded ( testSpec , totals , groupIndex , groupsCount ) ;
2012-08-23 21:08:50 +02:00
return totals ;
2011-05-24 09:23:02 +02:00
}
2013-04-24 20:10:02 +02:00
Totals runTest ( TestCase const & testCase ) {
2012-08-23 21:08:50 +02:00
Totals prevTotals = m_totals ;
2012-08-13 08:46:10 +02:00
2012-08-23 21:08:50 +02:00
std : : string redirectedCout ;
std : : string redirectedCerr ;
2012-08-13 08:46:10 +02:00
2012-12-02 00:57:18 +01:00
TestCaseInfo testInfo = testCase . getTestCaseInfo ( ) ;
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
m_reporter - > testCaseStarting ( testInfo ) ;
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
m_runningTest = new RunningTest ( testCase ) ;
2012-08-23 21:08:50 +02:00
do {
do {
runCurrentTest ( redirectedCout , redirectedCerr ) ;
}
while ( m_runningTest - > hasUntestedSections ( ) & & ! aborting ( ) ) ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
while ( getCurrentContext ( ) . advanceGeneratorsForCurrentTest ( ) & & ! aborting ( ) ) ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
Totals deltaTotals = m_totals . delta ( prevTotals ) ;
2012-12-02 00:57:18 +01:00
bool missingAssertions = false ;
2013-06-07 20:07:50 +02:00
if ( deltaTotals . assertions . total ( ) = = 0 & & m_config - > warnAboutMissingAssertions ( ) ) {
2012-11-21 19:06:13 +01:00
m_totals . assertions . failed + + ;
deltaTotals = m_totals . delta ( prevTotals ) ;
2012-12-02 00:57:18 +01:00
missingAssertions = true ;
2012-11-21 19:06:13 +01:00
}
2012-12-02 00:57:18 +01:00
2012-08-23 21:08:50 +02:00
m_totals . testCases + = deltaTotals . testCases ;
2012-11-21 19:06:13 +01:00
2013-01-16 10:44:43 +01:00
m_reporter - > testCaseEnded ( TestCaseStats ( testInfo ,
deltaTotals ,
redirectedCout ,
redirectedCerr ,
missingAssertions ,
aborting ( ) ) ) ;
2012-11-21 19:06:13 +01:00
2012-08-23 21:08:50 +02:00
delete m_runningTest ;
m_runningTest = NULL ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
return deltaTotals ;
2011-05-24 09:23:02 +02:00
}
2013-06-07 20:07:50 +02:00
Ptr < IConfig const > config ( ) const {
2012-08-23 21:08:50 +02:00
return m_config ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
private : // IResultCapture
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
virtual ResultAction : : Value acceptExpression ( ExpressionResultBuilder const & assertionResult , AssertionInfo const & assertionInfo ) {
2012-11-13 23:04:29 +01:00
m_lastAssertionInfo = assertionInfo ;
return actOnCurrentResult ( assertionResult . buildResult ( assertionInfo ) ) ;
2012-08-23 21:08:50 +02:00
}
2012-02-10 09:30:13 +01:00
2013-04-24 20:10:02 +02:00
virtual void assertionEnded ( AssertionResult const & result ) {
2012-08-23 21:08:50 +02:00
if ( result . getResultType ( ) = = ResultWas : : Ok ) {
m_totals . assertions . passed + + ;
}
2012-11-13 23:04:29 +01:00
else if ( ! result . isOk ( ) ) {
2012-08-23 21:08:50 +02:00
m_totals . assertions . failed + + ;
}
2011-05-24 09:23:02 +02:00
2013-06-28 18:09:57 +02:00
if ( m_reporter - > assertionEnded ( AssertionStats ( result , m_messages , m_totals ) ) )
m_messages . clear ( ) ;
2012-11-19 20:59:10 +01:00
2013-02-02 21:37:58 +01:00
// Reset working state
m_lastAssertionInfo = AssertionInfo ( " " , m_lastAssertionInfo . lineInfo , " {Unknown expression after the reported line} " , m_lastAssertionInfo . resultDisposition ) ;
2012-08-23 21:08:50 +02:00
}
2012-06-01 20:40:27 +02:00
2012-08-23 21:08:50 +02:00
virtual bool sectionStarted (
2012-12-02 00:57:18 +01:00
SectionInfo const & sectionInfo ,
2012-08-23 21:08:50 +02:00
Counts & assertions
)
{
std : : ostringstream oss ;
2012-12-02 00:57:18 +01:00
oss < < sectionInfo . name < < " @ " < < sectionInfo . lineInfo ;
2012-06-01 20:40:27 +02:00
2012-08-23 21:08:50 +02:00
if ( ! m_runningTest - > addSection ( oss . str ( ) ) )
return false ;
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
m_lastAssertionInfo . lineInfo = sectionInfo . lineInfo ;
m_reporter - > sectionStarting ( sectionInfo ) ;
2012-11-13 23:04:29 +01:00
2012-08-23 21:08:50 +02:00
assertions = m_totals . assertions ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
return true ;
}
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
virtual void sectionEnded ( SectionInfo const & info , Counts const & prevAssertions ) {
2013-02-19 20:47:18 +01:00
if ( std : : uncaught_exception ( ) ) {
m_unfinishedSections . push_back ( UnfinishedSections ( info , prevAssertions ) ) ;
return ;
}
2012-08-31 09:10:36 +02:00
Counts assertions = m_totals . assertions - prevAssertions ;
2012-12-02 00:57:18 +01:00
bool missingAssertions = false ;
2012-11-21 19:06:13 +01:00
if ( assertions . total ( ) = = 0 & &
2013-06-07 20:07:50 +02:00
m_config - > warnAboutMissingAssertions ( ) & &
! m_runningTest - > isBranchSection ( ) ) {
2012-08-31 09:10:36 +02:00
m_totals . assertions . failed + + ;
assertions . failed + + ;
2012-12-02 00:57:18 +01:00
missingAssertions = true ;
2012-08-31 09:10:36 +02:00
}
2013-03-04 12:19:15 +01:00
m_runningTest - > endSection ( info . name , false ) ;
2012-12-02 00:57:18 +01:00
2013-01-16 10:44:43 +01:00
m_reporter - > sectionEnded ( SectionStats ( info , assertions , missingAssertions ) ) ;
2013-02-02 21:37:58 +01:00
m_messages . clear ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-05-21 19:52:09 +02:00
2013-06-28 18:09:57 +02:00
virtual void pushScopedMessage ( MessageInfo const & message ) {
m_messages . push_back ( message ) ;
2012-08-13 08:46:10 +02:00
}
2011-05-24 09:23:02 +02:00
2013-06-28 18:09:57 +02:00
virtual void popScopedMessage ( MessageInfo const & message ) {
m_messages . erase ( std : : remove ( m_messages . begin ( ) , m_messages . end ( ) , message ) , m_messages . end ( ) ) ;
2012-08-23 21:08:50 +02:00
}
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
virtual bool shouldDebugBreak ( ) const {
2013-06-07 20:07:50 +02:00
return m_config - > shouldDebugBreak ( ) ;
2012-08-13 08:46:10 +02:00
}
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
virtual std : : string getCurrentTestName ( ) const {
return m_runningTest
2012-12-02 00:57:18 +01:00
? m_runningTest - > getTestCase ( ) . getTestCaseInfo ( ) . name
2012-08-23 21:08:50 +02:00
: " " ;
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
virtual const AssertionResult * getLastResult ( ) const {
2012-08-23 21:08:50 +02:00
return & m_lastResult ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
public :
// !TBD We need to do this another way!
bool aborting ( ) const {
2013-06-07 20:07:50 +02:00
return m_totals . assertions . failed = = static_cast < std : : size_t > ( m_config - > abortAfter ( ) ) ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
private :
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
ResultAction : : Value actOnCurrentResult ( AssertionResult const & result ) {
2012-11-13 23:04:29 +01:00
m_lastResult = result ;
2013-02-02 21:37:58 +01:00
assertionEnded ( m_lastResult ) ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
ResultAction : : Value action = ResultAction : : None ;
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
if ( ! m_lastResult . isOk ( ) ) {
2012-08-23 21:08:50 +02:00
action = ResultAction : : Failed ;
if ( shouldDebugBreak ( ) )
action = ( ResultAction : : Value ) ( action | ResultAction : : Debug ) ;
if ( aborting ( ) )
action = ( ResultAction : : Value ) ( action | ResultAction : : Abort ) ;
2012-08-13 08:46:10 +02:00
}
2012-08-23 21:08:50 +02:00
return action ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
void runCurrentTest ( std : : string & redirectedCout , std : : string & redirectedCerr ) {
try {
2012-12-02 00:57:18 +01:00
m_lastAssertionInfo = AssertionInfo ( " TEST_CASE " , m_runningTest - > getTestCase ( ) . getTestCaseInfo ( ) . lineInfo , " " , ResultDisposition : : Normal ) ;
2012-08-23 21:08:50 +02:00
m_runningTest - > reset ( ) ;
2013-02-04 01:05:16 +01:00
2012-12-02 00:57:18 +01:00
if ( m_reporter - > getPreferences ( ) . shouldRedirectStdOut ) {
2012-08-23 21:08:50 +02:00
StreamRedirect coutRedir ( std : : cout , redirectedCout ) ;
StreamRedirect cerrRedir ( std : : cerr , redirectedCerr ) ;
2012-12-02 00:57:18 +01:00
m_runningTest - > getTestCase ( ) . invoke ( ) ;
2012-08-23 21:08:50 +02:00
}
else {
2012-12-02 00:57:18 +01:00
m_runningTest - > getTestCase ( ) . invoke ( ) ;
2012-08-31 09:10:36 +02:00
}
2012-08-23 21:08:50 +02:00
m_runningTest - > ranToCompletion ( ) ;
}
catch ( TestFailureException & ) {
// This just means the test was aborted due to failure
}
catch ( . . . ) {
2012-11-13 23:04:29 +01:00
ExpressionResultBuilder exResult ( ResultWas : : ThrewException ) ;
exResult < < translateActiveException ( ) ;
actOnCurrentResult ( exResult . buildResult ( m_lastAssertionInfo ) ) ;
2012-08-23 21:08:50 +02:00
}
2013-02-19 20:47:18 +01:00
for ( std : : vector < UnfinishedSections > : : const_iterator it = m_unfinishedSections . begin ( ) ,
itEnd = m_unfinishedSections . end ( ) ;
it ! = itEnd ;
+ + it )
sectionEnded ( it - > info , it - > prevAssertions ) ;
m_unfinishedSections . clear ( ) ;
2013-02-04 01:05:16 +01:00
m_messages . clear ( ) ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
private :
2013-02-19 20:47:18 +01:00
struct UnfinishedSections {
UnfinishedSections ( SectionInfo const & _info , Counts const & _prevAssertions )
: info ( _info ) , prevAssertions ( _prevAssertions )
{ }
SectionInfo info ;
Counts prevAssertions ;
} ;
2012-12-02 00:57:18 +01:00
TestRunInfo m_runInfo ;
2012-08-23 21:08:50 +02:00
IMutableContext & m_context ;
RunningTest * m_runningTest ;
2012-11-01 09:27:09 +01:00
AssertionResult m_lastResult ;
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
Ptr < IConfig const > m_config ;
2012-08-23 21:08:50 +02:00
Totals m_totals ;
2012-12-02 00:57:18 +01:00
Ptr < IStreamingReporter > m_reporter ;
2013-02-02 21:37:58 +01:00
std : : vector < MessageInfo > m_messages ;
2012-08-23 21:08:50 +02:00
IRunner * m_prevRunner ;
IResultCapture * m_prevResultCapture ;
2013-06-07 20:07:50 +02:00
Ptr < IConfig const > m_prevConfig ;
2012-11-13 23:04:29 +01:00
AssertionInfo m_lastAssertionInfo ;
2013-02-19 20:47:18 +01:00
std : : vector < UnfinishedSections > m_unfinishedSections ;
2012-08-23 21:08:50 +02:00
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2011-05-24 09:23:02 +02:00
2012-11-15 23:15:41 +01:00
// #included from: internal/catch_version.h
# define TWOBLUECUBES_CATCH_VERSION_H_INCLUDED
namespace Catch {
// Versioning information
struct Version {
2012-12-02 00:57:18 +01:00
Version ( unsigned int _majorVersion ,
unsigned int _minorVersion ,
unsigned int _buildNumber ,
std : : string const & _branchName )
: majorVersion ( _majorVersion ) ,
minorVersion ( _minorVersion ) ,
buildNumber ( _buildNumber ) ,
branchName ( _branchName )
{ }
const unsigned int majorVersion ;
const unsigned int minorVersion ;
const unsigned int buildNumber ;
const std : : string branchName ;
2013-03-25 10:25:31 +01:00
private :
void operator = ( Version const & ) ;
2012-11-15 23:15:41 +01:00
} ;
extern Version libraryVersion ;
}
2012-08-23 21:08:50 +02:00
# include <fstream>
# include <stdlib.h>
# include <limits>
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
class Runner {
2012-02-28 21:04:42 +01:00
2012-08-23 21:08:50 +02:00
public :
2013-06-07 20:07:50 +02:00
Runner ( Ptr < Config > const & config )
: m_config ( config )
2012-08-23 21:08:50 +02:00
{
2012-09-26 19:38:26 +02:00
openStream ( ) ;
2012-08-23 21:08:50 +02:00
makeReporter ( ) ;
2012-08-09 08:47:30 +02:00
}
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
Totals runTests ( ) {
2013-06-07 20:07:50 +02:00
std : : vector < TestCaseFilters > filterGroups = m_config - > filters ( ) ;
2012-08-23 21:08:50 +02:00
if ( filterGroups . empty ( ) ) {
TestCaseFilters filterGroup ( " " ) ;
filterGroups . push_back ( filterGroup ) ;
}
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
RunContext context ( m_config . get ( ) , m_reporter ) ;
2012-08-23 21:08:50 +02:00
Totals totals ;
2012-02-28 21:04:42 +01:00
2013-01-16 10:44:43 +01:00
for ( std : : size_t i = 0 ; i < filterGroups . size ( ) & & ! context . aborting ( ) ; + + i ) {
context . testGroupStarting ( filterGroups [ i ] . getName ( ) , i , filterGroups . size ( ) ) ;
totals + = runTestsForGroup ( context , filterGroups [ i ] ) ;
context . testGroupEnded ( filterGroups [ i ] . getName ( ) , totals , i , filterGroups . size ( ) ) ;
2012-08-13 08:46:10 +02:00
}
2012-08-23 21:08:50 +02:00
return totals ;
2012-08-09 08:47:30 +02:00
}
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
Totals runTestsForGroup ( RunContext & context , const TestCaseFilters & filterGroup ) {
2012-08-23 21:08:50 +02:00
Totals totals ;
2012-12-02 00:57:18 +01:00
std : : vector < TestCase > : : const_iterator it = getRegistryHub ( ) . getTestCaseRegistry ( ) . getAllTests ( ) . begin ( ) ;
std : : vector < TestCase > : : const_iterator itEnd = getRegistryHub ( ) . getTestCaseRegistry ( ) . getAllTests ( ) . end ( ) ;
2012-08-23 21:08:50 +02:00
int testsRunForGroup = 0 ;
2012-08-13 08:46:10 +02:00
for ( ; it ! = itEnd ; + + it ) {
2012-08-23 21:08:50 +02:00
if ( filterGroup . shouldInclude ( * it ) ) {
testsRunForGroup + + ;
if ( m_testsAlreadyRun . find ( * it ) = = m_testsAlreadyRun . end ( ) ) {
2012-02-28 21:04:42 +01:00
2012-08-23 21:08:50 +02:00
if ( context . aborting ( ) )
break ;
totals + = context . runTest ( * it ) ;
m_testsAlreadyRun . insert ( * it ) ;
}
2012-08-13 08:46:10 +02:00
}
2012-08-09 08:47:30 +02:00
}
2013-02-19 20:47:18 +01:00
if ( testsRunForGroup = = 0 & & ! filterGroup . getName ( ) . empty ( ) )
2013-03-13 13:19:30 +01:00
m_reporter - > noMatchingTestCases ( filterGroup . getName ( ) ) ;
2012-08-23 21:08:50 +02:00
return totals ;
2012-08-09 08:47:30 +02:00
}
2012-02-28 21:04:42 +01:00
2012-08-13 08:46:10 +02:00
private :
2012-09-26 19:38:26 +02:00
void openStream ( ) {
2012-08-23 21:08:50 +02:00
// Open output file, if specified
2013-06-07 20:07:50 +02:00
if ( ! m_config - > getFilename ( ) . empty ( ) ) {
m_ofs . open ( m_config - > getFilename ( ) . c_str ( ) ) ;
2012-08-23 21:08:50 +02:00
if ( m_ofs . fail ( ) ) {
std : : ostringstream oss ;
2013-06-07 20:07:50 +02:00
oss < < " Unable to open file: ' " < < m_config - > getFilename ( ) < < " ' " ;
2012-08-23 21:08:50 +02:00
throw std : : domain_error ( oss . str ( ) ) ;
}
2013-06-07 20:07:50 +02:00
m_config - > setStreamBuf ( m_ofs . rdbuf ( ) ) ;
2012-08-23 21:08:50 +02:00
}
}
void makeReporter ( ) {
2013-06-07 20:07:50 +02:00
std : : string reporterName = m_config - > getReporterName ( ) . empty ( )
? " console "
: m_config - > getReporterName ( ) ;
2012-08-23 21:08:50 +02:00
2013-06-07 20:07:50 +02:00
m_reporter = getRegistryHub ( ) . getReporterRegistry ( ) . create ( reporterName , m_config . get ( ) ) ;
2012-08-23 21:08:50 +02:00
if ( ! m_reporter ) {
std : : ostringstream oss ;
oss < < " No reporter registered with name: ' " < < reporterName < < " ' " ;
throw std : : domain_error ( oss . str ( ) ) ;
}
}
private :
2013-06-07 20:07:50 +02:00
Ptr < Config > m_config ;
2012-08-23 21:08:50 +02:00
std : : ofstream m_ofs ;
2012-12-02 00:57:18 +01:00
Ptr < IStreamingReporter > m_reporter ;
std : : set < TestCase > m_testsAlreadyRun ;
2012-08-13 08:46:10 +02:00
} ;
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
class Session {
static bool alreadyInstantiated ;
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
public :
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
struct OnUnusedOptions { enum DoWhat { Ignore , Fail } ; } ;
2012-05-16 16:09:17 +02:00
2013-06-07 20:07:50 +02:00
Session ( )
: m_cli ( makeCommandLineParser ( ) ) {
if ( alreadyInstantiated ) {
std : : string msg = " Only one instance of Catch::Session can ever be used " ;
std : : cerr < < msg < < std : : endl ;
throw std : : logic_error ( msg ) ;
}
alreadyInstantiated = true ;
2012-08-23 21:08:50 +02:00
}
2013-06-07 20:07:50 +02:00
~ Session ( ) {
Catch : : cleanUp ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-05-16 16:09:17 +02:00
2013-06-07 20:07:50 +02:00
void showHelp ( std : : string const & processName ) {
std : : cout < < " \n Catch v " < < libraryVersion . majorVersion < < " . "
< < libraryVersion . minorVersion < < " build "
< < libraryVersion . buildNumber ;
if ( libraryVersion . branchName ! = " master " )
std : : cout < < " ( " < < libraryVersion . branchName < < " branch) " ;
std : : cout < < " \n " ;
2012-09-07 18:52:35 +02:00
2013-06-07 20:07:50 +02:00
m_cli . usage ( std : : cout , processName ) ;
std : : cout < < " For more detail usage please see the project docs \n " < < std : : endl ;
2012-08-27 22:42:55 +02:00
}
2012-09-07 18:52:35 +02:00
2013-06-07 20:07:50 +02:00
int applyCommandLine ( int argc , char * const argv [ ] , OnUnusedOptions : : DoWhat unusedOptionBehaviour = OnUnusedOptions : : Fail ) {
try {
m_unusedTokens = m_cli . parseInto ( argc , argv , m_configData ) ;
if ( unusedOptionBehaviour = = OnUnusedOptions : : Fail )
enforceNoUsedTokens ( ) ;
if ( m_configData . showHelp )
showHelp ( m_configData . processName ) ;
m_config . reset ( ) ;
2012-09-07 18:52:35 +02:00
}
2013-06-07 20:07:50 +02:00
catch ( std : : exception & ex ) {
std : : cerr < < " \n Error in input: \n "
< < Text ( ex . what ( ) , TextAttributes ( )
. setInitialIndent ( 2 )
. setIndent ( 4 ) )
< < " \n \n " ;
m_cli . usage ( std : : cout , m_configData . processName ) ;
return ( std : : numeric_limits < int > : : max ) ( ) ;
2012-09-07 18:52:35 +02:00
}
2013-06-07 20:07:50 +02:00
return 0 ;
2012-09-07 18:52:35 +02:00
}
2012-09-09 12:44:30 +02:00
2013-06-07 20:07:50 +02:00
void useConfigData ( ConfigData const & _configData ) {
m_configData = _configData ;
m_config . reset ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
void enforceNoUsedTokens ( ) const {
if ( ! m_unusedTokens . empty ( ) ) {
std : : vector < Clara : : Parser : : Token > : : const_iterator
it = m_unusedTokens . begin ( ) ,
itEnd = m_unusedTokens . end ( ) ;
std : : string msg ;
for ( ; it ! = itEnd ; + + it )
msg + = " unrecognised option: " + it - > data + " \n " ;
throw std : : runtime_error ( msg . substr ( 0 , msg . size ( ) - 1 ) ) ;
2012-09-07 18:52:35 +02:00
}
}
2013-06-07 20:07:50 +02:00
int run ( int argc , char * const argv [ ] ) {
2012-11-15 23:15:41 +01:00
2013-06-07 20:07:50 +02:00
int returnCode = applyCommandLine ( argc , argv ) ;
if ( returnCode = = 0 )
returnCode = run ( ) ;
return returnCode ;
2012-09-07 18:52:35 +02:00
}
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
int run ( ) {
if ( m_configData . showHelp )
return 0 ;
2012-02-28 21:04:42 +01:00
2013-06-07 20:07:50 +02:00
try
{
config ( ) ; // Force config to be constructed
Runner runner ( m_config ) ;
2012-05-11 09:17:16 +02:00
2013-06-07 20:07:50 +02:00
// Handle list request
if ( Option < std : : size_t > listed = list ( config ( ) ) )
return static_cast < int > ( * listed ) ;
2012-05-21 22:51:16 +02:00
2013-06-07 20:07:50 +02:00
return static_cast < int > ( runner . runTests ( ) . assertions . failed ) ;
2012-08-23 21:08:50 +02:00
}
2013-06-07 20:07:50 +02:00
catch ( std : : exception & ex ) {
std : : cerr < < ex . what ( ) < < std : : endl ;
return ( std : : numeric_limits < int > : : max ) ( ) ;
}
}
2012-05-16 16:09:17 +02:00
2013-06-07 20:07:50 +02:00
Clara : : CommandLine < ConfigData > const & cli ( ) const {
return m_cli ;
2012-08-23 21:08:50 +02:00
}
2013-06-07 20:07:50 +02:00
std : : vector < Clara : : Parser : : Token > const & unusedTokens ( ) const {
return m_unusedTokens ;
}
ConfigData & configData ( ) {
return m_configData ;
}
Config & config ( ) {
if ( ! m_config )
m_config = new Config ( m_configData ) ;
return * m_config ;
2012-08-23 21:08:50 +02:00
}
2012-05-11 09:17:16 +02:00
2013-06-07 20:07:50 +02:00
private :
Clara : : CommandLine < ConfigData > m_cli ;
std : : vector < Clara : : Parser : : Token > m_unusedTokens ;
ConfigData m_configData ;
Ptr < Config > m_config ;
} ;
2012-05-11 09:17:16 +02:00
2013-06-07 20:07:50 +02:00
bool Session : : alreadyInstantiated = false ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_registry_hub.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_REGISTRY_HUB_HPP_INCLUDED
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_test_case_registry_impl.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_TEST_CASE_REGISTRY_IMPL_HPP_INCLUDED
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
# include <vector>
# include <set>
# include <sstream>
# include <iostream>
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
class TestRegistry : public ITestCaseRegistry {
public :
TestRegistry ( ) : m_unnamedCount ( 0 ) { }
virtual ~ TestRegistry ( ) ;
2012-05-16 16:09:17 +02:00
2013-04-24 20:10:02 +02:00
virtual void registerTest ( TestCase const & testCase ) {
2012-12-02 00:57:18 +01:00
std : : string name = testCase . getTestCaseInfo ( ) . name ;
if ( name = = " " ) {
2012-08-23 21:08:50 +02:00
std : : ostringstream oss ;
2013-03-16 21:21:51 +01:00
oss < < " Anonymous test case " < < + + m_unnamedCount ;
2012-12-02 00:57:18 +01:00
return registerTest ( testCase . withName ( oss . str ( ) ) ) ;
2012-08-23 21:08:50 +02:00
}
2012-05-16 16:09:17 +02:00
2012-12-02 00:57:18 +01:00
if ( m_functions . find ( testCase ) = = m_functions . end ( ) ) {
m_functions . insert ( testCase ) ;
m_functionsInOrder . push_back ( testCase ) ;
if ( ! testCase . isHidden ( ) )
m_nonHiddenFunctions . push_back ( testCase ) ;
2012-08-23 21:08:50 +02:00
}
else {
2013-04-24 20:10:02 +02:00
TestCase const & prev = * m_functions . find ( testCase ) ;
2012-12-02 00:57:18 +01:00
std : : cerr < < " error: TEST_CASE( \" " < < name < < " \" ) already defined. \n "
< < " \t First seen at " < < SourceLineInfo ( prev . getTestCaseInfo ( ) . lineInfo ) < < " \n "
< < " \t Redefined at " < < SourceLineInfo ( testCase . getTestCaseInfo ( ) . lineInfo ) < < std : : endl ;
2012-08-23 21:08:50 +02:00
exit ( 1 ) ;
}
2012-08-13 08:46:10 +02:00
}
2012-05-11 09:17:16 +02:00
2013-04-24 20:10:02 +02:00
virtual std : : vector < TestCase > const & getAllTests ( ) const {
2012-08-23 21:08:50 +02:00
return m_functionsInOrder ;
2012-08-13 08:46:10 +02:00
}
2012-05-11 09:17:16 +02:00
2013-04-24 20:10:02 +02:00
virtual std : : vector < TestCase > const & getAllNonHiddenTests ( ) const {
2012-08-23 21:08:50 +02:00
return m_nonHiddenFunctions ;
2012-08-13 08:46:10 +02:00
}
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
// !TBD deprecated
2013-04-24 20:10:02 +02:00
virtual std : : vector < TestCase > getMatchingTestCases ( std : : string const & rawTestSpec ) const {
2012-12-02 00:57:18 +01:00
std : : vector < TestCase > matchingTests ;
2012-08-23 21:08:50 +02:00
getMatchingTestCases ( rawTestSpec , matchingTests ) ;
return matchingTests ;
2012-08-13 08:46:10 +02:00
}
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
// !TBD deprecated
2013-04-24 20:10:02 +02:00
virtual void getMatchingTestCases ( std : : string const & rawTestSpec , std : : vector < TestCase > & matchingTestsOut ) const {
2012-08-23 21:08:50 +02:00
TestCaseFilter filter ( rawTestSpec ) ;
2012-08-09 08:47:30 +02:00
2012-12-02 00:57:18 +01:00
std : : vector < TestCase > : : const_iterator it = m_functionsInOrder . begin ( ) ;
std : : vector < TestCase > : : const_iterator itEnd = m_functionsInOrder . end ( ) ;
2012-08-23 21:08:50 +02:00
for ( ; it ! = itEnd ; + + it ) {
if ( filter . shouldInclude ( * it ) ) {
matchingTestsOut . push_back ( * it ) ;
}
}
2012-08-13 08:46:10 +02:00
}
2013-04-24 20:10:02 +02:00
virtual void getMatchingTestCases ( TestCaseFilters const & filters , std : : vector < TestCase > & matchingTestsOut ) const {
2012-12-02 00:57:18 +01:00
std : : vector < TestCase > : : const_iterator it = m_functionsInOrder . begin ( ) ;
std : : vector < TestCase > : : const_iterator itEnd = m_functionsInOrder . end ( ) ;
2012-08-23 21:08:50 +02:00
// !TBD: replace with algorithm
for ( ; it ! = itEnd ; + + it )
if ( filters . shouldInclude ( * it ) )
matchingTestsOut . push_back ( * it ) ;
2012-08-13 08:46:10 +02:00
}
2012-05-11 09:17:16 +02:00
2012-08-23 21:08:50 +02:00
private :
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
std : : set < TestCase > m_functions ;
std : : vector < TestCase > m_functionsInOrder ;
std : : vector < TestCase > m_nonHiddenFunctions ;
2012-08-23 21:08:50 +02:00
size_t m_unnamedCount ;
} ;
///////////////////////////////////////////////////////////////////////////
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
class FreeFunctionTestCase : public SharedImpl < ITestCase > {
public :
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
FreeFunctionTestCase ( TestFunction fun ) : m_fun ( fun ) { }
virtual void invoke ( ) const {
m_fun ( ) ;
2011-05-24 09:23:02 +02:00
}
2012-08-13 08:46:10 +02:00
private :
2012-08-23 21:08:50 +02:00
virtual ~ FreeFunctionTestCase ( ) ;
2012-08-13 08:46:10 +02:00
2012-08-23 21:08:50 +02:00
TestFunction m_fun ;
2012-08-09 08:47:30 +02:00
} ;
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
inline std : : string extractClassName ( std : : string const & classOrQualifiedMethodName ) {
2012-11-13 23:04:29 +01:00
std : : string className = classOrQualifiedMethodName ;
if ( className [ 0 ] = = ' & ' )
{
std : : size_t lastColons = className . rfind ( " :: " ) ;
std : : size_t penultimateColons = className . rfind ( " :: " , lastColons - 1 ) ;
if ( penultimateColons = = std : : string : : npos )
penultimateColons = 1 ;
className = className . substr ( penultimateColons , lastColons - penultimateColons ) ;
}
return className ;
}
2012-08-23 21:08:50 +02:00
///////////////////////////////////////////////////////////////////////////
AutoReg : : AutoReg ( TestFunction function ,
2013-03-16 21:21:51 +01:00
SourceLineInfo const & lineInfo ,
NameAndDesc const & nameAndDesc ) {
registerTestCase ( new FreeFunctionTestCase ( function ) , " global " , nameAndDesc , lineInfo ) ;
2012-08-23 21:08:50 +02:00
}
AutoReg : : ~ AutoReg ( ) { }
void AutoReg : : registerTestCase ( ITestCase * testCase ,
2013-03-16 21:21:51 +01:00
char const * classOrQualifiedMethodName ,
NameAndDesc const & nameAndDesc ,
SourceLineInfo const & lineInfo ) {
2012-11-13 23:04:29 +01:00
2013-03-16 21:21:51 +01:00
getMutableRegistryHub ( ) . registerTest
( makeTestCase ( testCase ,
extractClassName ( classOrQualifiedMethodName ) ,
nameAndDesc . name ,
nameAndDesc . description ,
lineInfo ) ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-13 08:46:10 +02:00
} // end namespace Catch
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_reporter_registry.hpp
# define TWOBLUECUBES_CATCH_REPORTER_REGISTRY_HPP_INCLUDED
# include <map>
2012-08-13 08:46:10 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
class ReporterRegistry : public IReporterRegistry {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
public :
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual ~ ReporterRegistry ( ) {
deleteAllValues ( m_factories ) ;
2012-08-13 08:46:10 +02:00
}
2011-05-24 09:23:02 +02:00
2013-06-07 20:07:50 +02:00
virtual IStreamingReporter * create ( std : : string const & name , Ptr < IConfig > const & config ) const {
2012-08-23 21:08:50 +02:00
FactoryMap : : const_iterator it = m_factories . find ( name ) ;
if ( it = = m_factories . end ( ) )
return NULL ;
2013-06-07 20:07:50 +02:00
return it - > second - > create ( ReporterConfig ( config ) ) ;
2012-08-13 08:46:10 +02:00
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
void registerReporter ( std : : string const & name , IReporterFactory * factory ) {
2012-08-23 21:08:50 +02:00
m_factories . insert ( std : : make_pair ( name , factory ) ) ;
}
2012-05-11 09:17:16 +02:00
2013-04-24 20:10:02 +02:00
FactoryMap const & getFactories ( ) const {
2012-08-23 21:08:50 +02:00
return m_factories ;
2011-05-24 09:23:02 +02:00
}
2012-05-31 20:40:26 +02:00
2012-08-09 08:47:30 +02:00
private :
2012-08-23 21:08:50 +02:00
FactoryMap m_factories ;
2012-08-13 08:46:10 +02:00
} ;
2012-08-23 21:08:50 +02:00
}
2012-05-31 20:40:26 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_exception_translator_registry.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_EXCEPTION_TRANSLATOR_REGISTRY_HPP_INCLUDED
2012-05-31 20:40:26 +02:00
2012-08-23 21:08:50 +02:00
# ifdef __OBJC__
# import "Foundation / Foundation.h"
# endif
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
class ExceptionTranslatorRegistry : public IExceptionTranslatorRegistry {
public :
~ ExceptionTranslatorRegistry ( ) {
deleteAll ( m_translators ) ;
2012-05-31 20:40:26 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void registerTranslator ( const IExceptionTranslator * translator ) {
m_translators . push_back ( translator ) ;
}
2012-05-31 20:40:26 +02:00
2012-08-23 21:08:50 +02:00
virtual std : : string translateActiveException ( ) const {
try {
# ifdef __OBJC__
// In Objective-C try objective-c exceptions first
@ try {
throw ;
}
@ catch ( NSException * exception ) {
return toString ( [ exception description ] ) ;
}
# else
throw ;
# endif
}
catch ( std : : exception & ex ) {
return ex . what ( ) ;
}
catch ( std : : string & msg ) {
return msg ;
}
catch ( const char * msg ) {
return msg ;
}
catch ( . . . ) {
return tryTranslators ( m_translators . begin ( ) ) ;
}
}
2012-08-13 08:46:10 +02:00
2012-08-23 21:08:50 +02:00
std : : string tryTranslators ( std : : vector < const IExceptionTranslator * > : : const_iterator it ) const {
if ( it = = m_translators . end ( ) )
return " Unknown exception " ;
2012-08-13 08:46:10 +02:00
2012-08-23 21:08:50 +02:00
try {
return ( * it ) - > translate ( ) ;
2012-08-13 08:46:10 +02:00
}
2012-08-23 21:08:50 +02:00
catch ( . . . ) {
return tryTranslators ( it + 1 ) ;
2012-05-31 20:40:26 +02:00
}
2012-07-05 19:37:58 +02:00
}
2012-05-31 20:40:26 +02:00
2012-08-23 21:08:50 +02:00
private :
std : : vector < const IExceptionTranslator * > m_translators ;
} ;
}
2012-05-31 20:40:26 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
namespace {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
class RegistryHub : public IRegistryHub , public IMutableRegistryHub {
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
RegistryHub ( RegistryHub const & ) ;
void operator = ( RegistryHub const & ) ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
public : // IRegistryHub
RegistryHub ( ) {
}
2013-04-24 20:10:02 +02:00
virtual IReporterRegistry const & getReporterRegistry ( ) const {
2012-08-23 21:08:50 +02:00
return m_reporterRegistry ;
}
2013-04-24 20:10:02 +02:00
virtual ITestCaseRegistry const & getTestCaseRegistry ( ) const {
2012-08-23 21:08:50 +02:00
return m_testCaseRegistry ;
}
virtual IExceptionTranslatorRegistry & getExceptionTranslatorRegistry ( ) {
return m_exceptionTranslatorRegistry ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
public : // IMutableRegistryHub
2013-04-24 20:10:02 +02:00
virtual void registerReporter ( std : : string const & name , IReporterFactory * factory ) {
2012-08-23 21:08:50 +02:00
m_reporterRegistry . registerReporter ( name , factory ) ;
2011-05-24 09:23:02 +02:00
}
2013-04-24 20:10:02 +02:00
virtual void registerTest ( TestCase const & testInfo ) {
2012-08-23 21:08:50 +02:00
m_testCaseRegistry . registerTest ( testInfo ) ;
}
virtual void registerTranslator ( const IExceptionTranslator * translator ) {
m_exceptionTranslatorRegistry . registerTranslator ( translator ) ;
}
private :
TestRegistry m_testCaseRegistry ;
ReporterRegistry m_reporterRegistry ;
ExceptionTranslatorRegistry m_exceptionTranslatorRegistry ;
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
// Single, global, instance
inline RegistryHub * & getTheRegistryHub ( ) {
static RegistryHub * theRegistryHub = NULL ;
if ( ! theRegistryHub )
theRegistryHub = new RegistryHub ( ) ;
return theRegistryHub ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
IRegistryHub & getRegistryHub ( ) {
return * getTheRegistryHub ( ) ;
}
IMutableRegistryHub & getMutableRegistryHub ( ) {
return * getTheRegistryHub ( ) ;
}
void cleanUp ( ) {
delete getTheRegistryHub ( ) ;
getTheRegistryHub ( ) = NULL ;
cleanUpContext ( ) ;
2012-08-13 08:46:10 +02:00
}
2012-11-01 09:27:09 +01:00
std : : string translateActiveException ( ) {
return getRegistryHub ( ) . getExceptionTranslatorRegistry ( ) . translateActiveException ( ) ;
}
2012-08-09 08:47:30 +02:00
2012-08-13 08:46:10 +02:00
} // end namespace Catch
2012-09-17 07:42:29 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_notimplemented_exception.hpp
# define TWOBLUECUBES_CATCH_NOTIMPLEMENTED_EXCEPTION_HPP_INCLUDED
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
# include <ostream>
2012-08-09 08:47:30 +02:00
2012-08-13 08:46:10 +02:00
namespace Catch {
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
NotImplementedException : : NotImplementedException ( SourceLineInfo const & lineInfo )
2012-08-23 21:08:50 +02:00
: m_lineInfo ( lineInfo ) {
std : : ostringstream oss ;
2013-01-16 10:44:43 +01:00
oss < < lineInfo < < " : function " ;
2012-08-23 21:08:50 +02:00
oss < < " not implemented " ;
m_what = oss . str ( ) ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
const char * NotImplementedException : : what ( ) const throw ( ) {
return m_what . c_str ( ) ;
2012-08-13 08:46:10 +02:00
}
2012-08-09 08:47:30 +02:00
} // end namespace Catch
2012-08-23 21:08:50 +02:00
// #included from: catch_context_impl.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_CONTEXT_IMPL_HPP_INCLUDED
2012-08-13 08:46:10 +02:00
namespace Catch {
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
class Context : public IMutableContext {
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
Context ( ) : m_config ( NULL ) { }
2013-04-24 20:10:02 +02:00
Context ( Context const & ) ;
void operator = ( Context const & ) ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
public : // IContext
virtual IResultCapture & getResultCapture ( ) {
return * m_resultCapture ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
virtual IRunner & getRunner ( ) {
return * m_runner ;
2012-08-09 08:47:30 +02:00
}
2013-04-24 20:10:02 +02:00
virtual size_t getGeneratorIndex ( std : : string const & fileInfo , size_t totalSize ) {
2012-08-23 21:08:50 +02:00
return getGeneratorsForCurrentTest ( )
. getGeneratorInfo ( fileInfo , totalSize )
. getCurrentIndex ( ) ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
virtual bool advanceGeneratorsForCurrentTest ( ) {
IGeneratorsForTest * generators = findGeneratorsForCurrentTest ( ) ;
return generators & & generators - > moveNext ( ) ;
2012-08-09 08:47:30 +02:00
}
2013-06-07 20:07:50 +02:00
virtual Ptr < IConfig const > getConfig ( ) const {
2012-08-23 21:08:50 +02:00
return m_config ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
public : // IMutableContext
virtual void setResultCapture ( IResultCapture * resultCapture ) {
m_resultCapture = resultCapture ;
}
virtual void setRunner ( IRunner * runner ) {
m_runner = runner ;
}
2013-06-07 20:07:50 +02:00
virtual void setConfig ( Ptr < IConfig const > const & config ) {
2012-08-23 21:08:50 +02:00
m_config = config ;
}
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
friend IMutableContext & getCurrentMutableContext ( ) ;
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
private :
IGeneratorsForTest * findGeneratorsForCurrentTest ( ) {
std : : string testName = getResultCapture ( ) . getCurrentTestName ( ) ;
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
std : : map < std : : string , IGeneratorsForTest * > : : const_iterator it =
m_generatorsByTestName . find ( testName ) ;
return it ! = m_generatorsByTestName . end ( )
? it - > second
: NULL ;
2012-08-16 19:48:50 +02:00
}
2012-08-23 21:08:50 +02:00
IGeneratorsForTest & getGeneratorsForCurrentTest ( ) {
IGeneratorsForTest * generators = findGeneratorsForCurrentTest ( ) ;
if ( ! generators ) {
std : : string testName = getResultCapture ( ) . getCurrentTestName ( ) ;
generators = createGeneratorsForTest ( ) ;
m_generatorsByTestName . insert ( std : : make_pair ( testName , generators ) ) ;
}
return * generators ;
2012-08-16 19:48:50 +02:00
}
private :
2012-08-23 21:08:50 +02:00
IRunner * m_runner ;
IResultCapture * m_resultCapture ;
2013-06-07 20:07:50 +02:00
Ptr < IConfig const > m_config ;
2012-08-23 21:08:50 +02:00
std : : map < std : : string , IGeneratorsForTest * > m_generatorsByTestName ;
2012-08-16 19:48:50 +02:00
} ;
2012-08-23 21:08:50 +02:00
namespace {
Context * currentContext = NULL ;
}
IMutableContext & getCurrentMutableContext ( ) {
if ( ! currentContext )
currentContext = new Context ( ) ;
return * currentContext ;
}
IContext & getCurrentContext ( ) {
return getCurrentMutableContext ( ) ;
}
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
Stream createStream ( std : : string const & streamName ) {
2012-09-26 19:38:26 +02:00
if ( streamName = = " stdout " ) return Stream ( std : : cout . rdbuf ( ) , false ) ;
if ( streamName = = " stderr " ) return Stream ( std : : cerr . rdbuf ( ) , false ) ;
if ( streamName = = " debug " ) return Stream ( new StreamBufImpl < OutputDebugWriter > , true ) ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
throw std : : domain_error ( " Unknown stream: " + streamName ) ;
}
2012-08-13 08:46:10 +02:00
2012-08-23 21:08:50 +02:00
void cleanUpContext ( ) {
delete currentContext ;
currentContext = NULL ;
}
}
2012-09-17 07:42:29 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_console_colour_impl.hpp
# define TWOBLUECUBES_CATCH_CONSOLE_COLOUR_IMPL_HPP_INCLUDED
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
namespace Catch { namespace Detail {
struct IColourImpl {
virtual ~ IColourImpl ( ) { }
virtual void use ( Colour : : Code _colourCode ) = 0 ;
} ;
} }
2012-08-09 08:47:30 +02:00
2013-03-11 19:38:29 +01:00
# if defined ( CATCH_PLATFORM_WINDOWS ) /////////////////////////////////////////
2012-08-09 08:47:30 +02:00
2013-06-29 21:31:17 +02:00
# ifndef NOMINMAX
2013-06-28 15:11:28 +02:00
# define NOMINMAX
2013-06-29 21:31:17 +02:00
# endif
2013-07-01 20:02:29 +02:00
# ifdef __AFXDLL
# include <AfxWin.h>
# else
2013-03-11 19:38:29 +01:00
# include <windows.h>
2013-07-01 20:02:29 +02:00
# endif
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
namespace Catch {
2013-03-11 19:38:29 +01:00
namespace {
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
class Win32ColourImpl : public Detail : : IColourImpl {
public :
Win32ColourImpl ( ) : stdoutHandle ( GetStdHandle ( STD_OUTPUT_HANDLE ) )
2013-03-11 19:38:29 +01:00
{
2013-04-05 21:58:01 +02:00
CONSOLE_SCREEN_BUFFER_INFO csbiInfo ;
GetConsoleScreenBufferInfo ( stdoutHandle , & csbiInfo ) ;
originalAttributes = csbiInfo . wAttributes ;
2013-03-11 19:38:29 +01:00
}
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
virtual void use ( Colour : : Code _colourCode ) {
switch ( _colourCode ) {
case Colour : : None : return setTextAttribute ( originalAttributes ) ;
case Colour : : White : return setTextAttribute ( FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE ) ;
case Colour : : Red : return setTextAttribute ( FOREGROUND_RED ) ;
case Colour : : Green : return setTextAttribute ( FOREGROUND_GREEN ) ;
case Colour : : Blue : return setTextAttribute ( FOREGROUND_BLUE ) ;
case Colour : : Cyan : return setTextAttribute ( FOREGROUND_BLUE | FOREGROUND_GREEN ) ;
case Colour : : Yellow : return setTextAttribute ( FOREGROUND_RED | FOREGROUND_GREEN ) ;
case Colour : : Grey : return setTextAttribute ( 0 ) ;
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
case Colour : : LightGrey : return setTextAttribute ( FOREGROUND_INTENSITY ) ;
case Colour : : BrightRed : return setTextAttribute ( FOREGROUND_INTENSITY | FOREGROUND_RED ) ;
case Colour : : BrightGreen : return setTextAttribute ( FOREGROUND_INTENSITY | FOREGROUND_GREEN ) ;
2013-06-28 15:11:28 +02:00
case Colour : : BrightWhite : return setTextAttribute ( FOREGROUND_INTENSITY | FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE ) ;
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
case Colour : : Bright : throw std : : logic_error ( " not a colour " ) ;
}
2013-03-11 19:38:29 +01:00
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
private :
2013-04-05 21:58:01 +02:00
void setTextAttribute ( WORD _textAttribute ) {
SetConsoleTextAttribute ( stdoutHandle , _textAttribute ) ;
}
HANDLE stdoutHandle ;
WORD originalAttributes ;
2012-08-23 21:08:50 +02:00
} ;
2012-08-09 08:47:30 +02:00
2013-03-11 19:38:29 +01:00
inline bool shouldUseColourForPlatform ( ) {
return true ;
}
2013-04-05 21:58:01 +02:00
Win32ColourImpl platformColourImpl ;
2013-03-11 19:38:29 +01:00
} // end anon namespace
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2012-08-09 08:47:30 +02:00
2013-03-11 19:38:29 +01:00
# else // Not Windows - assumed to be POSIX compatible //////////////////////////
2012-11-01 09:27:09 +01:00
# include <unistd.h>
namespace Catch {
2013-03-11 19:38:29 +01:00
namespace {
2012-11-01 09:27:09 +01:00
// use POSIX/ ANSI console terminal codes
2013-04-05 21:58:01 +02:00
// Thanks to Adam Strzelecki for original contribution
// (http://github.com/nanoant)
2012-11-01 09:27:09 +01:00
// https://github.com/philsquared/Catch/pull/131
2013-04-05 21:58:01 +02:00
class PosixColourImpl : public Detail : : IColourImpl {
public :
virtual void use ( Colour : : Code _colourCode ) {
switch ( _colourCode ) {
case Colour : : None :
case Colour : : White : return setColour ( " [0m " ) ;
case Colour : : Red : return setColour ( " [0;31m " ) ;
case Colour : : Green : return setColour ( " [0;32m " ) ;
case Colour : : Blue : return setColour ( " [0:34m " ) ;
case Colour : : Cyan : return setColour ( " [0;36m " ) ;
case Colour : : Yellow : return setColour ( " [0;33m " ) ;
case Colour : : Grey : return setColour ( " [1;30m " ) ;
2012-11-01 09:27:09 +01:00
2013-04-05 21:58:01 +02:00
case Colour : : LightGrey : return setColour ( " [0;37m " ) ;
case Colour : : BrightRed : return setColour ( " [1;31m " ) ;
2013-04-12 11:43:06 +02:00
case Colour : : BrightGreen : return setColour ( " [1;32m " ) ;
2013-04-05 21:58:01 +02:00
case Colour : : BrightWhite : return setColour ( " [1;37m " ) ;
2012-11-01 09:27:09 +01:00
2013-04-05 21:58:01 +02:00
case Colour : : Bright : throw std : : logic_error ( " not a colour " ) ;
2012-11-01 09:27:09 +01:00
}
}
2013-04-05 21:58:01 +02:00
private :
void setColour ( const char * _escapeCode ) {
std : : cout < < ' \033 ' < < _escapeCode ;
2012-10-29 21:49:22 +01:00
}
2013-03-11 19:38:29 +01:00
} ;
inline bool shouldUseColourForPlatform ( ) {
return isatty ( fileno ( stdout ) ) ;
2012-11-01 09:27:09 +01:00
}
2013-04-05 21:58:01 +02:00
PosixColourImpl platformColourImpl ;
2012-11-01 09:27:09 +01:00
2013-04-05 21:58:01 +02:00
} // end anon namespace
} // end namespace Catch
2012-08-09 08:47:30 +02:00
2013-03-11 19:38:29 +01:00
# endif // not Windows
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
namespace {
2013-04-05 21:58:01 +02:00
struct NoColourImpl : Detail : : IColourImpl {
void use ( Colour : : Code ) { }
} ;
NoColourImpl noColourImpl ;
static const bool shouldUseColour = shouldUseColourForPlatform ( ) & &
! isDebuggerActive ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
Colour : : Colour ( Code _colourCode ) { use ( _colourCode ) ; }
Colour : : ~ Colour ( ) { use ( None ) ; }
void Colour : : use ( Code _colourCode ) {
impl - > use ( _colourCode ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2013-04-05 21:58:01 +02:00
Detail : : IColourImpl * Colour : : impl = shouldUseColour
? static_cast < Detail : : IColourImpl * > ( & platformColourImpl )
: static_cast < Detail : : IColourImpl * > ( & noColourImpl ) ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
// #included from: catch_generators_impl.hpp
# define TWOBLUECUBES_CATCH_GENERATORS_IMPL_HPP_INCLUDED
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
# include <vector>
# include <string>
# include <map>
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
struct GeneratorInfo : IGeneratorInfo {
2012-08-13 08:46:10 +02:00
2012-08-23 21:08:50 +02:00
GeneratorInfo ( std : : size_t size )
: m_size ( size ) ,
m_currentIndex ( 0 )
{ }
2012-08-13 08:46:10 +02:00
2012-08-23 21:08:50 +02:00
bool moveNext ( ) {
if ( + + m_currentIndex = = m_size ) {
m_currentIndex = 0 ;
return false ;
}
2012-08-13 08:46:10 +02:00
return true ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
std : : size_t getCurrentIndex ( ) const {
return m_currentIndex ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
std : : size_t m_size ;
std : : size_t m_currentIndex ;
} ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
///////////////////////////////////////////////////////////////////////////
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
class GeneratorsForTest : public IGeneratorsForTest {
public :
~ GeneratorsForTest ( ) {
deleteAll ( m_generatorsInOrder ) ;
2012-08-13 08:46:10 +02:00
}
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
IGeneratorInfo & getGeneratorInfo ( std : : string const & fileInfo , std : : size_t size ) {
2012-08-23 21:08:50 +02:00
std : : map < std : : string , IGeneratorInfo * > : : const_iterator it = m_generatorsByName . find ( fileInfo ) ;
if ( it = = m_generatorsByName . end ( ) ) {
IGeneratorInfo * info = new GeneratorInfo ( size ) ;
m_generatorsByName . insert ( std : : make_pair ( fileInfo , info ) ) ;
m_generatorsInOrder . push_back ( info ) ;
return * info ;
}
return * it - > second ;
2012-08-13 08:46:10 +02:00
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
bool moveNext ( ) {
std : : vector < IGeneratorInfo * > : : const_iterator it = m_generatorsInOrder . begin ( ) ;
std : : vector < IGeneratorInfo * > : : const_iterator itEnd = m_generatorsInOrder . end ( ) ;
for ( ; it ! = itEnd ; + + it ) {
if ( ( * it ) - > moveNext ( ) )
return true ;
}
return false ;
2012-08-13 08:46:10 +02:00
}
2012-08-09 08:47:30 +02:00
2012-08-13 08:46:10 +02:00
private :
2012-08-23 21:08:50 +02:00
std : : map < std : : string , IGeneratorInfo * > m_generatorsByName ;
std : : vector < IGeneratorInfo * > m_generatorsInOrder ;
} ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
IGeneratorsForTest * createGeneratorsForTest ( )
{
return new GeneratorsForTest ( ) ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_assertionresult.hpp
# define TWOBLUECUBES_CATCH_ASSERTIONRESULT_HPP_INCLUDED
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2012-08-13 08:46:10 +02:00
2013-04-24 20:10:02 +02:00
AssertionInfo : : AssertionInfo ( std : : string const & _macroName ,
SourceLineInfo const & _lineInfo ,
std : : string const & _capturedExpression ,
2012-11-13 23:04:29 +01:00
ResultDisposition : : Flags _resultDisposition )
: macroName ( _macroName ) ,
lineInfo ( _lineInfo ) ,
capturedExpression ( _capturedExpression ) ,
resultDisposition ( _resultDisposition )
2013-06-07 20:07:50 +02:00
{ }
2012-11-13 23:04:29 +01:00
2012-11-01 09:27:09 +01:00
AssertionResult : : AssertionResult ( ) { }
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
AssertionResult : : AssertionResult ( AssertionInfo const & info , AssertionResultData const & data )
2012-11-01 09:27:09 +01:00
: m_info ( info ) ,
m_resultData ( data )
{ }
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
AssertionResult : : ~ AssertionResult ( ) { }
2012-08-09 08:47:30 +02:00
2012-11-13 23:04:29 +01:00
// Result was a success
bool AssertionResult : : succeeded ( ) const {
return Catch : : isOk ( m_resultData . resultType ) ;
}
// Result was a success, or failure is suppressed
bool AssertionResult : : isOk ( ) const {
return Catch : : isOk ( m_resultData . resultType ) | | shouldSuppressFailure ( m_info . resultDisposition ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
ResultWas : : OfType AssertionResult : : getResultType ( ) const {
return m_resultData . resultType ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
bool AssertionResult : : hasExpression ( ) const {
return ! m_info . capturedExpression . empty ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
bool AssertionResult : : hasMessage ( ) const {
return ! m_resultData . message . empty ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
std : : string AssertionResult : : getExpression ( ) const {
2013-06-07 20:07:50 +02:00
if ( shouldNegate ( m_info . resultDisposition ) )
return " ! " + m_info . capturedExpression ;
else
return m_info . capturedExpression ;
}
std : : string AssertionResult : : getExpressionInMacro ( ) const {
if ( m_info . macroName . empty ( ) )
return m_info . capturedExpression ;
else
return m_info . macroName + " ( " + m_info . capturedExpression + " ) " ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
bool AssertionResult : : hasExpandedExpression ( ) const {
2012-10-09 12:48:55 +02:00
return hasExpression ( ) & & getExpandedExpression ( ) ! = getExpression ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-16 19:48:50 +02:00
2012-11-01 09:27:09 +01:00
std : : string AssertionResult : : getExpandedExpression ( ) const {
return m_resultData . reconstructedExpression ;
2012-08-16 19:48:50 +02:00
}
2012-11-01 09:27:09 +01:00
std : : string AssertionResult : : getMessage ( ) const {
return m_resultData . message ;
2012-08-23 21:08:50 +02:00
}
2012-11-01 09:27:09 +01:00
SourceLineInfo AssertionResult : : getSourceInfo ( ) const {
return m_info . lineInfo ;
2012-08-23 21:08:50 +02:00
}
2012-11-01 09:27:09 +01:00
std : : string AssertionResult : : getTestMacroName ( ) const {
return m_info . macroName ;
2012-08-16 19:48:50 +02:00
}
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2012-08-16 19:48:50 +02:00
2012-11-01 09:27:09 +01:00
// #included from: catch_expressionresult_builder.hpp
# define TWOBLUECUBES_CATCH_EXPRESSIONRESULT_BUILDER_HPP_INCLUDED
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
# include <assert.h>
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
namespace Catch {
2012-10-04 09:14:48 +02:00
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder : : ExpressionResultBuilder ( ResultWas : : OfType resultType ) {
m_data . resultType = resultType ;
2012-10-09 12:48:55 +02:00
}
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder : : ExpressionResultBuilder ( ExpressionResultBuilder const & other )
2012-11-01 09:27:09 +01:00
: m_data ( other . m_data ) ,
m_exprComponents ( other . m_exprComponents )
{
m_stream < < other . m_stream . str ( ) ;
2012-10-09 12:48:55 +02:00
}
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & ExpressionResultBuilder : : operator = ( ExpressionResultBuilder const & other ) {
2012-11-01 09:27:09 +01:00
m_data = other . m_data ;
m_exprComponents = other . m_exprComponents ;
m_stream . str ( " " ) ;
m_stream < < other . m_stream . str ( ) ;
2012-10-04 09:14:48 +02:00
return * this ;
2012-08-09 08:47:30 +02:00
}
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder & ExpressionResultBuilder : : setResultType ( ResultWas : : OfType result ) {
m_data . resultType = result ;
2012-10-04 09:14:48 +02:00
return * this ;
2012-08-09 08:47:30 +02:00
}
2012-11-01 09:27:09 +01:00
ExpressionResultBuilder & ExpressionResultBuilder : : setResultType ( bool result ) {
m_data . resultType = result ? ResultWas : : Ok : ResultWas : : ExpressionFailed ;
2012-10-09 12:48:55 +02:00
return * this ;
}
2012-11-13 23:04:29 +01:00
ExpressionResultBuilder & ExpressionResultBuilder : : endExpression ( ResultDisposition : : Flags resultDisposition ) {
m_exprComponents . shouldNegate = shouldNegate ( resultDisposition ) ;
2012-10-04 09:14:48 +02:00
return * this ;
2012-08-09 08:47:30 +02:00
}
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & ExpressionResultBuilder : : setLhs ( std : : string const & lhs ) {
2012-11-01 09:27:09 +01:00
m_exprComponents . lhs = lhs ;
2012-10-04 09:14:48 +02:00
return * this ;
2012-08-23 21:08:50 +02:00
}
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & ExpressionResultBuilder : : setRhs ( std : : string const & rhs ) {
2012-11-01 09:27:09 +01:00
m_exprComponents . rhs = rhs ;
2012-10-04 09:14:48 +02:00
return * this ;
2012-08-23 21:08:50 +02:00
}
2013-04-24 20:10:02 +02:00
ExpressionResultBuilder & ExpressionResultBuilder : : setOp ( std : : string const & op ) {
2012-11-01 09:27:09 +01:00
m_exprComponents . op = op ;
2012-10-04 09:14:48 +02:00
return * this ;
}
2013-04-24 20:10:02 +02:00
AssertionResult ExpressionResultBuilder : : buildResult ( AssertionInfo const & info ) const
2012-10-09 12:48:55 +02:00
{
2012-11-01 09:27:09 +01:00
assert ( m_data . resultType ! = ResultWas : : Unknown ) ;
AssertionResultData data = m_data ;
2012-10-28 13:07:17 +01:00
// Flip bool results if shouldNegate is set
if ( m_exprComponents . shouldNegate & & data . resultType = = ResultWas : : Ok )
data . resultType = ResultWas : : ExpressionFailed ;
else if ( m_exprComponents . shouldNegate & & data . resultType = = ResultWas : : ExpressionFailed )
data . resultType = ResultWas : : Ok ;
data . message = m_stream . str ( ) ;
data . reconstructedExpression = reconstructExpression ( info ) ;
if ( m_exprComponents . shouldNegate ) {
if ( m_exprComponents . op = = " " )
2012-10-09 12:48:55 +02:00
data . reconstructedExpression = " ! " + data . reconstructedExpression ;
2012-10-28 13:07:17 +01:00
else
2012-10-09 12:48:55 +02:00
data . reconstructedExpression = " !( " + data . reconstructedExpression + " ) " ;
}
2012-10-28 13:07:17 +01:00
return AssertionResult ( info , data ) ;
}
2013-04-24 20:10:02 +02:00
std : : string ExpressionResultBuilder : : reconstructExpression ( AssertionInfo const & info ) const {
2012-10-28 13:07:17 +01:00
if ( m_exprComponents . op = = " " )
return m_exprComponents . lhs . empty ( ) ? info . capturedExpression : m_exprComponents . op + m_exprComponents . lhs ;
else if ( m_exprComponents . op = = " matches " )
return m_exprComponents . lhs + " " + m_exprComponents . rhs ;
else if ( m_exprComponents . op ! = " ! " ) {
2013-03-25 10:25:31 +01:00
if ( m_exprComponents . lhs . size ( ) + m_exprComponents . rhs . size ( ) < 40 & &
m_exprComponents . lhs . find ( " \n " ) = = std : : string : : npos & &
m_exprComponents . rhs . find ( " \n " ) = = std : : string : : npos )
2012-10-28 13:07:17 +01:00
return m_exprComponents . lhs + " " + m_exprComponents . op + " " + m_exprComponents . rhs ;
2012-10-09 12:48:55 +02:00
else
2013-03-04 12:19:15 +01:00
return m_exprComponents . lhs + " \n " + m_exprComponents . op + " \n " + m_exprComponents . rhs ;
2012-10-09 12:48:55 +02:00
}
else
2012-10-28 13:07:17 +01:00
return " {can't expand - use " + info . macroName + " _FALSE( " + info . capturedExpression . substr ( 1 ) + " ) instead of " + info . macroName + " ( " + info . capturedExpression + " ) for better diagnostics} " ;
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
} // end namespace Catch
2012-08-23 21:08:50 +02:00
// #included from: catch_test_case_info.hpp
2012-09-17 07:42:29 +02:00
# define TWOBLUECUBES_CATCH_TEST_CASE_INFO_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
namespace Catch {
2012-12-02 00:57:18 +01:00
TestCase makeTestCase ( ITestCase * _testCase ,
2013-04-24 20:10:02 +02:00
std : : string const & _className ,
std : : string const & _name ,
std : : string const & _descOrTags ,
SourceLineInfo const & _lineInfo )
2012-09-15 18:53:27 +02:00
{
2012-12-02 00:57:18 +01:00
std : : string desc = _descOrTags ;
bool isHidden ( startsWith ( _name , " ./ " ) ) ;
std : : set < std : : string > tags ;
TagExtracter ( tags ) . parse ( desc ) ;
2013-06-28 18:09:57 +02:00
if ( tags . find ( " hide " ) ! = tags . end ( ) | | tags . find ( " . " ) ! = tags . end ( ) )
2012-12-02 00:57:18 +01:00
isHidden = true ;
TestCaseInfo info ( _name , _className , desc , tags , isHidden , _lineInfo ) ;
return TestCase ( _testCase , info ) ;
}
2013-04-24 20:10:02 +02:00
TestCaseInfo : : TestCaseInfo ( std : : string const & _name ,
std : : string const & _className ,
std : : string const & _description ,
std : : set < std : : string > const & _tags ,
2012-12-02 00:57:18 +01:00
bool _isHidden ,
2013-04-24 20:10:02 +02:00
SourceLineInfo const & _lineInfo )
2012-12-02 00:57:18 +01:00
: name ( _name ) ,
className ( _className ) ,
description ( _description ) ,
tags ( _tags ) ,
2012-12-06 09:44:51 +01:00
lineInfo ( _lineInfo ) ,
isHidden ( _isHidden )
2013-03-29 14:44:33 +01:00
{
std : : ostringstream oss ;
for ( std : : set < std : : string > : : const_iterator it = _tags . begin ( ) , itEnd = _tags . end ( ) ; it ! = itEnd ; + + it )
oss < < " [ " < < * it < < " ] " ;
tagsAsString = oss . str ( ) ;
}
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
TestCaseInfo : : TestCaseInfo ( TestCaseInfo const & other )
2012-12-02 00:57:18 +01:00
: name ( other . name ) ,
className ( other . className ) ,
description ( other . description ) ,
tags ( other . tags ) ,
2013-03-29 14:44:33 +01:00
tagsAsString ( other . tagsAsString ) ,
2012-12-06 09:44:51 +01:00
lineInfo ( other . lineInfo ) ,
isHidden ( other . isHidden )
2012-08-23 21:08:50 +02:00
{ }
2012-08-16 19:48:50 +02:00
2013-04-24 20:10:02 +02:00
TestCase : : TestCase ( ITestCase * testCase , TestCaseInfo const & info ) : TestCaseInfo ( info ) , test ( testCase ) { }
2012-12-02 00:57:18 +01:00
2013-04-24 20:10:02 +02:00
TestCase : : TestCase ( TestCase const & other )
2012-12-02 00:57:18 +01:00
: TestCaseInfo ( other ) ,
test ( other . test )
2012-08-23 21:08:50 +02:00
{ }
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
TestCase TestCase : : withName ( std : : string const & _newName ) const {
2012-12-02 00:57:18 +01:00
TestCase other ( * this ) ;
other . name = _newName ;
return other ;
2012-08-23 21:08:50 +02:00
}
2012-08-16 19:48:50 +02:00
2012-12-02 00:57:18 +01:00
void TestCase : : invoke ( ) const {
test - > invoke ( ) ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
bool TestCase : : isHidden ( ) const {
return TestCaseInfo : : isHidden ;
2012-09-15 18:53:27 +02:00
}
2013-04-24 20:10:02 +02:00
bool TestCase : : hasTag ( std : : string const & tag ) const {
2013-03-25 10:25:31 +01:00
return tags . find ( toLower ( tag ) ) ! = tags . end ( ) ;
2012-09-15 18:53:27 +02:00
}
2013-04-24 20:10:02 +02:00
bool TestCase : : matchesTags ( std : : string const & tagPattern ) const {
2012-09-24 09:30:13 +02:00
TagExpression exp ;
TagExpressionParser ( exp ) . parse ( tagPattern ) ;
2012-12-02 00:57:18 +01:00
return exp . matches ( tags ) ;
2012-09-24 09:30:13 +02:00
}
2013-04-24 20:10:02 +02:00
std : : set < std : : string > const & TestCase : : getTags ( ) const {
2012-12-02 00:57:18 +01:00
return tags ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
void TestCase : : swap ( TestCase & other ) {
test . swap ( other . test ) ;
className . swap ( other . className ) ;
name . swap ( other . name ) ;
description . swap ( other . description ) ;
std : : swap ( lineInfo , other . lineInfo ) ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
bool TestCase : : operator = = ( TestCase const & other ) const {
2012-12-02 00:57:18 +01:00
return test . get ( ) = = other . test . get ( ) & &
name = = other . name & &
className = = other . className ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
bool TestCase : : operator < ( TestCase const & other ) const {
2012-12-02 00:57:18 +01:00
return name < other . name ;
2012-08-23 21:08:50 +02:00
}
2013-04-24 20:10:02 +02:00
TestCase & TestCase : : operator = ( TestCase const & other ) {
2012-12-02 00:57:18 +01:00
TestCase temp ( other ) ;
2012-08-23 21:08:50 +02:00
swap ( temp ) ;
return * this ;
}
2012-09-24 09:30:13 +02:00
2013-04-24 20:10:02 +02:00
TestCaseInfo const & TestCase : : getTestCaseInfo ( ) const
2012-12-02 00:57:18 +01:00
{
return * this ;
}
2012-09-24 09:30:13 +02:00
} // end namespace Catch
2011-05-24 09:23:02 +02:00
2012-11-15 23:15:41 +01:00
// #included from: catch_version.hpp
# define TWOBLUECUBES_CATCH_VERSION_HPP_INCLUDED
namespace Catch {
// These numbers are maintained by a script
2013-07-02 09:49:29 +02:00
Version libraryVersion ( 1 , 0 , 5 , " master " ) ;
2012-11-15 23:15:41 +01:00
}
2013-04-20 22:07:32 +02:00
// #included from: catch_text.hpp
# define TWOBLUECUBES_CATCH_TEXT_HPP_INCLUDED
2013-01-16 10:44:43 +01:00
2013-04-20 22:07:32 +02:00
# include <string>
# include <vector>
2013-03-29 14:44:33 +01:00
2013-04-20 22:07:32 +02:00
namespace Catch {
2012-11-01 09:27:09 +01:00
2013-04-20 22:07:32 +02:00
Text : : Text ( std : : string const & _str , TextAttributes const & _attr )
: attr ( _attr )
{
std : : string wrappableChars = " [({.,/| \\ - " ;
std : : size_t indent = _attr . initialIndent ! = std : : string : : npos
? _attr . initialIndent
: _attr . indent ;
std : : string remainder = _str ;
while ( ! remainder . empty ( ) ) {
assert ( lines . size ( ) < 1000 ) ;
std : : size_t tabPos = std : : string : : npos ;
std : : size_t width = ( std : : min ) ( remainder . size ( ) , _attr . width - indent ) ;
std : : size_t pos = remainder . find_first_of ( ' \n ' ) ;
if ( pos < = width ) {
width = pos ;
}
pos = remainder . find_last_of ( _attr . tabChar , width ) ;
if ( pos ! = std : : string : : npos ) {
tabPos = pos ;
if ( remainder [ width ] = = ' \n ' )
width - - ;
remainder = remainder . substr ( 0 , tabPos ) + remainder . substr ( tabPos + 1 ) ;
}
if ( width = = remainder . size ( ) ) {
spliceLine ( indent , remainder , width ) ;
}
else if ( remainder [ width ] = = ' \n ' ) {
spliceLine ( indent , remainder , width ) ;
if ( width < = 1 | | remainder . size ( ) ! = 1 )
remainder = remainder . substr ( 1 ) ;
indent = _attr . indent ;
}
else {
pos = remainder . find_last_of ( wrappableChars , width ) ;
if ( pos ! = std : : string : : npos & & pos > 0 ) {
spliceLine ( indent , remainder , pos ) ;
if ( remainder [ 0 ] = = ' ' )
remainder = remainder . substr ( 1 ) ;
2013-01-16 10:44:43 +01:00
}
2013-03-29 14:44:33 +01:00
else {
2013-04-20 22:07:32 +02:00
spliceLine ( indent , remainder , width - 1 ) ;
lines . back ( ) + = " - " ;
2013-01-16 10:44:43 +01:00
}
2013-04-20 22:07:32 +02:00
if ( lines . size ( ) = = 1 )
indent = _attr . indent ;
if ( tabPos ! = std : : string : : npos )
indent + = tabPos ;
2013-01-16 10:44:43 +01:00
}
2012-10-09 12:48:55 +02:00
}
2012-11-01 09:27:09 +01:00
}
2013-01-16 10:44:43 +01:00
2013-04-20 22:07:32 +02:00
void Text : : spliceLine ( std : : size_t _indent , std : : string & _remainder , std : : size_t _pos ) {
lines . push_back ( std : : string ( _indent , ' ' ) + _remainder . substr ( 0 , _pos ) ) ;
_remainder = _remainder . substr ( _pos ) ;
2013-01-16 10:44:43 +01:00
}
2013-04-20 22:07:32 +02:00
std : : string Text : : toString ( ) const {
2013-01-16 10:44:43 +01:00
std : : ostringstream oss ;
2013-03-29 14:44:33 +01:00
oss < < * this ;
2013-01-16 10:44:43 +01:00
return oss . str ( ) ;
}
2013-04-20 22:07:32 +02:00
std : : ostream & operator < < ( std : : ostream & _stream , Text const & _text ) {
for ( Text : : const_iterator it = _text . begin ( ) , itEnd = _text . end ( ) ;
it ! = itEnd ; + + it ) {
if ( it ! = _text . begin ( ) )
_stream < < " \n " ;
_stream < < * it ;
2012-10-09 12:48:55 +02:00
}
2013-04-20 22:07:32 +02:00
return _stream ;
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
} // end namespace Catch
2013-02-02 21:37:58 +01:00
// #included from: catch_message.hpp
# define TWOBLUECUBES_CATCH_MESSAGE_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-05-16 16:09:17 +02:00
namespace Catch {
2013-02-02 21:37:58 +01:00
MessageInfo : : MessageInfo ( std : : string const & _macroName ,
SourceLineInfo const & _lineInfo ,
ResultWas : : OfType _type )
: macroName ( _macroName ) ,
lineInfo ( _lineInfo ) ,
type ( _type ) ,
sequence ( + + globalCount )
{ }
2011-05-24 09:23:02 +02:00
2013-02-02 21:37:58 +01:00
// This may need protecting if threading support is added
unsigned int MessageInfo : : globalCount = 0 ;
////////////////////////////////////////////////////////////////////////////
2013-06-28 18:09:57 +02:00
ScopedMessage : : ScopedMessage ( MessageBuilder const & builder )
: m_info ( builder . m_info )
{
m_info . message = builder . m_stream . str ( ) ;
getResultCapture ( ) . pushScopedMessage ( m_info ) ;
2013-02-02 21:37:58 +01:00
}
2013-06-28 18:09:57 +02:00
ScopedMessage : : ~ ScopedMessage ( ) {
getResultCapture ( ) . popScopedMessage ( m_info ) ;
2013-02-02 21:37:58 +01:00
}
} // end namespace Catch
2013-04-12 11:43:06 +02:00
// #included from: catch_legacy_reporter_adapter.hpp
2013-04-08 12:50:41 +02:00
# define TWOBLUECUBES_CATCH_LEGACY_REPORTER_ADAPTER_HPP_INCLUDED
// #included from: catch_legacy_reporter_adapter.h
# define TWOBLUECUBES_CATCH_LEGACY_REPORTER_ADAPTER_H_INCLUDED
namespace Catch
{
class LegacyReporterAdapter : public SharedImpl < IStreamingReporter >
{
public :
2013-06-07 20:07:50 +02:00
LegacyReporterAdapter ( Ptr < IReporter > const & legacyReporter ) ;
2013-04-08 12:50:41 +02:00
virtual ~ LegacyReporterAdapter ( ) ;
virtual ReporterPreferences getPreferences ( ) const ;
virtual void noMatchingTestCases ( std : : string const & ) ;
virtual void testRunStarting ( TestRunInfo const & ) ;
virtual void testGroupStarting ( GroupInfo const & groupInfo ) ;
virtual void testCaseStarting ( TestCaseInfo const & testInfo ) ;
virtual void sectionStarting ( SectionInfo const & sectionInfo ) ;
virtual void assertionStarting ( AssertionInfo const & ) ;
2013-06-28 18:09:57 +02:00
virtual bool assertionEnded ( AssertionStats const & assertionStats ) ;
2013-04-08 12:50:41 +02:00
virtual void sectionEnded ( SectionStats const & sectionStats ) ;
virtual void testCaseEnded ( TestCaseStats const & testCaseStats ) ;
virtual void testGroupEnded ( TestGroupStats const & testGroupStats ) ;
virtual void testRunEnded ( TestRunStats const & testRunStats ) ;
private :
Ptr < IReporter > m_legacyReporter ;
} ;
}
namespace Catch
{
2013-06-07 20:07:50 +02:00
LegacyReporterAdapter : : LegacyReporterAdapter ( Ptr < IReporter > const & legacyReporter )
: m_legacyReporter ( legacyReporter )
2012-08-23 21:08:50 +02:00
{ }
2013-04-08 12:50:41 +02:00
LegacyReporterAdapter : : ~ LegacyReporterAdapter ( ) { }
2012-08-09 08:47:30 +02:00
2013-04-08 12:50:41 +02:00
ReporterPreferences LegacyReporterAdapter : : getPreferences ( ) const {
ReporterPreferences prefs ;
prefs . shouldRedirectStdOut = m_legacyReporter - > shouldRedirectStdout ( ) ;
return prefs ;
2012-08-23 21:08:50 +02:00
}
2012-08-16 19:48:50 +02:00
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : noMatchingTestCases ( std : : string const & ) { }
void LegacyReporterAdapter : : testRunStarting ( TestRunInfo const & ) {
m_legacyReporter - > StartTesting ( ) ;
2012-11-13 23:04:29 +01:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : testGroupStarting ( GroupInfo const & groupInfo ) {
m_legacyReporter - > StartGroup ( groupInfo . name ) ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : testCaseStarting ( TestCaseInfo const & testInfo ) {
m_legacyReporter - > StartTestCase ( testInfo ) ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : sectionStarting ( SectionInfo const & sectionInfo ) {
m_legacyReporter - > StartSection ( sectionInfo . name , sectionInfo . description ) ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : assertionStarting ( AssertionInfo const & ) {
// Not on legacy interface
2012-09-15 18:53:27 +02:00
}
2013-06-28 18:09:57 +02:00
bool LegacyReporterAdapter : : assertionEnded ( AssertionStats const & assertionStats ) {
2013-04-08 12:50:41 +02:00
if ( assertionStats . assertionResult . getResultType ( ) ! = ResultWas : : Ok ) {
for ( std : : vector < MessageInfo > : : const_iterator it = assertionStats . infoMessages . begin ( ) , itEnd = assertionStats . infoMessages . end ( ) ;
it ! = itEnd ;
+ + it ) {
if ( it - > type = = ResultWas : : Info ) {
ExpressionResultBuilder expressionBuilder ( it - > type ) ;
expressionBuilder < < it - > message ;
AssertionInfo info ( it - > macroName , it - > lineInfo , " " , ResultDisposition : : Normal ) ;
AssertionResult result = expressionBuilder . buildResult ( info ) ;
m_legacyReporter - > Result ( result ) ;
}
}
}
m_legacyReporter - > Result ( assertionStats . assertionResult ) ;
2013-06-28 18:09:57 +02:00
return true ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : sectionEnded ( SectionStats const & sectionStats ) {
if ( sectionStats . missingAssertions )
m_legacyReporter - > NoAssertionsInSection ( sectionStats . sectionInfo . name ) ;
m_legacyReporter - > EndSection ( sectionStats . sectionInfo . name , sectionStats . assertions ) ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : testCaseEnded ( TestCaseStats const & testCaseStats ) {
if ( testCaseStats . missingAssertions )
m_legacyReporter - > NoAssertionsInTestCase ( testCaseStats . testInfo . name ) ;
m_legacyReporter - > EndTestCase
( testCaseStats . testInfo ,
testCaseStats . totals ,
testCaseStats . stdOut ,
testCaseStats . stdErr ) ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : testGroupEnded ( TestGroupStats const & testGroupStats ) {
if ( testGroupStats . aborting )
m_legacyReporter - > Aborted ( ) ;
m_legacyReporter - > EndGroup ( testGroupStats . groupInfo . name , testGroupStats . totals ) ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
void LegacyReporterAdapter : : testRunEnded ( TestRunStats const & testRunStats ) {
m_legacyReporter - > EndTesting ( testRunStats . totals ) ;
2012-08-23 21:08:50 +02:00
}
2013-04-08 12:50:41 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
// #included from: ../reporters/catch_reporter_basic.hpp
# define TWOBLUECUBES_CATCH_REPORTER_BASIC_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
// #included from: ../internal/catch_reporter_registrars.hpp
# define TWOBLUECUBES_CATCH_REPORTER_REGISTRARS_HPP_INCLUDED
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-12-02 00:57:18 +01:00
template < typename T >
class LegacyReporterRegistrar {
class ReporterFactory : public IReporterFactory {
2013-04-24 20:10:02 +02:00
virtual IStreamingReporter * create ( ReporterConfig const & config ) const {
2013-06-07 20:07:50 +02:00
return new LegacyReporterAdapter ( new T ( config ) ) ;
2012-12-02 00:57:18 +01:00
}
virtual std : : string getDescription ( ) const {
return T : : getDescription ( ) ;
}
} ;
public :
2013-04-24 20:10:02 +02:00
LegacyReporterRegistrar ( std : : string const & name ) {
2012-12-02 00:57:18 +01:00
getMutableRegistryHub ( ) . registerReporter ( name , new ReporterFactory ( ) ) ;
}
} ;
2012-08-23 21:08:50 +02:00
template < typename T >
class ReporterRegistrar {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
class ReporterFactory : public IReporterFactory {
2012-08-09 08:47:30 +02:00
2012-12-02 00:57:18 +01:00
// *** Please Note ***:
// - If you end up here looking at a compiler error because it's trying to register
// your custom reporter class be aware that the native reporter interface has changed
// to IStreamingReporter. The "legacy" interface, IReporter, is still supported via
// an adapter. Just use REGISTER_LEGACY_REPORTER to take advantage of the adapter.
// However please consider updating to the new interface as the old one is now
// deprecated and will probably be removed quite soon!
// Please contact me via github if you have any questions at all about this.
// In fact, ideally, please contact me anyway to let me know you've hit this - as I have
// no idea who is actually using custom reporters at all (possibly no-one!).
// The new interface is designed to minimise exposure to interface changes in the future.
2013-04-24 20:10:02 +02:00
virtual IStreamingReporter * create ( ReporterConfig const & config ) const {
2012-08-23 21:08:50 +02:00
return new T ( config ) ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
virtual std : : string getDescription ( ) const {
return T : : getDescription ( ) ;
}
} ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
public :
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
ReporterRegistrar ( std : : string const & name ) {
2012-08-23 21:08:50 +02:00
getMutableRegistryHub ( ) . registerReporter ( name , new ReporterFactory ( ) ) ;
}
} ;
}
2012-08-09 08:47:30 +02:00
2012-12-02 00:57:18 +01:00
# define INTERNAL_CATCH_REGISTER_LEGACY_REPORTER( name, reporterType ) \
Catch : : LegacyReporterRegistrar < reporterType > catch_internal_RegistrarFor # # reporterType ( name ) ;
2012-08-23 21:08:50 +02:00
# define INTERNAL_CATCH_REGISTER_REPORTER( name, reporterType ) \
Catch : : ReporterRegistrar < reporterType > catch_internal_RegistrarFor # # reporterType ( name ) ;
2012-08-09 08:47:30 +02:00
namespace Catch {
2012-08-23 21:08:50 +02:00
class BasicReporter : public SharedImpl < IReporter > {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
struct SpanInfo {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
SpanInfo ( )
: emitted ( false )
{ }
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
SpanInfo ( const std : : string & spanName )
: name ( spanName ) ,
emitted ( false )
{ }
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
SpanInfo ( const SpanInfo & other )
: name ( other . name ) ,
emitted ( other . emitted )
{ }
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
std : : string name ;
bool emitted ;
} ;
2012-08-09 08:47:30 +02:00
public :
2012-08-23 21:08:50 +02:00
BasicReporter ( const ReporterConfig & config )
: m_config ( config ) ,
m_firstSectionInTestCase ( true ) ,
m_aborted ( false )
{ }
virtual ~ BasicReporter ( ) ;
static std : : string getDescription ( ) {
return " Reports test results as lines of text " ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
private :
void ReportCounts ( const std : : string & label , const Counts & counts , const std : : string & allPrefix = " All " ) {
if ( counts . passed )
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < counts . failed < < " of " < < counts . total ( ) < < " " < < label < < " s failed " ;
2012-08-23 21:08:50 +02:00
else
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < ( counts . failed > 1 ? allPrefix : " " ) < < pluralise ( counts . failed , label ) < < " failed " ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
void ReportCounts ( const Totals & totals , const std : : string & allPrefix = " All " ) {
if ( totals . assertions . total ( ) = = 0 ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " No tests ran " ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
else if ( totals . assertions . failed ) {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultError ) ;
2012-08-23 21:08:50 +02:00
ReportCounts ( " test case " , totals . testCases , allPrefix ) ;
if ( totals . testCases . failed > 0 ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " ( " ;
2012-08-23 21:08:50 +02:00
ReportCounts ( " assertion " , totals . assertions , allPrefix ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " ) " ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
else {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultSuccess ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < allPrefix < < " tests passed ( "
2012-08-23 21:08:50 +02:00
< < pluralise ( totals . assertions . passed , " assertion " ) < < " in "
< < pluralise ( totals . testCases . passed , " test case " ) < < " ) " ;
2011-05-24 09:23:02 +02:00
}
}
2012-08-23 21:08:50 +02:00
private : // IReporter
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual bool shouldRedirectStdout ( ) const {
return false ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
virtual void StartTesting ( ) {
m_testingSpan = SpanInfo ( ) ;
}
2012-02-17 20:50:59 +01:00
2012-08-23 21:08:50 +02:00
virtual void Aborted ( ) {
m_aborted = true ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndTesting ( const Totals & totals ) {
// Output the overall test results even if "Started Testing" was not emitted
if ( m_aborted ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n [Testing aborted. " ;
2012-08-23 21:08:50 +02:00
ReportCounts ( totals , " The first " ) ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
else {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n [Testing completed. " ;
2012-08-23 21:08:50 +02:00
ReportCounts ( totals ) ;
2012-08-09 08:47:30 +02:00
}
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " ] \n " < < std : : endl ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
virtual void StartGroup ( const std : : string & groupName ) {
m_groupSpan = groupName ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndGroup ( const std : : string & groupName , const Totals & totals ) {
if ( m_groupSpan . emitted & & ! groupName . empty ( ) ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " [End of group: ' " < < groupName < < " '. " ;
2012-08-23 21:08:50 +02:00
ReportCounts ( totals ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " ] \n " < < std : : endl ;
2012-08-23 21:08:50 +02:00
m_groupSpan = SpanInfo ( ) ;
}
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartTestCase ( const TestCaseInfo & testInfo ) {
2012-12-02 00:57:18 +01:00
m_testSpan = testInfo . name ;
2012-08-23 21:08:50 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartSection ( const std : : string & sectionName , const std : : string & ) {
m_sectionSpans . push_back ( SpanInfo ( sectionName ) ) ;
}
2011-05-24 09:23:02 +02:00
2012-08-31 09:10:36 +02:00
virtual void NoAssertionsInSection ( const std : : string & sectionName ) {
2012-09-07 18:52:35 +02:00
startSpansLazily ( ) ;
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultError ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n No assertions in section, ' " < < sectionName < < " ' \n " < < std : : endl ;
2012-08-31 09:10:36 +02:00
}
virtual void NoAssertionsInTestCase ( const std : : string & testName ) {
2012-09-07 18:52:35 +02:00
startSpansLazily ( ) ;
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultError ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n No assertions in test case, ' " < < testName < < " ' \n " < < std : : endl ;
2012-08-31 09:10:36 +02:00
}
2012-08-23 21:08:50 +02:00
virtual void EndSection ( const std : : string & sectionName , const Counts & assertions ) {
2012-08-31 09:10:36 +02:00
2012-08-23 21:08:50 +02:00
SpanInfo & sectionSpan = m_sectionSpans . back ( ) ;
if ( sectionSpan . emitted & & ! sectionSpan . name . empty ( ) ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " [End of section: ' " < < sectionName < < " ' " ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
if ( assertions . failed ) {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultError ) ;
2012-08-23 21:08:50 +02:00
ReportCounts ( " assertion " , assertions ) ;
}
else {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultSuccess ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < ( assertions . passed > 1 ? " All " : " " )
2012-08-23 21:08:50 +02:00
< < pluralise ( assertions . passed , " assertion " ) < < " passed " ;
}
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " ] \n " < < std : : endl ;
2012-08-23 21:08:50 +02:00
}
m_sectionSpans . pop_back ( ) ;
}
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
virtual void Result ( const AssertionResult & assertionResult ) {
2013-06-07 20:07:50 +02:00
if ( ! m_config . fullConfig ( ) - > includeSuccessfulResults ( ) & & assertionResult . getResultType ( ) = = ResultWas : : Ok )
2012-08-23 21:08:50 +02:00
return ;
2012-08-09 08:47:30 +02:00
2012-09-07 18:52:35 +02:00
startSpansLazily ( ) ;
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
if ( ! assertionResult . getSourceInfo ( ) . empty ( ) ) {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : FileName ) ;
2013-01-16 10:44:43 +01:00
m_config . stream ( ) < < assertionResult . getSourceInfo ( ) < < " : " ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
if ( assertionResult . hasExpression ( ) ) {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : OriginalExpression ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < assertionResult . getExpression ( ) ;
2012-11-13 23:04:29 +01:00
if ( assertionResult . succeeded ( ) ) {
2013-04-05 21:58:01 +02:00
Colour successColour ( Colour : : Success ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " succeeded " ;
2012-08-23 21:08:50 +02:00
}
else {
2013-04-05 21:58:01 +02:00
Colour errorColour ( Colour : : Error ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " failed " ;
2012-11-13 23:04:29 +01:00
if ( assertionResult . isOk ( ) ) {
2013-04-05 21:58:01 +02:00
Colour okAnywayColour ( Colour : : Success ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " - but was ok " ;
2012-11-13 23:04:29 +01:00
}
2012-08-23 21:08:50 +02:00
}
}
2012-11-01 09:27:09 +01:00
switch ( assertionResult . getResultType ( ) ) {
2012-08-23 21:08:50 +02:00
case ResultWas : : ThrewException :
2012-11-01 09:27:09 +01:00
{
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : Error ) ;
2012-11-01 09:27:09 +01:00
if ( assertionResult . hasExpression ( ) )
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " with unexpected " ;
2012-11-01 09:27:09 +01:00
else
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " Unexpected " ;
m_config . stream ( ) < < " exception with message: ' " < < assertionResult . getMessage ( ) < < " ' " ;
2012-11-01 09:27:09 +01:00
}
2012-08-23 21:08:50 +02:00
break ;
case ResultWas : : DidntThrowException :
2012-11-01 09:27:09 +01:00
{
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : Error ) ;
2012-11-01 09:27:09 +01:00
if ( assertionResult . hasExpression ( ) )
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " because no exception was thrown where one was expected " ;
2012-11-01 09:27:09 +01:00
else
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " No exception thrown where one was expected " ;
2012-11-01 09:27:09 +01:00
}
2012-08-23 21:08:50 +02:00
break ;
case ResultWas : : Info :
2012-11-01 09:27:09 +01:00
{
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ReconstructedExpression ) ;
2012-11-01 09:27:09 +01:00
streamVariableLengthText ( " info " , assertionResult . getMessage ( ) ) ;
}
2012-08-23 21:08:50 +02:00
break ;
case ResultWas : : Warning :
2012-11-01 09:27:09 +01:00
{
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ReconstructedExpression ) ;
2012-11-01 09:27:09 +01:00
streamVariableLengthText ( " warning " , assertionResult . getMessage ( ) ) ;
}
2012-08-23 21:08:50 +02:00
break ;
case ResultWas : : ExplicitFailure :
2012-11-01 09:27:09 +01:00
{
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : Error ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " failed with message: ' " < < assertionResult . getMessage ( ) < < " ' " ;
2012-11-01 09:27:09 +01:00
}
2012-08-23 21:08:50 +02:00
break ;
case ResultWas : : Unknown : // These cases are here to prevent compiler warnings
case ResultWas : : Ok :
case ResultWas : : FailureBit :
case ResultWas : : ExpressionFailed :
case ResultWas : : Exception :
2012-11-01 09:27:09 +01:00
if ( ! assertionResult . hasExpression ( ) ) {
2012-11-13 23:04:29 +01:00
if ( assertionResult . succeeded ( ) ) {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : Success ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " succeeded " ;
2012-08-23 21:08:50 +02:00
}
else {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : Error ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " failed " ;
2012-11-13 23:04:29 +01:00
if ( assertionResult . isOk ( ) ) {
2013-04-05 21:58:01 +02:00
Colour okAnywayColour ( Colour : : Success ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " - but was ok " ;
2012-11-13 23:04:29 +01:00
}
2012-08-23 21:08:50 +02:00
}
}
2012-11-21 19:06:13 +01:00
if ( assertionResult . hasMessage ( ) ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n " ;
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ReconstructedExpression ) ;
2012-11-21 19:06:13 +01:00
streamVariableLengthText ( " with message " , assertionResult . getMessage ( ) ) ;
}
2012-08-23 21:08:50 +02:00
break ;
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
if ( assertionResult . hasExpandedExpression ( ) ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " for: " ;
2012-11-01 09:27:09 +01:00
if ( assertionResult . getExpandedExpression ( ) . size ( ) > 40 ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n " ;
2012-11-01 09:27:09 +01:00
if ( assertionResult . getExpandedExpression ( ) . size ( ) < 70 )
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \t " ;
2012-11-01 09:27:09 +01:00
}
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ReconstructedExpression ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < assertionResult . getExpandedExpression ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < std : : endl ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndTestCase ( const TestCaseInfo & testInfo ,
const Totals & totals ,
const std : : string & stdOut ,
const std : : string & stdErr ) {
if ( ! stdOut . empty ( ) ) {
2012-09-07 18:52:35 +02:00
startSpansLazily ( ) ;
2012-08-23 21:08:50 +02:00
streamVariableLengthText ( " stdout " , stdOut ) ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
if ( ! stdErr . empty ( ) ) {
2012-09-07 18:52:35 +02:00
startSpansLazily ( ) ;
2012-08-23 21:08:50 +02:00
streamVariableLengthText ( " stderr " , stdErr ) ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
if ( m_testSpan . emitted ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " [Finished: ' " < < testInfo . name < < " ' " ;
2012-08-23 21:08:50 +02:00
ReportCounts ( totals ) ;
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " ] " < < std : : endl ;
2012-08-23 21:08:50 +02:00
}
2012-06-01 20:40:27 +02:00
}
2012-08-23 21:08:50 +02:00
private : // helpers
2012-08-09 08:47:30 +02:00
2012-09-07 18:52:35 +02:00
void startSpansLazily ( ) {
2012-08-23 21:08:50 +02:00
if ( ! m_testingSpan . emitted ) {
2013-06-07 20:07:50 +02:00
if ( m_config . fullConfig ( ) - > name ( ) . empty ( ) )
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " [Started testing] " < < std : : endl ;
2012-08-09 08:47:30 +02:00
else
2013-06-07 20:07:50 +02:00
m_config . stream ( ) < < " [Started testing: " < < m_config . fullConfig ( ) - > name ( ) < < " ] " < < std : : endl ;
2012-08-23 21:08:50 +02:00
m_testingSpan . emitted = true ;
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
if ( ! m_groupSpan . emitted & & ! m_groupSpan . name . empty ( ) ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " [Started group: ' " < < m_groupSpan . name < < " '] " < < std : : endl ;
2012-08-23 21:08:50 +02:00
m_groupSpan . emitted = true ;
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
if ( ! m_testSpan . emitted ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < std : : endl < < " [Running: " < < m_testSpan . name < < " ] " < < std : : endl ;
2012-08-23 21:08:50 +02:00
m_testSpan . emitted = true ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
if ( ! m_sectionSpans . empty ( ) ) {
SpanInfo & sectionSpan = m_sectionSpans . back ( ) ;
if ( ! sectionSpan . emitted & & ! sectionSpan . name . empty ( ) ) {
if ( m_firstSectionInTestCase ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n " ;
2012-08-23 21:08:50 +02:00
m_firstSectionInTestCase = false ;
}
std : : vector < SpanInfo > : : iterator it = m_sectionSpans . begin ( ) ;
std : : vector < SpanInfo > : : iterator itEnd = m_sectionSpans . end ( ) ;
for ( ; it ! = itEnd ; + + it ) {
SpanInfo & prevSpan = * it ;
if ( ! prevSpan . emitted & & ! prevSpan . name . empty ( ) ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " [Started section: ' " < < prevSpan . name < < " '] " < < std : : endl ;
2012-08-23 21:08:50 +02:00
prevSpan . emitted = true ;
}
}
}
}
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
void streamVariableLengthText ( const std : : string & prefix , const std : : string & text ) {
std : : string trimmed = trim ( text ) ;
if ( trimmed . find_first_of ( " \r \n " ) = = std : : string : : npos ) {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " [ " < < prefix < < " : " < < trimmed < < " ] " ;
2012-08-23 21:08:50 +02:00
}
else {
2012-12-02 00:57:18 +01:00
m_config . stream ( ) < < " \n [ " < < prefix < < " ] >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> \n " < < trimmed
2012-08-23 21:08:50 +02:00
< < " \n [end of " < < prefix < < " ] <<<<<<<<<<<<<<<<<<<<<<<< \n " ;
}
}
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
private :
ReporterConfig m_config ;
bool m_firstSectionInTestCase ;
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
SpanInfo m_testingSpan ;
SpanInfo m_groupSpan ;
SpanInfo m_testSpan ;
std : : vector < SpanInfo > m_sectionSpans ;
bool m_aborted ;
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
// #included from: ../reporters/catch_reporter_xml.hpp
# define TWOBLUECUBES_CATCH_REPORTER_XML_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
// #included from: ../internal/catch_xmlwriter.hpp
# define TWOBLUECUBES_CATCH_XMLWRITER_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
# include <sstream>
2012-09-29 21:54:03 +02:00
# include <iostream>
2012-08-23 21:08:50 +02:00
# include <string>
# include <vector>
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
namespace Catch {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
class XmlWriter {
public :
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
class ScopedElement {
public :
ScopedElement ( XmlWriter * writer )
: m_writer ( writer )
{ }
2013-04-24 20:10:02 +02:00
ScopedElement ( ScopedElement const & other )
2012-08-23 21:08:50 +02:00
: m_writer ( other . m_writer ) {
other . m_writer = NULL ;
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
~ ScopedElement ( ) {
if ( m_writer )
m_writer - > endElement ( ) ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
ScopedElement & writeText ( std : : string const & text , bool indent = true ) {
2012-12-02 00:57:18 +01:00
m_writer - > writeText ( text , indent ) ;
2012-08-23 21:08:50 +02:00
return * this ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
template < typename T >
2013-04-24 20:10:02 +02:00
ScopedElement & writeAttribute ( std : : string const & name , T const & attribute ) {
2012-08-23 21:08:50 +02:00
m_writer - > writeAttribute ( name , attribute ) ;
return * this ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
private :
mutable XmlWriter * m_writer ;
} ;
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
XmlWriter ( )
: m_tagIsOpen ( false ) ,
m_needsNewline ( false ) ,
m_os ( & std : : cout )
{ }
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
XmlWriter ( std : : ostream & os )
: m_tagIsOpen ( false ) ,
m_needsNewline ( false ) ,
m_os ( & os )
{ }
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
~ XmlWriter ( ) {
while ( ! m_tags . empty ( ) )
endElement ( ) ;
}
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
XmlWriter & operator = ( XmlWriter const & other ) {
2012-08-23 21:08:50 +02:00
XmlWriter temp ( other ) ;
swap ( temp ) ;
return * this ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
void swap ( XmlWriter & other ) {
std : : swap ( m_tagIsOpen , other . m_tagIsOpen ) ;
std : : swap ( m_needsNewline , other . m_needsNewline ) ;
std : : swap ( m_tags , other . m_tags ) ;
std : : swap ( m_indent , other . m_indent ) ;
std : : swap ( m_os , other . m_os ) ;
}
2012-08-09 08:47:30 +02:00
2013-04-24 20:10:02 +02:00
XmlWriter & startElement ( std : : string const & name ) {
2012-08-23 21:08:50 +02:00
ensureTagClosed ( ) ;
newlineIfNecessary ( ) ;
stream ( ) < < m_indent < < " < " < < name ;
m_tags . push_back ( name ) ;
m_indent + = " " ;
m_tagIsOpen = true ;
return * this ;
2012-02-17 20:50:59 +01:00
}
2013-04-24 20:10:02 +02:00
ScopedElement scopedElement ( std : : string const & name ) {
2012-08-23 21:08:50 +02:00
ScopedElement scoped ( this ) ;
startElement ( name ) ;
return scoped ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
XmlWriter & endElement ( ) {
newlineIfNecessary ( ) ;
m_indent = m_indent . substr ( 0 , m_indent . size ( ) - 2 ) ;
if ( m_tagIsOpen ) {
stream ( ) < < " /> \n " ;
m_tagIsOpen = false ;
}
else {
stream ( ) < < m_indent < < " </ " < < m_tags . back ( ) < < " > \n " ;
}
m_tags . pop_back ( ) ;
return * this ;
2011-05-24 09:23:02 +02:00
}
2013-04-24 20:10:02 +02:00
XmlWriter & writeAttribute ( std : : string const & name , std : : string const & attribute ) {
2012-08-23 21:08:50 +02:00
if ( ! name . empty ( ) & & ! attribute . empty ( ) ) {
stream ( ) < < " " < < name < < " = \" " ;
writeEncodedText ( attribute ) ;
stream ( ) < < " \" " ;
}
return * this ;
2011-05-24 09:23:02 +02:00
}
2013-04-24 20:10:02 +02:00
XmlWriter & writeAttribute ( std : : string const & name , bool attribute ) {
2012-08-23 21:08:50 +02:00
stream ( ) < < " " < < name < < " = \" " < < ( attribute ? " true " : " false " ) < < " \" " ;
return * this ;
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
template < typename T >
2013-04-24 20:10:02 +02:00
XmlWriter & writeAttribute ( std : : string const & name , T const & attribute ) {
2012-08-23 21:08:50 +02:00
if ( ! name . empty ( ) )
stream ( ) < < " " < < name < < " = \" " < < attribute < < " \" " ;
return * this ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
XmlWriter & writeText ( std : : string const & text , bool indent = true ) {
2012-08-23 21:08:50 +02:00
if ( ! text . empty ( ) ) {
bool tagWasOpen = m_tagIsOpen ;
ensureTagClosed ( ) ;
2012-12-02 00:57:18 +01:00
if ( tagWasOpen & & indent )
2012-08-23 21:08:50 +02:00
stream ( ) < < m_indent ;
writeEncodedText ( text ) ;
m_needsNewline = true ;
}
return * this ;
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
XmlWriter & writeComment ( std : : string const & text ) {
2012-08-23 21:08:50 +02:00
ensureTagClosed ( ) ;
stream ( ) < < m_indent < < " <!-- " < < text < < " --> " ;
m_needsNewline = true ;
return * this ;
}
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
XmlWriter & writeBlankLine ( ) {
ensureTagClosed ( ) ;
stream ( ) < < " \n " ;
return * this ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
private :
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
std : : ostream & stream ( ) {
return * m_os ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
void ensureTagClosed ( ) {
if ( m_tagIsOpen ) {
stream ( ) < < " > \n " ;
m_tagIsOpen = false ;
}
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
void newlineIfNecessary ( ) {
if ( m_needsNewline ) {
stream ( ) < < " \n " ;
m_needsNewline = false ;
}
}
2011-05-24 09:23:02 +02:00
2013-04-24 20:10:02 +02:00
void writeEncodedText ( std : : string const & text ) {
2012-08-23 21:08:50 +02:00
static const char * charsToEncode = " <& \" " ;
std : : string mtext = text ;
std : : string : : size_type pos = mtext . find_first_of ( charsToEncode ) ;
while ( pos ! = std : : string : : npos ) {
stream ( ) < < mtext . substr ( 0 , pos ) ;
2012-06-01 20:40:27 +02:00
2012-08-23 21:08:50 +02:00
switch ( mtext [ pos ] ) {
case ' < ' :
stream ( ) < < " < " ;
break ;
case ' & ' :
stream ( ) < < " & " ;
break ;
case ' \" ' :
stream ( ) < < " " " ;
break ;
}
mtext = mtext . substr ( pos + 1 ) ;
pos = mtext . find_first_of ( charsToEncode ) ;
}
stream ( ) < < mtext ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
bool m_tagIsOpen ;
bool m_needsNewline ;
std : : vector < std : : string > m_tags ;
std : : string m_indent ;
std : : ostream * m_os ;
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
namespace Catch {
2012-08-23 21:08:50 +02:00
class XmlReporter : public SharedImpl < IReporter > {
public :
2013-06-07 20:07:50 +02:00
XmlReporter ( ReporterConfig const & config ) : m_config ( config ) { }
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
static std : : string getDescription ( ) {
return " Reports test results as an XML document " ;
}
virtual ~ XmlReporter ( ) ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
private : // IReporter
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual bool shouldRedirectStdout ( ) const {
2012-08-09 08:47:30 +02:00
return true ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
virtual void StartTesting ( ) {
2012-12-02 00:57:18 +01:00
m_xml = XmlWriter ( m_config . stream ( ) ) ;
2012-08-23 21:08:50 +02:00
m_xml . startElement ( " Catch " ) ;
2013-06-07 20:07:50 +02:00
if ( ! m_config . fullConfig ( ) - > name ( ) . empty ( ) )
m_xml . writeAttribute ( " name " , m_config . fullConfig ( ) - > name ( ) ) ;
2012-08-09 08:47:30 +02:00
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndTesting ( const Totals & totals ) {
m_xml . scopedElement ( " OverallResults " )
. writeAttribute ( " successes " , totals . assertions . passed )
. writeAttribute ( " failures " , totals . assertions . failed ) ;
m_xml . endElement ( ) ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartGroup ( const std : : string & groupName ) {
m_xml . startElement ( " Group " )
. writeAttribute ( " name " , groupName ) ;
2012-08-09 08:47:30 +02:00
}
2012-08-23 21:08:50 +02:00
virtual void EndGroup ( const std : : string & , const Totals & totals ) {
m_xml . scopedElement ( " OverallResults " )
. writeAttribute ( " successes " , totals . assertions . passed )
. writeAttribute ( " failures " , totals . assertions . failed ) ;
m_xml . endElement ( ) ;
2011-05-24 09:23:02 +02:00
}
2012-08-23 21:08:50 +02:00
virtual void StartSection ( const std : : string & sectionName , const std : : string & description ) {
m_xml . startElement ( " Section " )
. writeAttribute ( " name " , sectionName )
. writeAttribute ( " description " , description ) ;
2011-05-24 09:23:02 +02:00
}
2012-08-31 09:10:36 +02:00
virtual void NoAssertionsInSection ( const std : : string & ) { }
virtual void NoAssertionsInTestCase ( const std : : string & ) { }
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndSection ( const std : : string & /*sectionName*/ , const Counts & assertions ) {
m_xml . scopedElement ( " OverallResults " )
. writeAttribute ( " successes " , assertions . passed )
. writeAttribute ( " failures " , assertions . failed ) ;
m_xml . endElement ( ) ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartTestCase ( const Catch : : TestCaseInfo & testInfo ) {
2012-12-02 00:57:18 +01:00
m_xml . startElement ( " TestCase " ) . writeAttribute ( " name " , testInfo . name ) ;
2012-08-23 21:08:50 +02:00
m_currentTestSuccess = true ;
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
virtual void Result ( const Catch : : AssertionResult & assertionResult ) {
2013-06-07 20:07:50 +02:00
if ( ! m_config . fullConfig ( ) - > includeSuccessfulResults ( ) & & assertionResult . getResultType ( ) = = ResultWas : : Ok )
2012-08-23 21:08:50 +02:00
return ;
2011-05-24 09:23:02 +02:00
2012-11-01 09:27:09 +01:00
if ( assertionResult . hasExpression ( ) ) {
2012-08-23 21:08:50 +02:00
m_xml . startElement ( " Expression " )
2012-11-13 23:04:29 +01:00
. writeAttribute ( " success " , assertionResult . succeeded ( ) )
2012-11-01 09:27:09 +01:00
. writeAttribute ( " filename " , assertionResult . getSourceInfo ( ) . file )
. writeAttribute ( " line " , assertionResult . getSourceInfo ( ) . line ) ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
m_xml . scopedElement ( " Original " )
2012-11-01 09:27:09 +01:00
. writeText ( assertionResult . getExpression ( ) ) ;
2012-08-23 21:08:50 +02:00
m_xml . scopedElement ( " Expanded " )
2012-11-01 09:27:09 +01:00
. writeText ( assertionResult . getExpandedExpression ( ) ) ;
2012-11-13 23:04:29 +01:00
m_currentTestSuccess & = assertionResult . succeeded ( ) ;
2012-08-23 21:08:50 +02:00
}
2012-05-16 16:09:17 +02:00
2012-11-01 09:27:09 +01:00
switch ( assertionResult . getResultType ( ) ) {
2012-08-23 21:08:50 +02:00
case ResultWas : : ThrewException :
m_xml . scopedElement ( " Exception " )
2012-11-01 09:27:09 +01:00
. writeAttribute ( " filename " , assertionResult . getSourceInfo ( ) . file )
. writeAttribute ( " line " , assertionResult . getSourceInfo ( ) . line )
. writeText ( assertionResult . getMessage ( ) ) ;
2012-08-23 21:08:50 +02:00
m_currentTestSuccess = false ;
break ;
case ResultWas : : Info :
m_xml . scopedElement ( " Info " )
2012-11-01 09:27:09 +01:00
. writeText ( assertionResult . getMessage ( ) ) ;
2012-08-23 21:08:50 +02:00
break ;
case ResultWas : : Warning :
m_xml . scopedElement ( " Warning " )
2012-11-01 09:27:09 +01:00
. writeText ( assertionResult . getMessage ( ) ) ;
2012-08-23 21:08:50 +02:00
break ;
case ResultWas : : ExplicitFailure :
m_xml . scopedElement ( " Failure " )
2012-11-01 09:27:09 +01:00
. writeText ( assertionResult . getMessage ( ) ) ;
2012-08-23 21:08:50 +02:00
m_currentTestSuccess = false ;
break ;
case ResultWas : : Unknown :
case ResultWas : : Ok :
case ResultWas : : FailureBit :
case ResultWas : : ExpressionFailed :
case ResultWas : : Exception :
case ResultWas : : DidntThrowException :
break ;
}
2012-11-01 09:27:09 +01:00
if ( assertionResult . hasExpression ( ) )
2012-08-23 21:08:50 +02:00
m_xml . endElement ( ) ;
}
2012-05-31 20:40:26 +02:00
2012-08-23 21:08:50 +02:00
virtual void Aborted ( ) {
// !TBD
2012-08-09 08:47:30 +02:00
}
2012-05-16 16:09:17 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndTestCase ( const Catch : : TestCaseInfo & , const Totals & , const std : : string & , const std : : string & ) {
m_xml . scopedElement ( " OverallResult " ) . writeAttribute ( " success " , m_currentTestSuccess ) ;
m_xml . endElement ( ) ;
}
2012-07-20 20:07:42 +02:00
2012-08-23 21:08:50 +02:00
private :
ReporterConfig m_config ;
bool m_currentTestSuccess ;
XmlWriter m_xml ;
} ;
2012-07-20 20:07:42 +02:00
2012-08-23 21:08:50 +02:00
} // end namespace Catch
2012-07-17 09:04:19 +02:00
2012-08-23 21:08:50 +02:00
// #included from: ../reporters/catch_reporter_junit.hpp
# define TWOBLUECUBES_CATCH_REPORTER_JUNIT_HPP_INCLUDED
2012-07-28 21:37:07 +02:00
2012-12-02 01:05:51 +01:00
# include <assert.h>
2012-08-23 21:08:50 +02:00
namespace Catch {
2012-07-17 09:04:19 +02:00
2012-08-23 21:08:50 +02:00
class JunitReporter : public SharedImpl < IReporter > {
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
struct TestStats {
std : : string m_element ;
std : : string m_resultType ;
std : : string m_message ;
std : : string m_content ;
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
struct TestCaseStats {
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
TestCaseStats ( const std : : string & className , const std : : string & name )
: m_className ( className ) ,
m_name ( name )
{ }
2012-05-25 09:52:05 +02:00
2012-08-23 21:08:50 +02:00
double m_timeInSeconds ;
std : : string m_status ;
std : : string m_className ;
std : : string m_name ;
2012-11-21 19:06:13 +01:00
std : : string m_stdOut ;
std : : string m_stdErr ;
2012-08-23 21:08:50 +02:00
std : : vector < TestStats > m_testStats ;
2012-12-02 00:57:18 +01:00
std : : vector < TestCaseStats > m_sections ;
2012-08-23 21:08:50 +02:00
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
struct Stats {
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
Stats ( const std : : string & name = std : : string ( ) )
: m_testsCount ( 0 ) ,
m_failuresCount ( 0 ) ,
m_disabledCount ( 0 ) ,
m_errorsCount ( 0 ) ,
m_timeInSeconds ( 0 ) ,
m_name ( name )
{ }
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
std : : size_t m_testsCount ;
std : : size_t m_failuresCount ;
std : : size_t m_disabledCount ;
std : : size_t m_errorsCount ;
double m_timeInSeconds ;
std : : string m_name ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
std : : vector < TestCaseStats > m_testCaseStats ;
} ;
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
public :
2013-06-07 20:07:50 +02:00
JunitReporter ( ReporterConfig const & config )
2012-08-23 21:08:50 +02:00
: m_config ( config ) ,
m_testSuiteStats ( " AllTests " ) ,
m_currentStats ( & m_testSuiteStats )
{ }
virtual ~ JunitReporter ( ) ;
2012-05-31 20:40:26 +02:00
2012-08-23 21:08:50 +02:00
static std : : string getDescription ( ) {
return " Reports test results in an XML format that looks like Ant's junitreport target " ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
private : // IReporter
2012-07-05 19:37:58 +02:00
2012-08-23 21:08:50 +02:00
virtual bool shouldRedirectStdout ( ) const {
return true ;
}
2012-07-05 19:37:58 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartTesting ( ) { }
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartGroup ( const std : : string & groupName ) {
2012-11-16 09:47:03 +01:00
if ( groupName . empty ( ) )
2013-06-07 20:07:50 +02:00
m_statsForSuites . push_back ( Stats ( m_config . fullConfig ( ) - > name ( ) ) ) ;
2012-11-16 09:47:03 +01:00
else
m_statsForSuites . push_back ( Stats ( groupName ) ) ;
2012-08-23 21:08:50 +02:00
m_currentStats = & m_statsForSuites . back ( ) ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndGroup ( const std : : string & , const Totals & totals ) {
m_currentStats - > m_testsCount = totals . assertions . total ( ) ;
m_currentStats = & m_testSuiteStats ;
}
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartSection ( const std : : string & , const std : : string & ) { }
2012-08-09 08:47:30 +02:00
2012-08-31 09:10:36 +02:00
virtual void NoAssertionsInSection ( const std : : string & ) { }
virtual void NoAssertionsInTestCase ( const std : : string & ) { }
virtual void EndSection ( const std : : string & , const Counts & ) { }
2012-08-09 08:47:30 +02:00
2012-08-23 21:08:50 +02:00
virtual void StartTestCase ( const Catch : : TestCaseInfo & testInfo ) {
2012-12-02 00:57:18 +01:00
m_currentStats - > m_testCaseStats . push_back ( TestCaseStats ( testInfo . className , testInfo . name ) ) ;
m_currentTestCaseStats . push_back ( & m_currentStats - > m_testCaseStats . back ( ) ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
virtual void Result ( const Catch : : AssertionResult & assertionResult ) {
2013-06-07 20:07:50 +02:00
if ( assertionResult . getResultType ( ) ! = ResultWas : : Ok | | m_config . fullConfig ( ) - > includeSuccessfulResults ( ) ) {
2012-08-23 21:08:50 +02:00
TestCaseStats & testCaseStats = m_currentStats - > m_testCaseStats . back ( ) ;
TestStats stats ;
std : : ostringstream oss ;
2012-11-01 09:27:09 +01:00
if ( ! assertionResult . getMessage ( ) . empty ( ) )
oss < < assertionResult . getMessage ( ) < < " at " ;
oss < < assertionResult . getSourceInfo ( ) ;
2012-08-23 21:08:50 +02:00
stats . m_content = oss . str ( ) ;
2012-11-01 09:27:09 +01:00
stats . m_message = assertionResult . getExpandedExpression ( ) ;
stats . m_resultType = assertionResult . getTestMacroName ( ) ;
2012-08-09 08:47:30 +02:00
2012-11-01 09:27:09 +01:00
switch ( assertionResult . getResultType ( ) ) {
2012-08-23 21:08:50 +02:00
case ResultWas : : ThrewException :
stats . m_element = " error " ;
m_currentStats - > m_errorsCount + + ;
break ;
case ResultWas : : Info :
stats . m_element = " info " ; // !TBD ?
break ;
case ResultWas : : Warning :
stats . m_element = " warning " ; // !TBD ?
break ;
case ResultWas : : ExplicitFailure :
stats . m_element = " failure " ;
m_currentStats - > m_failuresCount + + ;
break ;
case ResultWas : : ExpressionFailed :
stats . m_element = " failure " ;
m_currentStats - > m_failuresCount + + ;
break ;
case ResultWas : : Ok :
stats . m_element = " success " ;
break ;
2013-04-12 11:43:06 +02:00
case ResultWas : : DidntThrowException :
stats . m_element = " failure " ;
m_currentStats - > m_failuresCount + + ;
break ;
2012-08-23 21:08:50 +02:00
case ResultWas : : Unknown :
case ResultWas : : FailureBit :
case ResultWas : : Exception :
2013-04-12 11:43:06 +02:00
stats . m_element = " * internal error * " ;
2012-08-23 21:08:50 +02:00
break ;
}
testCaseStats . m_testStats . push_back ( stats ) ;
}
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndTestCase ( const Catch : : TestCaseInfo & , const Totals & , const std : : string & stdOut , const std : : string & stdErr ) {
2012-12-02 00:57:18 +01:00
m_currentTestCaseStats . pop_back ( ) ;
assert ( m_currentTestCaseStats . empty ( ) ) ;
2012-11-21 19:06:13 +01:00
TestCaseStats & testCaseStats = m_currentStats - > m_testCaseStats . back ( ) ;
testCaseStats . m_stdOut = stdOut ;
testCaseStats . m_stdErr = stdErr ;
2012-08-23 21:08:50 +02:00
if ( ! stdOut . empty ( ) )
m_stdOut < < stdOut < < " \n " ;
if ( ! stdErr . empty ( ) )
m_stdErr < < stdErr < < " \n " ;
}
2011-05-24 09:23:02 +02:00
2012-08-23 21:08:50 +02:00
virtual void Aborted ( ) {
// !TBD
}
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
virtual void EndTesting ( const Totals & ) {
2012-12-02 00:57:18 +01:00
XmlWriter xml ( m_config . stream ( ) ) ;
2012-08-16 19:48:50 +02:00
2012-12-02 00:57:18 +01:00
if ( m_statsForSuites . size ( ) > 0 )
xml . startElement ( " testsuites " ) ;
2012-08-16 19:48:50 +02:00
2012-12-02 00:57:18 +01:00
std : : vector < Stats > : : const_iterator it = m_statsForSuites . begin ( ) ;
std : : vector < Stats > : : const_iterator itEnd = m_statsForSuites . end ( ) ;
2012-08-16 19:48:50 +02:00
2012-12-02 00:57:18 +01:00
for ( ; it ! = itEnd ; + + it ) {
XmlWriter : : ScopedElement e = xml . scopedElement ( " testsuite " ) ;
xml . writeAttribute ( " name " , it - > m_name ) ;
xml . writeAttribute ( " errors " , it - > m_errorsCount ) ;
xml . writeAttribute ( " failures " , it - > m_failuresCount ) ;
xml . writeAttribute ( " tests " , it - > m_testsCount ) ;
xml . writeAttribute ( " hostname " , " tbd " ) ;
xml . writeAttribute ( " time " , " tbd " ) ;
xml . writeAttribute ( " timestamp " , " tbd " ) ;
2012-08-23 20:48:57 +02:00
2012-12-02 00:57:18 +01:00
OutputTestCases ( xml , * it ) ;
2012-08-23 21:08:50 +02:00
}
2012-12-02 00:57:18 +01:00
xml . scopedElement ( " system-out " ) . writeText ( trim ( m_stdOut . str ( ) ) , false ) ;
xml . scopedElement ( " system-err " ) . writeText ( trim ( m_stdErr . str ( ) ) , false ) ;
2012-08-23 21:08:50 +02:00
}
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
void OutputTestCases ( XmlWriter & xml , const Stats & stats ) {
std : : vector < TestCaseStats > : : const_iterator it = stats . m_testCaseStats . begin ( ) ;
std : : vector < TestCaseStats > : : const_iterator itEnd = stats . m_testCaseStats . end ( ) ;
for ( ; it ! = itEnd ; + + it ) {
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
XmlWriter : : ScopedElement e = xml . scopedElement ( " testcase " ) ;
xml . writeAttribute ( " classname " , it - > m_className ) ;
xml . writeAttribute ( " name " , it - > m_name ) ;
xml . writeAttribute ( " time " , " tbd " ) ;
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
OutputTestResult ( xml , * it ) ;
2012-11-21 19:06:13 +01:00
std : : string stdOut = trim ( it - > m_stdOut ) ;
if ( ! stdOut . empty ( ) )
2012-12-02 00:57:18 +01:00
xml . scopedElement ( " system-out " ) . writeText ( stdOut , false ) ;
2012-11-21 19:06:13 +01:00
std : : string stdErr = trim ( it - > m_stdErr ) ;
if ( ! stdErr . empty ( ) )
2012-12-02 00:57:18 +01:00
xml . scopedElement ( " system-err " ) . writeText ( stdErr , false ) ;
2012-08-23 21:08:50 +02:00
}
}
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
void OutputTestResult ( XmlWriter & xml , const TestCaseStats & stats ) {
std : : vector < TestStats > : : const_iterator it = stats . m_testStats . begin ( ) ;
std : : vector < TestStats > : : const_iterator itEnd = stats . m_testStats . end ( ) ;
for ( ; it ! = itEnd ; + + it ) {
if ( it - > m_element ! = " success " ) {
XmlWriter : : ScopedElement e = xml . scopedElement ( it - > m_element ) ;
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
xml . writeAttribute ( " message " , it - > m_message ) ;
xml . writeAttribute ( " type " , it - > m_resultType ) ;
if ( ! it - > m_content . empty ( ) )
xml . writeText ( it - > m_content ) ;
}
}
}
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
private :
ReporterConfig m_config ;
2012-08-16 19:48:50 +02:00
2012-08-23 21:08:50 +02:00
Stats m_testSuiteStats ;
Stats * m_currentStats ;
std : : vector < Stats > m_statsForSuites ;
2012-12-02 00:57:18 +01:00
std : : vector < const TestCaseStats * > m_currentTestCaseStats ;
2012-08-23 21:08:50 +02:00
std : : ostringstream m_stdOut ;
std : : ostringstream m_stdErr ;
} ;
} // end namespace Catch
2012-08-16 19:48:50 +02:00
2012-12-06 09:44:51 +01:00
// #included from: ../reporters/catch_reporter_console.hpp
# define TWOBLUECUBES_CATCH_REPORTER_CONSOLE_HPP_INCLUDED
namespace Catch {
2013-01-16 10:44:43 +01:00
struct ConsoleReporter : StreamingReporterBase {
2012-12-06 09:44:51 +01:00
ConsoleReporter ( ReporterConfig const & _config )
2013-01-16 10:44:43 +01:00
: StreamingReporterBase ( _config ) ,
2013-03-08 10:30:25 +01:00
m_headerPrinted ( false ) ,
2012-12-14 08:50:08 +01:00
m_atLeastOneTestCasePrinted ( false )
2012-12-06 09:44:51 +01:00
{ }
virtual ~ ConsoleReporter ( ) ;
static std : : string getDescription ( ) {
return " Reports test results as plain lines of text " ;
}
virtual ReporterPreferences getPreferences ( ) const {
ReporterPreferences prefs ;
prefs . shouldRedirectStdOut = false ;
return prefs ;
}
2013-01-16 10:44:43 +01:00
2013-03-13 13:19:30 +01:00
virtual void noMatchingTestCases ( std : : string const & spec ) {
stream < < " No test cases matched ' " < < spec < < " ' " < < std : : endl ;
}
2013-01-16 10:44:43 +01:00
virtual void assertionStarting ( AssertionInfo const & ) {
2012-12-06 09:44:51 +01:00
}
2013-01-16 10:44:43 +01:00
2013-06-28 18:09:57 +02:00
virtual bool assertionEnded ( AssertionStats const & _assertionStats ) {
2013-01-16 10:44:43 +01:00
AssertionResult const & result = _assertionStats . assertionResult ;
// Drop out if result was successful and we're not printing those
2013-06-07 20:07:50 +02:00
if ( ! m_config - > includeSuccessfulResults ( ) & & result . isOk ( ) )
2013-06-28 18:09:57 +02:00
return false ;
2013-01-16 10:44:43 +01:00
lazyPrint ( ) ;
2013-01-26 21:18:30 +01:00
AssertionPrinter printer ( stream , _assertionStats ) ;
printer . print ( ) ;
stream < < std : : endl ;
2013-06-28 18:09:57 +02:00
return true ;
2013-01-26 21:18:30 +01:00
}
virtual void sectionStarting ( SectionInfo const & _sectionInfo ) {
2013-03-08 10:30:25 +01:00
m_headerPrinted = false ;
2013-01-26 21:18:30 +01:00
StreamingReporterBase : : sectionStarting ( _sectionInfo ) ;
}
virtual void sectionEnded ( SectionStats const & _sectionStats ) {
if ( _sectionStats . missingAssertions ) {
lazyPrint ( ) ;
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultError ) ;
2013-01-26 21:18:30 +01:00
stream < < " \n No assertions in section, ' " < < _sectionStats . sectionInfo . name < < " ' \n " < < std : : endl ;
2013-01-16 10:44:43 +01:00
}
2013-03-08 10:30:25 +01:00
m_headerPrinted = false ;
2013-01-26 21:18:30 +01:00
StreamingReporterBase : : sectionEnded ( _sectionStats ) ;
}
2013-01-17 13:07:34 +01:00
2013-01-26 21:18:30 +01:00
virtual void testCaseEnded ( TestCaseStats const & _testCaseStats ) {
if ( _testCaseStats . missingAssertions ) {
lazyPrint ( ) ;
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultError ) ;
2013-01-26 21:18:30 +01:00
stream < < " \n No assertions in test case, ' " < < _testCaseStats . testInfo . name < < " ' \n " < < std : : endl ;
}
StreamingReporterBase : : testCaseEnded ( _testCaseStats ) ;
2013-03-08 10:30:25 +01:00
m_headerPrinted = false ;
2013-01-26 21:18:30 +01:00
}
virtual void testGroupEnded ( TestGroupStats const & _testGroupStats ) {
if ( ! unusedGroupInfo ) {
printSummaryDivider ( ) ;
stream < < " Summary for group ' " < < _testGroupStats . groupInfo . name < < " ': \n " ;
printTotals ( _testGroupStats . totals ) ;
stream < < " \n " < < std : : endl ;
}
StreamingReporterBase : : testGroupEnded ( _testGroupStats ) ;
2012-12-06 09:44:51 +01:00
}
2013-01-26 21:18:30 +01:00
virtual void testRunEnded ( TestRunStats const & _testRunStats ) {
if ( m_atLeastOneTestCasePrinted )
printTotalsDivider ( ) ;
printTotals ( _testRunStats . totals ) ;
stream < < " \n " < < std : : endl ;
StreamingReporterBase : : testRunEnded ( _testRunStats ) ;
2013-01-17 13:07:34 +01:00
}
2012-12-10 09:54:57 +01:00
2013-01-26 21:18:30 +01:00
private :
class AssertionPrinter {
2013-03-25 10:25:31 +01:00
void operator = ( AssertionPrinter const & ) ;
2013-01-26 21:18:30 +01:00
public :
AssertionPrinter ( std : : ostream & _stream , AssertionStats const & _stats )
: stream ( _stream ) ,
stats ( _stats ) ,
result ( _stats . assertionResult ) ,
2013-04-05 21:58:01 +02:00
colour ( Colour : : None ) ,
2013-02-02 21:37:58 +01:00
message ( result . getMessage ( ) ) ,
messages ( _stats . infoMessages )
2013-01-16 10:44:43 +01:00
{
2013-01-26 21:18:30 +01:00
switch ( result . getResultType ( ) ) {
2013-01-16 10:44:43 +01:00
case ResultWas : : Ok :
2013-04-05 21:58:01 +02:00
colour = Colour : : Success ;
2013-01-17 13:07:34 +01:00
passOrFail = " PASSED " ;
2013-02-02 21:37:58 +01:00
//if( result.hasMessage() )
if ( _stats . infoMessages . size ( ) = = 1 )
2013-01-16 10:44:43 +01:00
messageLabel = " with message " ;
2013-02-02 21:37:58 +01:00
if ( _stats . infoMessages . size ( ) > 1 )
messageLabel = " with messages " ;
2013-01-16 10:44:43 +01:00
break ;
case ResultWas : : ExpressionFailed :
2013-01-26 21:18:30 +01:00
if ( result . isOk ( ) ) {
2013-04-05 21:58:01 +02:00
colour = Colour : : Success ;
2013-01-17 13:07:34 +01:00
passOrFail = " FAILED - but was ok " ;
2013-01-16 10:44:43 +01:00
}
else {
2013-04-05 21:58:01 +02:00
colour = Colour : : Error ;
2013-01-17 13:07:34 +01:00
passOrFail = " FAILED " ;
2013-01-16 10:44:43 +01:00
}
2013-02-02 21:37:58 +01:00
if ( _stats . infoMessages . size ( ) = = 1 )
2013-01-16 10:44:43 +01:00
messageLabel = " with message " ;
2013-02-02 21:37:58 +01:00
if ( _stats . infoMessages . size ( ) > 1 )
messageLabel = " with messages " ;
2013-01-16 10:44:43 +01:00
break ;
case ResultWas : : ThrewException :
2013-04-05 21:58:01 +02:00
colour = Colour : : Error ;
2013-01-17 13:07:34 +01:00
passOrFail = " FAILED " ;
2013-01-16 10:44:43 +01:00
messageLabel = " due to unexpected exception with message " ;
break ;
case ResultWas : : DidntThrowException :
2013-04-05 21:58:01 +02:00
colour = Colour : : Error ;
2013-01-17 13:07:34 +01:00
passOrFail = " FAILED " ;
2013-01-16 10:44:43 +01:00
messageLabel = " because no exception was thrown where one was expected " ;
break ;
case ResultWas : : Info :
messageLabel = " info " ;
break ;
case ResultWas : : Warning :
messageLabel = " warning " ;
break ;
case ResultWas : : ExplicitFailure :
2013-01-17 13:07:34 +01:00
passOrFail = " FAILED " ;
2013-04-05 21:58:01 +02:00
colour = Colour : : Error ;
2013-02-02 21:37:58 +01:00
if ( _stats . infoMessages . size ( ) = = 1 )
messageLabel = " explicitly with message " ;
if ( _stats . infoMessages . size ( ) > 1 )
messageLabel = " explicitly with messages " ;
2013-01-16 10:44:43 +01:00
break ;
// These cases are here to prevent compiler warnings
case ResultWas : : Unknown :
case ResultWas : : FailureBit :
2013-04-12 11:43:06 +02:00
case ResultWas : : Exception :
2013-01-16 10:44:43 +01:00
passOrFail = " ** internal error ** " ;
2013-04-05 21:58:01 +02:00
colour = Colour : : Error ;
2013-01-16 10:44:43 +01:00
break ;
}
}
2013-01-26 21:18:30 +01:00
void print ( ) const {
2013-03-04 12:19:15 +01:00
printSourceInfo ( ) ;
2013-01-26 21:18:30 +01:00
if ( stats . totals . assertions . total ( ) > 0 ) {
2013-03-04 12:19:15 +01:00
if ( result . isOk ( ) )
stream < < " \n " ;
2013-01-26 21:18:30 +01:00
printResultType ( ) ;
printOriginalExpression ( ) ;
printReconstructedExpression ( ) ;
}
2013-03-04 12:19:15 +01:00
else {
stream < < " \n " ;
}
2013-01-26 21:18:30 +01:00
printMessage ( ) ;
2012-12-10 09:54:57 +01:00
}
2013-01-26 21:18:30 +01:00
private :
void printResultType ( ) const {
if ( ! passOrFail . empty ( ) ) {
2013-04-05 21:58:01 +02:00
Colour colourGuard ( colour ) ;
2013-01-26 21:18:30 +01:00
stream < < passOrFail < < " : \n " ;
}
2013-01-16 10:44:43 +01:00
}
2013-01-26 21:18:30 +01:00
void printOriginalExpression ( ) const {
if ( result . hasExpression ( ) ) {
2013-04-05 21:58:01 +02:00
Colour colourGuard ( Colour : : OriginalExpression ) ;
2013-01-26 21:18:30 +01:00
stream < < " " ;
2013-06-07 20:07:50 +02:00
stream < < result . getExpressionInMacro ( ) ;
2013-01-26 21:18:30 +01:00
stream < < " \n " ;
}
2013-01-16 10:44:43 +01:00
}
2013-01-26 21:18:30 +01:00
void printReconstructedExpression ( ) const {
if ( result . hasExpandedExpression ( ) ) {
stream < < " with expansion: \n " ;
2013-04-05 21:58:01 +02:00
Colour colourGuard ( Colour : : ReconstructedExpression ) ;
2013-04-20 22:07:32 +02:00
stream < < Text ( result . getExpandedExpression ( ) , TextAttributes ( ) . setIndent ( 2 ) ) < < " \n " ;
2013-01-26 21:18:30 +01:00
}
2013-01-16 10:44:43 +01:00
}
2013-01-26 21:18:30 +01:00
void printMessage ( ) const {
if ( ! messageLabel . empty ( ) )
stream < < messageLabel < < " : " < < " \n " ;
2013-02-02 21:37:58 +01:00
for ( std : : vector < MessageInfo > : : const_iterator it = messages . begin ( ) , itEnd = messages . end ( ) ;
it ! = itEnd ;
+ + it ) {
2013-04-20 22:07:32 +02:00
stream < < Text ( it - > message , TextAttributes ( ) . setIndent ( 2 ) ) < < " \n " ;
2013-02-02 21:37:58 +01:00
}
2013-01-16 10:44:43 +01:00
}
2013-01-26 21:18:30 +01:00
void printSourceInfo ( ) const {
2013-04-05 21:58:01 +02:00
Colour colourGuard ( Colour : : FileName ) ;
2013-03-04 12:19:15 +01:00
stream < < result . getSourceInfo ( ) < < " : " ;
2013-01-16 10:44:43 +01:00
}
2013-01-26 21:18:30 +01:00
std : : ostream & stream ;
AssertionStats const & stats ;
AssertionResult const & result ;
2013-04-05 21:58:01 +02:00
Colour : : Code colour ;
2013-01-26 21:18:30 +01:00
std : : string passOrFail ;
std : : string messageLabel ;
std : : string message ;
2013-02-02 21:37:58 +01:00
std : : vector < MessageInfo > messages ;
2013-01-26 21:18:30 +01:00
} ;
2012-12-14 08:50:08 +01:00
2012-12-06 09:44:51 +01:00
void lazyPrint ( ) {
2013-01-16 10:44:43 +01:00
2012-12-06 09:44:51 +01:00
if ( testRunInfo )
lazyPrintRunInfo ( ) ;
if ( unusedGroupInfo )
lazyPrintGroupInfo ( ) ;
2013-03-08 10:30:25 +01:00
if ( ! m_headerPrinted ) {
printTestCaseAndSectionHeader ( ) ;
m_headerPrinted = true ;
}
2013-01-16 10:44:43 +01:00
m_atLeastOneTestCasePrinted = true ;
2012-12-06 09:44:51 +01:00
}
2013-01-16 10:44:43 +01:00
void lazyPrintRunInfo ( ) {
2013-03-16 21:21:51 +01:00
stream < < " \n " < < getTildes ( ) < < " \n " ;
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : SecondaryText ) ;
2013-03-16 21:21:51 +01:00
stream < < testRunInfo - > name
2013-06-07 20:07:50 +02:00
< < " is a Catch v " < < libraryVersion . majorVersion < < " . "
2013-01-16 10:44:43 +01:00
< < libraryVersion . minorVersion < < " b "
< < libraryVersion . buildNumber ;
if ( libraryVersion . branchName ! = " master " )
stream < < " ( " < < libraryVersion . branchName < < " ) " ;
stream < < " host application. \n "
< < " Run with -? for options \n \n " ;
2012-12-10 09:54:57 +01:00
2013-01-16 10:44:43 +01:00
testRunInfo . reset ( ) ;
}
void lazyPrintGroupInfo ( ) {
if ( ! unusedGroupInfo - > name . empty ( ) & & unusedGroupInfo - > groupsCounts > 1 ) {
2013-03-08 10:30:25 +01:00
printClosedHeader ( " Group: " + unusedGroupInfo - > name ) ;
2013-01-16 10:44:43 +01:00
unusedGroupInfo . reset ( ) ;
}
}
2013-03-08 10:30:25 +01:00
void printTestCaseAndSectionHeader ( ) {
2013-04-20 22:07:32 +02:00
printOpenHeader ( unusedTestCaseInfo - > name ) ;
2013-03-08 10:30:25 +01:00
if ( currentSectionInfo ) {
2013-04-05 21:58:01 +02:00
Colour colourGuard ( Colour : : Headers ) ;
2013-03-08 10:30:25 +01:00
std : : vector < ThreadedSectionInfo * > sections ;
for ( ThreadedSectionInfo * section = currentSectionInfo . get ( ) ;
section ;
section = section - > parent )
sections . push_back ( section ) ;
// Sections
if ( ! sections . empty ( ) ) {
typedef std : : vector < ThreadedSectionInfo * > : : const_reverse_iterator It ;
for ( It it = sections . rbegin ( ) , itEnd = sections . rend ( ) ; it ! = itEnd ; + + it )
2013-04-20 22:07:32 +02:00
printHeaderString ( ( * it ) - > name , 2 ) ;
2013-04-05 21:58:01 +02:00
2013-03-08 10:30:25 +01:00
}
2012-12-10 09:54:57 +01:00
}
2013-04-20 22:07:32 +02:00
SourceLineInfo lineInfo = currentSectionInfo
? currentSectionInfo - > lineInfo
: unusedTestCaseInfo - > lineInfo ;
if ( ! lineInfo . empty ( ) ) {
stream < < getDashes ( ) < < " \n " ;
Colour colourGuard ( Colour : : FileName ) ;
stream < < lineInfo < < " \n " ;
}
2013-03-08 10:30:25 +01:00
stream < < getDots ( ) < < " \n " < < std : : endl ;
2013-01-16 10:44:43 +01:00
}
2012-12-10 09:54:57 +01:00
2013-03-08 10:30:25 +01:00
void printClosedHeader ( std : : string const & _name ) {
printOpenHeader ( _name ) ;
stream < < getDots ( ) < < " \n " ;
}
2013-04-20 22:07:32 +02:00
void printOpenHeader ( std : : string const & _name ) {
2013-04-01 12:27:10 +02:00
stream < < getDashes ( ) < < " \n " ;
{
2013-04-05 21:58:01 +02:00
Colour colourGuard ( Colour : : Headers ) ;
2013-04-20 22:07:32 +02:00
printHeaderString ( _name ) ;
2013-04-01 12:27:10 +02:00
}
2012-12-10 09:54:57 +01:00
}
2013-04-05 21:58:01 +02:00
// if string has a : in first line will set indent to follow it on
// subsequent lines
2013-04-20 22:07:32 +02:00
void printHeaderString ( std : : string const & _string , std : : size_t indent = 0 ) {
2013-04-05 21:58:01 +02:00
std : : size_t i = _string . find ( " : " ) ;
if ( i ! = std : : string : : npos )
i + = 2 ;
else
i = 0 ;
2013-04-20 22:07:32 +02:00
stream < < Text ( _string , TextAttributes ( )
. setIndent ( indent + i )
. setInitialIndent ( indent ) ) < < " \n " ;
2013-04-05 21:58:01 +02:00
}
2013-01-16 10:44:43 +01:00
void printTotals ( const Totals & totals ) {
if ( totals . assertions . total ( ) = = 0 ) {
stream < < " No tests ran " ;
}
else if ( totals . assertions . failed ) {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultError ) ;
2013-01-16 10:44:43 +01:00
printCounts ( " test case " , totals . testCases ) ;
if ( totals . testCases . failed > 0 ) {
stream < < " ( " ;
printCounts ( " assertion " , totals . assertions ) ;
stream < < " ) " ;
}
2012-12-10 09:54:57 +01:00
}
else {
2013-04-05 21:58:01 +02:00
Colour colour ( Colour : : ResultSuccess ) ;
2013-01-16 10:44:43 +01:00
stream < < " All tests passed ( "
< < pluralise ( totals . assertions . passed , " assertion " ) < < " in "
< < pluralise ( totals . testCases . passed , " test case " ) < < " ) " ;
2012-12-10 09:54:57 +01:00
}
2012-12-06 09:44:51 +01:00
}
2013-01-16 10:44:43 +01:00
void printCounts ( std : : string const & label , Counts const & counts ) {
2012-12-11 10:03:14 +01:00
if ( counts . total ( ) = = 1 ) {
stream < < " 1 " < < label < < " - " ;
if ( counts . failed )
stream < < " failed " ;
else
stream < < " passed " ;
}
else {
stream < < counts . total ( ) < < " " < < label < < " s " ;
if ( counts . passed ) {
if ( counts . failed )
stream < < " - " < < counts . failed < < " failed " ;
else if ( counts . passed = = 2 )
stream < < " - both passed " ;
else
stream < < " - all passed " ;
}
else {
if ( counts . failed = = 2 )
stream < < " - both failed " ;
else
stream < < " - all failed " ;
}
}
2012-12-06 09:44:51 +01:00
}
2013-01-16 10:44:43 +01:00
void printTotalsDivider ( ) {
stream < < getDoubleDashes ( ) < < " \n " ;
2012-12-06 09:44:51 +01:00
}
2013-01-16 10:44:43 +01:00
void printSummaryDivider ( ) {
stream < < getDashes ( ) < < " \n " ;
2012-12-06 09:44:51 +01:00
}
2013-01-16 10:44:43 +01:00
static std : : string const & getDashes ( ) {
2013-01-26 21:18:30 +01:00
static const std : : string dashes ( CATCH_CONFIG_CONSOLE_WIDTH - 1 , ' - ' ) ;
2013-01-16 10:44:43 +01:00
return dashes ;
2012-12-06 09:44:51 +01:00
}
2013-01-18 09:09:28 +01:00
static std : : string const & getDots ( ) {
2013-01-26 21:18:30 +01:00
static const std : : string dots ( CATCH_CONFIG_CONSOLE_WIDTH - 1 , ' . ' ) ;
2013-01-18 09:09:28 +01:00
return dots ;
}
2013-01-16 10:44:43 +01:00
static std : : string const & getDoubleDashes ( ) {
2013-01-26 21:18:30 +01:00
static const std : : string doubleDashes ( CATCH_CONFIG_CONSOLE_WIDTH - 1 , ' = ' ) ;
2013-01-16 10:44:43 +01:00
return doubleDashes ;
2012-12-06 09:44:51 +01:00
}
2013-03-16 21:21:51 +01:00
static std : : string const & getTildes ( ) {
static const std : : string dots ( CATCH_CONFIG_CONSOLE_WIDTH - 1 , ' ~ ' ) ;
return dots ;
}
2012-12-06 09:44:51 +01:00
2012-12-14 08:50:08 +01:00
private :
2013-03-08 10:30:25 +01:00
bool m_headerPrinted ;
2012-12-14 08:50:08 +01:00
bool m_atLeastOneTestCasePrinted ;
2012-12-06 09:44:51 +01:00
} ;
INTERNAL_CATCH_REGISTER_REPORTER ( " console " , ConsoleReporter )
} // end namespace Catch
2012-08-13 08:46:10 +02:00
namespace Catch {
NonCopyable : : ~ NonCopyable ( ) { }
IShared : : ~ IShared ( ) { }
2013-07-02 09:49:29 +02:00
StreamBufBase : : ~ StreamBufBase ( ) throw ( ) { }
2012-08-13 08:46:10 +02:00
IContext : : ~ IContext ( ) { }
IResultCapture : : ~ IResultCapture ( ) { }
ITestCase : : ~ ITestCase ( ) { }
ITestCaseRegistry : : ~ ITestCaseRegistry ( ) { }
IRegistryHub : : ~ IRegistryHub ( ) { }
IMutableRegistryHub : : ~ IMutableRegistryHub ( ) { }
IExceptionTranslator : : ~ IExceptionTranslator ( ) { }
IExceptionTranslatorRegistry : : ~ IExceptionTranslatorRegistry ( ) { }
IReporter : : ~ IReporter ( ) { }
IReporterFactory : : ~ IReporterFactory ( ) { }
IReporterRegistry : : ~ IReporterRegistry ( ) { }
2012-12-02 00:57:18 +01:00
IStreamingReporter : : ~ IStreamingReporter ( ) { }
AssertionStats : : ~ AssertionStats ( ) { }
SectionStats : : ~ SectionStats ( ) { }
TestCaseStats : : ~ TestCaseStats ( ) { }
TestGroupStats : : ~ TestGroupStats ( ) { }
TestRunStats : : ~ TestRunStats ( ) { }
2012-12-06 09:44:51 +01:00
ThreadedSectionInfo : : ~ ThreadedSectionInfo ( ) { }
2013-01-16 10:44:43 +01:00
TestGroupNode : : ~ TestGroupNode ( ) { }
TestRunNode : : ~ TestRunNode ( ) { }
2012-12-02 00:57:18 +01:00
2012-08-13 08:46:10 +02:00
BasicReporter : : ~ BasicReporter ( ) { }
2013-01-16 10:44:43 +01:00
StreamingReporterBase : : ~ StreamingReporterBase ( ) { }
2012-12-06 09:44:51 +01:00
ConsoleReporter : : ~ ConsoleReporter ( ) { }
2012-08-13 08:46:10 +02:00
IRunner : : ~ IRunner ( ) { }
IMutableContext : : ~ IMutableContext ( ) { }
IConfig : : ~ IConfig ( ) { }
XmlReporter : : ~ XmlReporter ( ) { }
JunitReporter : : ~ JunitReporter ( ) { }
TestRegistry : : ~ TestRegistry ( ) { }
FreeFunctionTestCase : : ~ FreeFunctionTestCase ( ) { }
IGeneratorInfo : : ~ IGeneratorInfo ( ) { }
IGeneratorsForTest : : ~ IGeneratorsForTest ( ) { }
2012-09-24 09:30:13 +02:00
TagParser : : ~ TagParser ( ) { }
TagExtracter : : ~ TagExtracter ( ) { }
TagExpressionParser : : ~ TagExpressionParser ( ) { }
2012-08-13 08:46:10 +02:00
2012-10-12 08:58:17 +02:00
Matchers : : Impl : : StdString : : Equals : : ~ Equals ( ) { }
Matchers : : Impl : : StdString : : Contains : : ~ Contains ( ) { }
Matchers : : Impl : : StdString : : StartsWith : : ~ StartsWith ( ) { }
Matchers : : Impl : : StdString : : EndsWith : : ~ EndsWith ( ) { }
2012-08-13 08:46:10 +02:00
void Config : : dummy ( ) { }
2012-12-02 00:57:18 +01:00
INTERNAL_CATCH_REGISTER_LEGACY_REPORTER ( " basic " , BasicReporter )
INTERNAL_CATCH_REGISTER_LEGACY_REPORTER ( " xml " , XmlReporter )
INTERNAL_CATCH_REGISTER_LEGACY_REPORTER ( " junit " , JunitReporter )
2012-08-23 21:08:50 +02:00
2012-08-13 08:46:10 +02:00
}
2012-08-16 19:48:50 +02:00
# ifdef __clang__
2012-08-13 08:46:10 +02:00
# pragma clang diagnostic pop
2011-05-24 09:23:02 +02:00
# endif
2012-09-17 07:42:29 +02:00
2012-08-16 19:48:50 +02:00
# endif
2011-05-24 09:23:02 +02:00
# ifdef CATCH_CONFIG_MAIN
// #included from: internal/catch_default_main.hpp
2012-08-09 08:47:30 +02:00
# define TWOBLUECUBES_CATCH_DEFAULT_MAIN_HPP_INCLUDED
2011-05-24 09:23:02 +02:00
2012-05-11 09:17:16 +02:00
# ifndef __OBJC__
// Standard C/C++ main entry point
int main ( int argc , char * const argv [ ] ) {
2013-06-07 20:07:50 +02:00
return Catch : : Session ( ) . run ( argc , argv ) ;
2012-05-11 09:17:16 +02:00
}
# else // __OBJC__
2011-05-24 09:23:02 +02:00
2012-05-11 09:17:16 +02:00
// Objective-C entry point
int main ( int argc , char * const argv [ ] ) {
2012-03-17 19:20:06 +01:00
# if !CATCH_ARC_ENABLED
2011-05-24 09:23:02 +02:00
NSAutoreleasePool * pool = [ [ NSAutoreleasePool alloc ] init ] ;
2012-03-17 19:20:06 +01:00
# endif
2011-05-24 09:23:02 +02:00
Catch : : registerTestMethods ( ) ;
2013-06-07 20:07:50 +02:00
int result = Catch : : Session ( ) . run ( argc , ( char * const * ) argv ) ;
2011-05-24 09:23:02 +02:00
2012-03-17 19:20:06 +01:00
# if !CATCH_ARC_ENABLED
2011-05-24 09:23:02 +02:00
[ pool drain ] ;
2012-03-17 19:20:06 +01:00
# endif
2011-05-24 09:23:02 +02:00
2012-02-18 10:58:30 +01:00
return result ;
2011-05-24 09:23:02 +02:00
}
2012-05-11 09:17:16 +02:00
# endif // __OBJC__
2011-05-24 09:23:02 +02:00
# endif
//////
2012-07-20 19:43:48 +02:00
// If this config identifier is defined then all CATCH macros are prefixed with CATCH_
# ifdef CATCH_CONFIG_PREFIX_ALL
2012-11-13 23:04:29 +01:00
# define CATCH_REQUIRE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal, "CATCH_REQUIRE" )
# define CATCH_REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal | Catch::ResultDisposition::NegateResult, "CATCH_REQUIRE_FALSE" )
2012-07-20 19:43:48 +02:00
2012-11-13 23:04:29 +01:00
# define CATCH_REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, ..., Catch::ResultDisposition::Normal, "CATCH_REQUIRE_THROWS" )
# define CATCH_REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::Normal, "CATCH_REQUIRE_THROWS_AS" )
# define CATCH_REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::Normal, "CATCH_REQUIRE_NOTHROW" )
2012-07-20 19:43:48 +02:00
2012-11-13 23:04:29 +01:00
# define CATCH_CHECK( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK" )
# define CATCH_CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::NegateResult, "CATCH_CHECK_FALSE" )
# define CATCH_CHECKED_IF( expr ) INTERNAL_CATCH_IF( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECKED_IF" )
# define CATCH_CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECKED_ELSE" )
# define CATCH_CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, "CATCH_CHECK_NOFAIL" )
2012-07-20 19:43:48 +02:00
2012-11-13 23:04:29 +01:00
# define CATCH_CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, ..., Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK_THROWS" )
# define CATCH_CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK_THROWS_AS" )
# define CATCH_CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK_NOTHROW" )
2012-07-20 19:43:48 +02:00
2012-11-13 23:04:29 +01:00
# define CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::ContinueOnFailure, "CATCH_CHECK_THAT" )
# define CATCH_REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::Normal, "CATCH_REQUIRE_THAT" )
2012-07-20 19:43:48 +02:00
2013-02-02 21:37:58 +01:00
# define CATCH_INFO( msg ) INTERNAL_CATCH_INFO( msg, "CATCH_INFO" )
2012-11-13 23:04:29 +01:00
# define CATCH_WARN( msg ) INTERNAL_CATCH_MSG( msg, Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, "CATCH_WARN" )
# define CATCH_FAIL( msg ) INTERNAL_CATCH_MSG( msg, Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, "CATCH_FAIL" )
# define CATCH_SUCCEED( msg ) INTERNAL_CATCH_MSG( msg, Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, "CATCH_SUCCEED" )
2013-06-28 18:09:57 +02:00
# define CATCH_SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( msg, "CATCH_INFO" )
2013-02-02 21:37:58 +01:00
# define CATCH_CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CATCH_CAPTURE" )
2013-06-28 18:09:57 +02:00
# define CATCH_SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CATCH_CAPTURE" )
2012-07-20 19:43:48 +02:00
2013-03-16 21:21:51 +01:00
# ifdef CATCH_CONFIG_VARIADIC_MACROS
# define CATCH_TEST_CASE( ... ) INTERNAL_CATCH_TESTCASE( __VA_ARGS__ )
2013-06-07 20:07:50 +02:00
# define CATCH_TEST_CASE_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, __VA_ARGS__ )
2013-03-16 21:21:51 +01:00
# define CATCH_METHOD_AS_TEST_CASE( method, ... ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, __VA_ARGS__ )
# define CATCH_SECTION( ... ) INTERNAL_CATCH_SECTION( __VA_ARGS__ )
# else
# define CATCH_TEST_CASE( name, description ) INTERNAL_CATCH_TESTCASE( name, description )
2013-06-07 20:07:50 +02:00
# define CATCH_TEST_CASE_METHOD( className, name, description ) INTERNAL_CATCH_TEST_CASE_METHOD( className, name, description )
2013-03-16 21:21:51 +01:00
# define CATCH_METHOD_AS_TEST_CASE( method, name, description ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, name, description )
# define CATCH_SECTION( name, description ) INTERNAL_CATCH_SECTION( name, description )
# endif
# define CATCH_ANON_TEST_CASE() INTERNAL_CATCH_TESTCASE( "", "" )
2012-07-20 19:43:48 +02:00
# define CATCH_REGISTER_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_REPORTER( name, reporterType )
2012-12-02 00:57:18 +01:00
# define CATCH_REGISTER_LEGACY_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_LEGACY_REPORTER( name, reporterType )
2012-07-20 19:43:48 +02:00
# define CATCH_GENERATE( expr) INTERNAL_CATCH_GENERATE( expr )
2013-03-13 13:19:30 +01:00
// "BDD-style" convenience wrappers
2013-03-21 10:00:24 +01:00
# ifdef CATCH_CONFIG_VARIADIC_MACROS
# define CATCH_SCENARIO( ... ) CATCH_TEST_CASE( "Scenario: " __VA_ARGS__ )
# else
2013-03-13 13:19:30 +01:00
# define CATCH_SCENARIO( name, tags ) CATCH_TEST_CASE( "Scenario: " name, tags )
2013-03-21 10:00:24 +01:00
# endif
2013-03-13 13:19:30 +01:00
# define CATCH_GIVEN( desc ) CATCH_SECTION( "Given: " desc, "" )
# define CATCH_WHEN( desc ) CATCH_SECTION( " When: " desc, "" )
# define CATCH_AND_WHEN( desc ) CATCH_SECTION( " And: " desc, "" )
# define CATCH_THEN( desc ) CATCH_SECTION( " Then: " desc, "" )
# define CATCH_AND_THEN( desc ) CATCH_SECTION( " And: " desc, "" )
2012-07-20 19:43:48 +02:00
// If CATCH_CONFIG_PREFIX_ALL is not defined then the CATCH_ prefix is not required
# else
2012-11-13 23:04:29 +01:00
# define REQUIRE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal, "REQUIRE" )
# define REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::Normal | Catch::ResultDisposition::NegateResult, "REQUIRE_FALSE" )
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
# define REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, ..., Catch::ResultDisposition::Normal, "REQUIRE_THROWS" )
# define REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::Normal, "REQUIRE_THROWS_AS" )
# define REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::Normal, "REQUIRE_NOTHROW" )
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
# define CHECK( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECK" )
# define CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::NegateResult, "CHECK_FALSE" )
# define CHECKED_IF( expr ) INTERNAL_CATCH_IF( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECKED_IF" )
# define CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECKED_ELSE" )
# define CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( expr, Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, "CHECK_NOFAIL" )
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
# define CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( expr, ..., Catch::ResultDisposition::ContinueOnFailure, "CHECK_THROWS" )
# define CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( expr, exceptionType, Catch::ResultDisposition::ContinueOnFailure, "CHECK_THROWS_AS" )
# define CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( expr, Catch::ResultDisposition::ContinueOnFailure, "CHECK_NOTHROW" )
2011-05-24 09:23:02 +02:00
2012-11-13 23:04:29 +01:00
# define CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::ContinueOnFailure, "CHECK_THAT" )
# define REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( arg, matcher, Catch::ResultDisposition::Normal, "REQUIRE_THAT" )
2012-03-04 22:22:36 +01:00
2013-02-02 21:37:58 +01:00
# define INFO( msg ) INTERNAL_CATCH_INFO( msg, "INFO" )
2012-11-13 23:04:29 +01:00
# define WARN( msg ) INTERNAL_CATCH_MSG( msg, Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, "WARN" )
# define FAIL( msg ) INTERNAL_CATCH_MSG( msg, Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, "FAIL" )
# define SUCCEED( msg ) INTERNAL_CATCH_MSG( msg, Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, "SUCCEED" )
2013-06-28 18:09:57 +02:00
# define SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( msg, "INFO" )
2013-02-02 21:37:58 +01:00
# define CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CAPTURE" )
2013-06-28 18:09:57 +02:00
# define SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( #msg " := " << msg, "CAPTURE" )
2011-05-24 09:23:02 +02:00
2013-03-16 21:21:51 +01:00
# ifdef CATCH_CONFIG_VARIADIC_MACROS
# define TEST_CASE( ... ) INTERNAL_CATCH_TESTCASE( __VA_ARGS__ )
2013-06-07 20:07:50 +02:00
# define TEST_CASE_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, __VA_ARGS__ )
2013-03-16 21:21:51 +01:00
# define METHOD_AS_TEST_CASE( method, ... ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, __VA_ARGS__ )
# define SECTION( ... ) INTERNAL_CATCH_SECTION( __VA_ARGS__ )
# else
# define TEST_CASE( name, description ) INTERNAL_CATCH_TESTCASE( name, description )
2013-06-07 20:07:50 +02:00
# define TEST_CASE_METHOD( className, name, description ) INTERNAL_CATCH_TEST_CASE_METHOD( className, name, description )
2013-03-16 21:21:51 +01:00
# define METHOD_AS_TEST_CASE( method, name, description ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, name, description )
# define SECTION( name, description ) INTERNAL_CATCH_SECTION( name, description )
# endif
# define ANON_TEST_CASE() INTERNAL_CATCH_TESTCASE( "", "" )
2011-05-24 09:23:02 +02:00
# define REGISTER_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_REPORTER( name, reporterType )
2012-12-02 00:57:18 +01:00
# define REGISTER_LEGACY_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_LEGACY_REPORTER( name, reporterType )
2011-05-24 09:23:02 +02:00
# define GENERATE( expr) INTERNAL_CATCH_GENERATE( expr )
2012-07-20 19:43:48 +02:00
# endif
# define CATCH_TRANSLATE_EXCEPTION( signature ) INTERNAL_CATCH_TRANSLATE_EXCEPTION( signature )
2011-05-24 09:23:02 +02:00
2013-03-13 13:19:30 +01:00
// "BDD-style" convenience wrappers
2013-03-21 10:00:24 +01:00
# ifdef CATCH_CONFIG_VARIADIC_MACROS
# define SCENARIO( ... ) TEST_CASE( "Scenario: " __VA_ARGS__ )
# else
2013-03-13 13:19:30 +01:00
# define SCENARIO( name, tags ) TEST_CASE( "Scenario: " name, tags )
2013-03-21 10:00:24 +01:00
# endif
2013-03-29 14:44:33 +01:00
# define GIVEN( desc ) SECTION( " Given: " desc, "" )
# define WHEN( desc ) SECTION( " When: " desc, "" )
# define AND_WHEN( desc ) SECTION( "And when: " desc, "" )
# define THEN( desc ) SECTION( " Then: " desc, "" )
# define AND_THEN( desc ) SECTION( " And: " desc, "" )
2013-03-13 13:19:30 +01:00
2011-05-24 09:23:02 +02:00
using Catch : : Detail : : Approx ;
2012-08-16 19:48:50 +02:00
# ifdef __clang__
2012-08-13 08:46:10 +02:00
# pragma clang diagnostic pop
2012-08-16 19:48:50 +02:00
# endif
2012-08-13 08:46:10 +02:00
2012-05-09 20:37:51 +02:00
# endif // TWOBLUECUBES_SINGLE_INCLUDE_CATCH_HPP_INCLUDED
2011-05-24 09:23:02 +02:00