2012-11-01 09:27:09 +01:00
|
|
|
/*
|
|
|
|
* Created by Phil on 11/5/2012.
|
|
|
|
* Copyright 2012 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_EXPRESSION_LHS_HPP_INCLUDED
|
|
|
|
#define TWOBLUECUBES_CATCH_EXPRESSION_LHS_HPP_INCLUDED
|
|
|
|
|
2014-05-28 19:53:01 +02:00
|
|
|
#include "catch_result_builder.h"
|
2012-11-01 09:27:09 +01:00
|
|
|
#include "catch_evaluate.hpp"
|
2014-04-23 07:51:58 +02:00
|
|
|
#include "catch_tostring.h"
|
2012-11-01 09:27:09 +01:00
|
|
|
|
|
|
|
namespace Catch {
|
|
|
|
|
2017-01-09 13:23:10 +01:00
|
|
|
template<typename LhsT, Internal::Operator Op, typename RhsT>
|
|
|
|
class BinaryExpression;
|
|
|
|
|
|
|
|
template<typename ArgT, typename MatcherT>
|
|
|
|
class MatchExpression;
|
2012-11-01 09:27:09 +01:00
|
|
|
|
2017-01-09 13:23:10 +01:00
|
|
|
// Wraps the LHS of an expression and overloads comparison operators
|
|
|
|
// for also capturing those and RHS (if any)
|
|
|
|
template<typename T>
|
|
|
|
class ExpressionLhs : public DecomposedExpression {
|
2012-11-01 09:27:09 +01:00
|
|
|
public:
|
2017-01-09 13:23:10 +01:00
|
|
|
ExpressionLhs( ResultBuilder& rb, T lhs ) : m_rb( rb ), m_lhs( lhs ), m_truthy(false) {}
|
2012-11-01 09:27:09 +01:00
|
|
|
|
2017-02-21 09:47:34 +01:00
|
|
|
ExpressionLhs& operator = ( const ExpressionLhs& );
|
|
|
|
|
2012-11-01 09:27:09 +01:00
|
|
|
template<typename RhsT>
|
2017-01-09 13:23:10 +01:00
|
|
|
BinaryExpression<T, Internal::IsEqualTo, RhsT const&>
|
2017-02-06 20:39:18 +01:00
|
|
|
operator == ( RhsT const& rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsEqualTo>( rhs );
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RhsT>
|
2017-01-09 13:23:10 +01:00
|
|
|
BinaryExpression<T, Internal::IsNotEqualTo, RhsT const&>
|
2017-02-06 20:39:18 +01:00
|
|
|
operator != ( RhsT const& rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsNotEqualTo>( rhs );
|
|
|
|
}
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2012-11-01 09:27:09 +01:00
|
|
|
template<typename RhsT>
|
2017-01-09 13:23:10 +01:00
|
|
|
BinaryExpression<T, Internal::IsLessThan, RhsT const&>
|
2017-02-06 20:39:18 +01:00
|
|
|
operator < ( RhsT const& rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsLessThan>( rhs );
|
|
|
|
}
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2012-11-01 09:27:09 +01:00
|
|
|
template<typename RhsT>
|
2017-01-09 13:23:10 +01:00
|
|
|
BinaryExpression<T, Internal::IsGreaterThan, RhsT const&>
|
2017-02-06 20:39:18 +01:00
|
|
|
operator > ( RhsT const& rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsGreaterThan>( rhs );
|
|
|
|
}
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2012-11-01 09:27:09 +01:00
|
|
|
template<typename RhsT>
|
2017-01-09 13:23:10 +01:00
|
|
|
BinaryExpression<T, Internal::IsLessThanOrEqualTo, RhsT const&>
|
2017-02-06 20:39:18 +01:00
|
|
|
operator <= ( RhsT const& rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsLessThanOrEqualTo>( rhs );
|
|
|
|
}
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2012-11-01 09:27:09 +01:00
|
|
|
template<typename RhsT>
|
2017-01-09 13:23:10 +01:00
|
|
|
BinaryExpression<T, Internal::IsGreaterThanOrEqualTo, RhsT const&>
|
2017-02-06 20:39:18 +01:00
|
|
|
operator >= ( RhsT const& rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsGreaterThanOrEqualTo>( rhs );
|
|
|
|
}
|
|
|
|
|
2017-02-06 20:39:18 +01:00
|
|
|
BinaryExpression<T, Internal::IsEqualTo, bool> operator == ( bool rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsEqualTo>( rhs );
|
|
|
|
}
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2017-02-06 20:39:18 +01:00
|
|
|
BinaryExpression<T, Internal::IsNotEqualTo, bool> operator != ( bool rhs ) {
|
2012-11-01 09:27:09 +01:00
|
|
|
return captureExpression<Internal::IsNotEqualTo>( rhs );
|
|
|
|
}
|
2013-07-03 20:14:59 +02:00
|
|
|
|
2014-05-28 19:53:01 +02:00
|
|
|
void endExpression() {
|
2017-01-09 13:23:10 +01:00
|
|
|
m_truthy = m_lhs ? true : false;
|
2014-05-29 08:50:19 +02:00
|
|
|
m_rb
|
2017-01-09 13:23:10 +01:00
|
|
|
.setResultType( m_truthy )
|
|
|
|
.endExpression( *this );
|
2012-11-01 09:27:09 +01:00
|
|
|
}
|
|
|
|
|
2017-02-06 23:37:23 +01:00
|
|
|
virtual void reconstructExpression( std::string& dest ) const CATCH_OVERRIDE {
|
|
|
|
dest = Catch::toString( m_truthy );
|
2017-01-09 13:23:10 +01:00
|
|
|
}
|
2012-11-01 09:27:09 +01:00
|
|
|
|
|
|
|
private:
|
|
|
|
template<Internal::Operator Op, typename RhsT>
|
2017-01-09 13:23:10 +01:00
|
|
|
BinaryExpression<T, Op, RhsT&> captureExpression( RhsT& rhs ) const {
|
|
|
|
return BinaryExpression<T, Op, RhsT&>( m_rb, m_lhs, rhs );
|
|
|
|
}
|
|
|
|
|
|
|
|
template<Internal::Operator Op>
|
|
|
|
BinaryExpression<T, Op, bool> captureExpression( bool rhs ) const {
|
|
|
|
return BinaryExpression<T, Op, bool>( m_rb, m_lhs, rhs );
|
2012-11-01 09:27:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2014-05-29 08:50:19 +02:00
|
|
|
ResultBuilder& m_rb;
|
2012-11-01 09:27:09 +01:00
|
|
|
T m_lhs;
|
2017-01-09 13:23:10 +01:00
|
|
|
bool m_truthy;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename LhsT, Internal::Operator Op, typename RhsT>
|
|
|
|
class BinaryExpression : public DecomposedExpression {
|
|
|
|
public:
|
|
|
|
BinaryExpression( ResultBuilder& rb, LhsT lhs, RhsT rhs )
|
|
|
|
: m_rb( rb ), m_lhs( lhs ), m_rhs( rhs ) {}
|
|
|
|
|
2017-02-21 09:47:34 +01:00
|
|
|
BinaryExpression& operator = ( BinaryExpression& );
|
|
|
|
|
2017-01-09 13:23:10 +01:00
|
|
|
void endExpression() const {
|
|
|
|
m_rb
|
|
|
|
.setResultType( Internal::compare<Op>( m_lhs, m_rhs ) )
|
|
|
|
.endExpression( *this );
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool isBinaryExpression() const CATCH_OVERRIDE {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-02-06 23:37:23 +01:00
|
|
|
virtual void reconstructExpression( std::string& dest ) const CATCH_OVERRIDE {
|
|
|
|
std::string lhs = Catch::toString( m_lhs );
|
|
|
|
std::string rhs = Catch::toString( m_rhs );
|
|
|
|
char delim = lhs.size() + rhs.size() < 40 &&
|
|
|
|
lhs.find('\n') == std::string::npos &&
|
|
|
|
rhs.find('\n') == std::string::npos ? ' ' : '\n';
|
|
|
|
dest.reserve( 7 + lhs.size() + rhs.size() );
|
|
|
|
// 2 for spaces around operator
|
|
|
|
// 2 for operator
|
|
|
|
// 2 for parentheses (conditionally added later)
|
|
|
|
// 1 for negation (conditionally added later)
|
|
|
|
dest = lhs;
|
|
|
|
dest += delim;
|
|
|
|
dest += Internal::OperatorTraits<Op>::getName();
|
|
|
|
dest += delim;
|
|
|
|
dest += rhs;
|
2017-01-09 13:23:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
ResultBuilder& m_rb;
|
|
|
|
LhsT m_lhs;
|
|
|
|
RhsT m_rhs;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename ArgT, typename MatcherT>
|
|
|
|
class MatchExpression : public DecomposedExpression {
|
|
|
|
public:
|
|
|
|
MatchExpression( ArgT arg, MatcherT matcher, char const* matcherString )
|
|
|
|
: m_arg( arg ), m_matcher( matcher ), m_matcherString( matcherString ) {}
|
|
|
|
|
|
|
|
virtual bool isBinaryExpression() const CATCH_OVERRIDE {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-02-06 23:37:23 +01:00
|
|
|
virtual void reconstructExpression( std::string& dest ) const CATCH_OVERRIDE {
|
2017-01-09 13:23:10 +01:00
|
|
|
std::string matcherAsString = m_matcher.toString();
|
2017-02-06 23:37:23 +01:00
|
|
|
dest = Catch::toString( m_arg );
|
|
|
|
dest += ' ';
|
2017-01-09 13:23:10 +01:00
|
|
|
if( matcherAsString == Detail::unprintableString )
|
|
|
|
dest += m_matcherString;
|
|
|
|
else
|
|
|
|
dest += matcherAsString;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
ArgT m_arg;
|
|
|
|
MatcherT m_matcher;
|
|
|
|
char const* m_matcherString;
|
2012-11-01 09:27:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end namespace Catch
|
|
|
|
|
|
|
|
#endif // TWOBLUECUBES_CATCH_EXPRESSION_LHS_HPP_INCLUDED
|