2010-11-10 00:24:00 +01:00
|
|
|
/*
|
|
|
|
* Created by Phil on 18/10/2010.
|
|
|
|
* Copyright 2010 Two Blue Cubes Ltd. All rights reserved.
|
|
|
|
*
|
|
|
|
* Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
* file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
*/
|
|
|
|
#ifndef TWOBLUECUBES_CATCH_CAPTURE_HPP_INCLUDED
|
|
|
|
#define TWOBLUECUBES_CATCH_CAPTURE_HPP_INCLUDED
|
|
|
|
|
2012-05-11 20:15:54 +02:00
|
|
|
#include "catch_expression_builder.hpp"
|
2011-01-11 10:13:31 +01:00
|
|
|
#include "catch_interfaces_capture.h"
|
2010-12-27 21:49:19 +01:00
|
|
|
#include "catch_debugger.hpp"
|
2012-05-10 08:58:48 +02:00
|
|
|
#include "catch_context.h"
|
2011-04-28 09:03:28 +02:00
|
|
|
#include "catch_common.h"
|
2012-05-11 20:15:54 +02:00
|
|
|
#include <ostream>
|
2010-11-10 00:24:00 +01:00
|
|
|
|
2012-05-15 08:42:26 +02:00
|
|
|
namespace Catch {
|
2012-05-11 20:22:28 +02:00
|
|
|
|
2012-05-08 08:59:54 +02:00
|
|
|
struct TestFailureException{};
|
2010-11-10 00:24:00 +01:00
|
|
|
|
2012-05-15 08:42:26 +02:00
|
|
|
class ScopedInfo {
|
2010-12-27 12:09:34 +01:00
|
|
|
public:
|
2012-05-09 09:17:51 +02:00
|
|
|
ScopedInfo() : m_oss() {
|
2012-05-21 19:52:09 +02:00
|
|
|
getCurrentContext().getResultCapture().pushScopedInfo( this );
|
2010-12-27 12:09:34 +01:00
|
|
|
}
|
|
|
|
|
2012-05-09 09:17:51 +02:00
|
|
|
~ScopedInfo() {
|
2012-05-21 19:52:09 +02:00
|
|
|
getCurrentContext().getResultCapture().popScopedInfo( this );
|
2010-12-27 12:09:34 +01:00
|
|
|
}
|
|
|
|
|
2012-05-09 09:17:51 +02:00
|
|
|
template<typename T>
|
|
|
|
ScopedInfo& operator << ( const T& value ) {
|
|
|
|
m_oss << value;
|
2010-12-27 12:09:34 +01:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2012-05-09 09:17:51 +02:00
|
|
|
std::string getInfo () const {
|
2010-12-27 12:09:34 +01:00
|
|
|
return m_oss.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::ostringstream m_oss;
|
|
|
|
};
|
2011-04-04 08:56:44 +02:00
|
|
|
|
2010-12-30 00:13:22 +01:00
|
|
|
// This is just here to avoid compiler warnings with macro constants
|
2012-05-09 09:17:51 +02:00
|
|
|
inline bool isTrue( bool value ){ return value; }
|
2010-11-10 00:24:00 +01:00
|
|
|
|
|
|
|
} // end namespace Catch
|
|
|
|
|
2011-02-03 21:00:46 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2012-05-09 09:17:51 +02:00
|
|
|
#define INTERNAL_CATCH_ACCEPT_EXPR( expr, stopOnFailure, originalExpr ) \
|
2012-05-21 19:52:09 +02:00
|
|
|
if( Catch::ResultAction::Value internal_catch_action = Catch::getCurrentContext().getResultCapture().acceptExpression( expr ) ) \
|
2010-12-27 21:49:19 +01:00
|
|
|
{ \
|
2011-06-23 09:13:25 +02:00
|
|
|
if( internal_catch_action == Catch::ResultAction::DebugFailed ) BreakIntoDebugger(); \
|
2010-12-30 00:13:22 +01:00
|
|
|
if( Catch::isTrue( stopOnFailure ) ) throw Catch::TestFailureException(); \
|
2012-05-09 09:17:51 +02:00
|
|
|
if( Catch::isTrue( false ) ){ bool this_is_here_to_invoke_warnings = ( originalExpr ); Catch::isTrue( this_is_here_to_invoke_warnings ); } \
|
2010-12-27 21:49:19 +01:00
|
|
|
}
|
|
|
|
|
2011-02-03 21:00:46 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2010-11-10 08:42:15 +01:00
|
|
|
#define INTERNAL_CATCH_TEST( expr, isNot, stopOnFailure, macroName ) \
|
2012-02-29 09:35:41 +01:00
|
|
|
do{ try{ \
|
2012-05-11 20:22:28 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( ( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #expr, isNot )->*expr ), stopOnFailure, expr ); \
|
2012-02-29 09:35:41 +01:00
|
|
|
}catch( Catch::TestFailureException& ){ \
|
2012-02-17 20:50:59 +01:00
|
|
|
throw; \
|
2012-02-29 09:35:41 +01:00
|
|
|
} catch( ... ){ \
|
2012-05-21 19:52:09 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( ( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #expr ) << Catch::getCurrentContext().getExceptionTranslatorRegistry().translateActiveException() ).setResultType( Catch::ResultWas::ThrewException ), false, expr ); \
|
2012-02-17 20:50:59 +01:00
|
|
|
throw; \
|
2012-02-29 09:47:18 +01:00
|
|
|
}}while( Catch::isTrue( false ) )
|
2010-11-10 00:24:00 +01:00
|
|
|
|
2012-02-10 09:30:13 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define INTERNAL_CATCH_IF( expr, isNot, stopOnFailure, macroName ) \
|
|
|
|
INTERNAL_CATCH_TEST( expr, isNot, stopOnFailure, macroName ); \
|
2012-05-21 19:52:09 +02:00
|
|
|
if( Catch::getCurrentContext().getResultCapture().getLastResult()->ok() )
|
2012-02-10 09:30:13 +01:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define INTERNAL_CATCH_ELSE( expr, isNot, stopOnFailure, macroName ) \
|
|
|
|
INTERNAL_CATCH_TEST( expr, isNot, stopOnFailure, macroName ); \
|
2012-05-21 19:52:09 +02:00
|
|
|
if( !Catch::getCurrentContext().getResultCapture().getLastResult()->ok() )
|
2012-02-10 09:30:13 +01:00
|
|
|
|
2011-03-14 09:45:55 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define INTERNAL_CATCH_NO_THROW( expr, stopOnFailure, macroName ) \
|
|
|
|
try \
|
|
|
|
{ \
|
|
|
|
expr; \
|
2012-05-11 20:22:28 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #expr ).setResultType( Catch::ResultWas::Ok ), stopOnFailure, false ); \
|
2011-03-14 09:45:55 +01:00
|
|
|
} \
|
|
|
|
catch( ... ) \
|
|
|
|
{ \
|
2012-05-21 19:52:09 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( ( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #expr ) << Catch::getCurrentContext().getExceptionTranslatorRegistry().translateActiveException() ).setResultType( Catch::ResultWas::ThrewException ), stopOnFailure, false ); \
|
2011-03-14 09:45:55 +01:00
|
|
|
}
|
|
|
|
|
2011-02-03 21:00:46 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2011-03-14 09:49:42 +01:00
|
|
|
#define INTERNAL_CATCH_THROWS( expr, exceptionType, stopOnFailure, macroName ) \
|
2010-11-10 00:24:00 +01:00
|
|
|
try \
|
|
|
|
{ \
|
|
|
|
expr; \
|
2012-05-11 20:22:28 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #expr ).setResultType( Catch::ResultWas::DidntThrowException ), stopOnFailure, false ); \
|
2011-03-14 20:21:35 +01:00
|
|
|
} \
|
|
|
|
catch( Catch::TestFailureException& ) \
|
|
|
|
{ \
|
|
|
|
throw; \
|
2010-11-10 00:24:00 +01:00
|
|
|
} \
|
|
|
|
catch( exceptionType ) \
|
|
|
|
{ \
|
2012-05-11 20:22:28 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #expr ).setResultType( Catch::ResultWas::Ok ), stopOnFailure, false ); \
|
2010-11-10 00:24:00 +01:00
|
|
|
}
|
|
|
|
|
2011-02-03 21:00:46 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2011-03-14 09:49:42 +01:00
|
|
|
#define INTERNAL_CATCH_THROWS_AS( expr, exceptionType, stopOnFailure, macroName ) \
|
|
|
|
INTERNAL_CATCH_THROWS( expr, exceptionType, stopOnFailure, macroName ) \
|
2011-03-14 09:45:55 +01:00
|
|
|
catch( ... ) \
|
|
|
|
{ \
|
2012-05-21 19:52:09 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( ( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #expr ) << Catch::getCurrentContext().getExceptionTranslatorRegistry().translateActiveException() ).setResultType( Catch::ResultWas::ThrewException ), stopOnFailure, false ); \
|
2011-03-14 09:45:55 +01:00
|
|
|
}
|
2010-11-10 00:24:00 +01:00
|
|
|
|
2011-02-03 21:00:46 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2010-11-10 08:42:15 +01:00
|
|
|
#define INTERNAL_CATCH_MSG( reason, resultType, stopOnFailure, macroName ) \
|
2012-05-21 19:52:09 +02:00
|
|
|
Catch::getCurrentContext().getResultCapture().acceptExpression( ( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName ) << reason ).setResultType( resultType ) );
|
2010-11-10 00:24:00 +01:00
|
|
|
|
2011-02-03 21:00:46 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2011-03-10 15:09:32 +01:00
|
|
|
#define INTERNAL_CATCH_SCOPED_INFO( log ) \
|
|
|
|
Catch::ScopedInfo INTERNAL_CATCH_UNIQUE_NAME( info ); \
|
|
|
|
INTERNAL_CATCH_UNIQUE_NAME( info ) << log
|
2010-11-10 00:24:00 +01:00
|
|
|
|
2012-03-04 12:14:21 +01:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define INTERNAL_CHECK_THAT( arg, matcher, stopOnFailure, macroName ) \
|
|
|
|
do{ try{ \
|
2012-05-11 20:22:28 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( ( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #arg " " #matcher, false ).acceptMatcher( matcher, arg, #matcher ) ), stopOnFailure, false ); \
|
2012-03-04 12:14:21 +01:00
|
|
|
}catch( Catch::TestFailureException& ){ \
|
|
|
|
throw; \
|
|
|
|
} catch( ... ){ \
|
2012-05-21 19:52:09 +02:00
|
|
|
INTERNAL_CATCH_ACCEPT_EXPR( ( Catch::ExpressionBuilder( CATCH_INTERNAL_LINEINFO, macroName, #arg " " #matcher ) << Catch::getCurrentContext().getExceptionTranslatorRegistry().translateActiveException() ).setResultType( Catch::ResultWas::ThrewException ), false, false ); \
|
2012-03-04 12:14:21 +01:00
|
|
|
throw; \
|
|
|
|
}}while( Catch::isTrue( false ) )
|
|
|
|
|
2010-11-10 00:24:00 +01:00
|
|
|
#endif // TWOBLUECUBES_CATCH_CAPTURE_HPP_INCLUDED
|