| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  catch_capture.hpp | 
					
						
							|  |  |  |  *  Catch | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  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
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "catch_resultinfo.hpp"
 | 
					
						
							|  |  |  | #include <sstream>
 | 
					
						
							| 
									
										
										
										
											2010-11-11 20:56:38 +00:00
										 |  |  | #include <cmath>
 | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Catch | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-16 19:30:41 +00:00
										 |  |  | namespace Detail | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // The following code, contributed by Sam Partington, allows us to choose an implementation
 | 
					
						
							|  |  |  |     // of toString() depending on whether a << overload is available
 | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     struct NonStreamable | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // allow construction from anything...
 | 
					
						
							|  |  |  |         template<typename Anything>  | 
					
						
							|  |  |  |         NonStreamable(Anything) | 
					
						
							|  |  |  |         {} | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     // a local operator<<  which may be called if there isn't a better one elsewhere...
 | 
					
						
							|  |  |  |     inline NonStreamable operator << ( std::ostream&, const NonStreamable& ns ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return ns; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template<typename T> | 
					
						
							|  |  |  |     struct IsStreamable | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         static NoType Deduce( const NonStreamable& ); | 
					
						
							|  |  |  |         static YesType Deduce( std::ostream& ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         enum | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             value = sizeof( Deduce( Synth<std::ostream&>() << Synth<const T&>() ) )  | 
					
						
							|  |  |  |                         == sizeof( YesType ) | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     // << is available, so use it with ostringstream to make the string
 | 
					
						
							|  |  |  |     template<typename T, bool streamable> | 
					
						
							|  |  |  |     struct StringMaker | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         static std::string apply( const T& value ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             std::ostringstream oss; | 
					
						
							|  |  |  |             oss << value; | 
					
						
							|  |  |  |             return oss.str(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |     // << not available - use a default string
 | 
					
						
							|  |  |  |     template<typename T> | 
					
						
							|  |  |  |     struct StringMaker<T, false> | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         static std::string apply( const T& value ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return "{?}"; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }// end namespace Detail
 | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | template<typename T> | 
					
						
							|  |  |  | std::string toString( const T& value ) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-16 19:30:41 +00:00
										 |  |  |     return Detail::StringMaker<T, Detail::IsStreamable<T>::value>::apply( value ); | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestFailureException | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | class DummyExceptionType_DontUse | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | class MutableResultInfo : public ResultInfo | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     MutableResultInfo() | 
					
						
							|  |  |  |     {} | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2010-11-11 20:56:38 +00:00
										 |  |  |     MutableResultInfo( const std::string& expr, bool isNot, const std::string& filename, std::size_t line, const std::string& macroName ) | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  |     : ResultInfo( ( isNot ? "!" : "" ) + expr, ResultWas::Unknown, isNot, filename, line, macroName ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     void setResultType( ResultWas::OfType result ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Flip bool results if isNot is set
 | 
					
						
							|  |  |  |         if( m_isNot && result == ResultWas::Ok ) | 
					
						
							|  |  |  |             m_result = ResultWas::ExpressionFailed; | 
					
						
							|  |  |  |         else if( m_isNot && result == ResultWas::ExpressionFailed ) | 
					
						
							|  |  |  |             m_result = ResultWas::Ok; | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             m_result = result;         | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     void setMessage( const std::string& message ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_message = message; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-10 19:18:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     template<typename RhsT> | 
					
						
							| 
									
										
										
										
											2010-11-12 19:32:13 +00:00
										 |  |  |     MutableResultInfo& operator ||( const RhsT& ) | 
					
						
							| 
									
										
										
										
											2010-11-10 19:18:46 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-11-11 07:21:57 +00:00
										 |  |  |         m_expressionIncomplete = true; | 
					
						
							| 
									
										
										
										
											2010-11-10 19:18:46 +00:00
										 |  |  |         return *this; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-11 20:37:46 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  | private: | 
					
						
							|  |  |  |     friend class ResultBuilder; | 
					
						
							|  |  |  |     void setLhs( const std::string& lhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_lhs = lhs; | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  |     MutableResultInfo& setRhs( const std::string& op, const std::string& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_op = op; | 
					
						
							|  |  |  |         m_rhs = rhs; | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | class ResultBuilder | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2010-11-11 20:56:38 +00:00
										 |  |  |     ResultBuilder( const char* expr, bool isNot, const std::string& filename, std::size_t line, const std::string& macroName ) | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  |     : m_result( expr, isNot, filename, line, macroName ) | 
					
						
							|  |  |  |     {} | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     template<typename T> | 
					
						
							|  |  |  |     ResultBuilder& operator->*(const T & operand) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_result.setLhs( toString( operand ) ); | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     template<typename RhsT> | 
					
						
							|  |  |  |     MutableResultInfo& operator == ( const RhsT& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_result.setRhs( "==", toString( rhs ) ); | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  |     template<typename RhsT> | 
					
						
							|  |  |  |     MutableResultInfo& operator != ( const RhsT& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_result.setRhs( "!=", toString( rhs ) ); | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  |     template<typename RhsT> | 
					
						
							|  |  |  |     MutableResultInfo& operator < ( const RhsT& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_result.setRhs( "<", toString( rhs ) ); | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  |     template<typename RhsT> | 
					
						
							|  |  |  |     MutableResultInfo& operator > ( const RhsT& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_result.setRhs( ">", toString( rhs ) ); | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  |     template<typename RhsT> | 
					
						
							|  |  |  |     MutableResultInfo& operator <= ( const RhsT& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_result.setRhs( "<=", toString( rhs ) ); | 
					
						
							|  |  |  |     }     | 
					
						
							|  |  |  |     template<typename RhsT> | 
					
						
							|  |  |  |     MutableResultInfo& operator >= ( const RhsT& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_result.setRhs( ">=", toString( rhs ) ); | 
					
						
							|  |  |  |     }     | 
					
						
							| 
									
										
										
										
											2010-11-10 19:18:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  |     operator MutableResultInfo&() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return m_result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     MutableResultInfo m_result; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestCaseInfo; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | struct IResultListener | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     virtual ~IResultListener(){} | 
					
						
							|  |  |  |     virtual void testEnded( const ResultInfo& result ) = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | class ResultsCapture | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     ResultsCapture() | 
					
						
							|  |  |  |     : m_listener( 0 ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static ResultsCapture& instance() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         static ResultsCapture instance; | 
					
						
							|  |  |  |         return instance; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     static IResultListener* setListener( IResultListener* listener ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         IResultListener* prevListener = instance().m_listener; | 
					
						
							|  |  |  |         instance().m_listener = listener; | 
					
						
							|  |  |  |         return prevListener; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static void acceptExpression( const MutableResultInfo& resultInfo ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         instance().currentResult = resultInfo; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static void acceptResult( bool result, bool stopOnFail ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         acceptResult( result ? ResultWas::Ok : ResultWas::ExpressionFailed, stopOnFail ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     static void acceptResult( ResultWas::OfType result, bool stopOnFail ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if( !acceptResult( result ) && stopOnFail ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             throw TestFailureException(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     static bool acceptResult( ResultWas::OfType result ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         MutableResultInfo& currentResult = instance().currentResult; | 
					
						
							|  |  |  |         currentResult.setResultType( result ); | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         if( instance().m_listener ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             instance().m_listener->testEnded( currentResult ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         bool ok = currentResult.ok(); | 
					
						
							|  |  |  |         instance().currentResult = MutableResultInfo(); | 
					
						
							|  |  |  |         return ok; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static bool acceptResult( bool expressionResult ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return acceptResult( expressionResult ? ResultWas::Ok : ResultWas::ExpressionFailed ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     static void acceptMessage( const std::string& msg ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         instance().currentResult.setMessage( msg ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     MutableResultInfo currentResult; | 
					
						
							|  |  |  |     IResultListener* m_listener; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // !TBD Need to clean this all up
 | 
					
						
							|  |  |  | #define CATCH_absTol 1e-10
 | 
					
						
							|  |  |  | #define CATCH_relTol 1e-10
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-12 08:12:01 +00:00
										 |  |  | inline double catch_max( double x, double y ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return x > y ? x : y; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  | class Approx | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     // !TBD more generic
 | 
					
						
							|  |  |  |     Approx( double d ) | 
					
						
							|  |  |  |     : m_d( d ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     template<typename T> | 
					
						
							|  |  |  |     friend bool operator == ( const T& lhs, const Approx& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // !TBD Use proper tolerance
 | 
					
						
							|  |  |  |         // From: http://realtimecollisiondetection.net/blog/?p=89
 | 
					
						
							|  |  |  |         // see also: http://www.cygnus-software.com/papers/comparingfloats/comparingfloats.htm
 | 
					
						
							| 
									
										
										
										
											2010-11-12 08:12:01 +00:00
										 |  |  |         return fabs( lhs - rhs.m_d ) <= catch_max( CATCH_absTol, CATCH_relTol * catch_max( fabs(lhs), fabs(rhs.m_d) ) ); | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     template<typename T> | 
					
						
							|  |  |  |     friend bool operator != ( const T& lhs, const Approx& rhs ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return ! operator==( lhs, rhs ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     double m_d; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template<> | 
					
						
							|  |  |  | inline std::string toString<Approx>( const Approx& value ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     std::ostringstream oss; | 
					
						
							|  |  |  |     oss << "Approx( " << value.m_d << ")"; | 
					
						
							|  |  |  |     return oss.str(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | } // end namespace Catch
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-10 07:42:15 +00:00
										 |  |  | #define INTERNAL_CATCH_TEST( expr, isNot, stopOnFailure, macroName ) \
 | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  |     Catch::ResultsCapture::acceptExpression( Catch::ResultBuilder( #expr, isNot, __FILE__, __LINE__, macroName )->*expr ); \ | 
					
						
							|  |  |  |     Catch::ResultsCapture::acceptResult( expr, stopOnFailure ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-10 07:42:15 +00:00
										 |  |  | #define INTERNAL_CATCH_THROWS( expr, exceptionType, nothrow, stopOnFailure, macroName ) \
 | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  |     Catch::ResultsCapture::acceptExpression( Catch::ResultBuilder( #expr, false, __FILE__, __LINE__, macroName ) ); \ | 
					
						
							|  |  |  |     try \ | 
					
						
							|  |  |  |     { \ | 
					
						
							|  |  |  |         expr; \ | 
					
						
							|  |  |  |         Catch::ResultsCapture::acceptResult( nothrow, stopOnFailure ); \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |     catch( exceptionType ) \ | 
					
						
							|  |  |  |     { \ | 
					
						
							|  |  |  |         Catch::ResultsCapture::acceptResult( !(nothrow), stopOnFailure ); \ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-10 07:42:15 +00:00
										 |  |  | #define INTERNAL_CATCH_THROWS_AS( expr, exceptionType, nothrow, stopOnFailure, macroName ) \
 | 
					
						
							|  |  |  | INTERNAL_CATCH_THROWS( expr, exceptionType, nothrow, stopOnFailure, macroName ) \ | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  | catch( ... ) \ | 
					
						
							|  |  |  | { \ | 
					
						
							|  |  |  |     Catch::ResultsCapture::acceptResult( false, stopOnFailure ); \ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-10 07:42:15 +00:00
										 |  |  | #define INTERNAL_CATCH_MSG( reason, resultType, stopOnFailure, macroName ) \
 | 
					
						
							| 
									
										
										
										
											2010-11-09 23:24:00 +00:00
										 |  |  |     Catch::ResultsCapture::acceptExpression( Catch::MutableResultInfo( "", false, __FILE__, __LINE__, macroName ) ); \ | 
					
						
							|  |  |  |     Catch::ResultsCapture::acceptMessage( reason ); \ | 
					
						
							|  |  |  |     Catch::ResultsCapture::acceptResult( resultType, stopOnFailure ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif // TWOBLUECUBES_CATCH_CAPTURE_HPP_INCLUDED
 |