2017-08-08 18:53:01 +02:00
|
|
|
/*
|
|
|
|
* Created by Phil Nash on 8/8/2017.
|
|
|
|
* Copyright 2017 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_DECOMPOSER_H_INCLUDED
|
|
|
|
#define TWOBLUECUBES_CATCH_DECOMPOSER_H_INCLUDED
|
|
|
|
|
|
|
|
#include "catch_tostring.h"
|
|
|
|
#include "catch_stringref.h"
|
2017-08-09 01:44:30 +02:00
|
|
|
#include "catch_matchers.hpp" // !TBD: for exception matchers - move this
|
2017-08-08 18:53:01 +02:00
|
|
|
|
|
|
|
#include <ostream>
|
|
|
|
|
|
|
|
namespace Catch {
|
|
|
|
|
|
|
|
struct ITransientExpression {
|
|
|
|
virtual auto isBinaryExpression() const -> bool = 0;
|
|
|
|
virtual auto getResult() const -> bool = 0;
|
|
|
|
virtual void streamReconstructedExpression( std::ostream &os ) const = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
void formatReconstructedExpression( std::ostream &os, std::string const& lhs, std::string const& op, std::string const& rhs );
|
|
|
|
|
|
|
|
template<typename LhsT, typename RhsT>
|
|
|
|
class BinaryExpr : public ITransientExpression {
|
|
|
|
bool m_result;
|
|
|
|
LhsT m_lhs;
|
|
|
|
std::string m_op;
|
|
|
|
RhsT m_rhs;
|
|
|
|
|
|
|
|
auto isBinaryExpression() const -> bool override { return true; }
|
|
|
|
auto getResult() const -> bool override { return m_result; }
|
|
|
|
|
|
|
|
void streamReconstructedExpression( std::ostream &os ) const override {
|
|
|
|
formatReconstructedExpression
|
|
|
|
( os, Catch::Detail::stringify( m_lhs ), m_op, Catch::Detail::stringify( m_rhs ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
BinaryExpr( bool comparisionResult, LhsT lhs, StringRef op, RhsT rhs )
|
|
|
|
: m_result( comparisionResult ),
|
|
|
|
m_lhs( lhs ),
|
|
|
|
m_op( op.c_str() ),
|
|
|
|
m_rhs( rhs )
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename LhsT>
|
|
|
|
class UnaryExpr : public ITransientExpression {
|
|
|
|
LhsT m_lhs;
|
|
|
|
|
|
|
|
auto isBinaryExpression() const -> bool override { return false; }
|
|
|
|
auto getResult() const -> bool override { return m_lhs ? true : false; }
|
|
|
|
|
|
|
|
void streamReconstructedExpression( std::ostream &os ) const override {
|
|
|
|
os << Catch::Detail::stringify( m_lhs );
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
UnaryExpr( LhsT lhs ) : m_lhs( lhs ) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// Specialised comparison functions to handle equality comparisons between ints and pointers (NULL deduces as an int)
|
|
|
|
template<typename LhsT, typename RhsT>
|
|
|
|
auto compareEqual( LhsT lhs, RhsT&& rhs ) -> bool { return lhs == rhs; };
|
|
|
|
template<typename T>
|
|
|
|
auto compareEqual( T* const& lhs, int rhs ) -> bool { return lhs == reinterpret_cast<void const*>( rhs ); };
|
|
|
|
template<typename T>
|
|
|
|
auto compareEqual( int lhs, T* const& rhs ) -> bool { return reinterpret_cast<void const*>( lhs ) == rhs; };
|
|
|
|
|
|
|
|
template<typename LhsT, typename RhsT>
|
|
|
|
auto compareNotEqual( LhsT lhs, RhsT&& rhs ) -> bool { return lhs != rhs; };
|
|
|
|
template<typename T>
|
|
|
|
auto compareNotEqual( T* const& lhs, int rhs ) -> bool { return lhs != reinterpret_cast<void const*>( rhs ); };
|
|
|
|
template<typename T>
|
|
|
|
auto compareNotEqual( int lhs, T* const& rhs ) -> bool { return reinterpret_cast<void const*>( lhs ) != rhs; };
|
|
|
|
|
|
|
|
|
|
|
|
template<typename LhsT>
|
|
|
|
class ExprLhs {
|
|
|
|
LhsT m_lhs;
|
|
|
|
public:
|
|
|
|
ExprLhs( LhsT lhs ) : m_lhs( lhs ) {}
|
|
|
|
|
|
|
|
template<typename RhsT>
|
|
|
|
auto operator == ( RhsT&& rhs ) -> BinaryExpr<LhsT, RhsT&> const {
|
|
|
|
return BinaryExpr<LhsT, RhsT&>( compareEqual( m_lhs, rhs ), m_lhs, "==", rhs );
|
|
|
|
}
|
|
|
|
auto operator == ( bool rhs ) -> BinaryExpr<LhsT, bool> const {
|
|
|
|
return BinaryExpr<LhsT, bool>( m_lhs == rhs, m_lhs, "==", rhs );
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RhsT>
|
|
|
|
auto operator != ( RhsT&& rhs ) -> BinaryExpr<LhsT, RhsT&> const {
|
|
|
|
return BinaryExpr<LhsT, RhsT&>( compareNotEqual( m_lhs, rhs ), m_lhs, "!=", rhs );
|
|
|
|
}
|
|
|
|
auto operator != ( bool rhs ) -> BinaryExpr<LhsT, bool> const {
|
|
|
|
return BinaryExpr<LhsT, bool>( m_lhs != rhs, m_lhs, "!=", rhs );
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RhsT>
|
|
|
|
auto operator > ( RhsT&& rhs ) -> BinaryExpr<LhsT, RhsT&> const {
|
|
|
|
return BinaryExpr<LhsT, RhsT&>( m_lhs > rhs, m_lhs, ">", rhs );
|
|
|
|
}
|
|
|
|
template<typename RhsT>
|
|
|
|
auto operator < ( RhsT&& rhs ) -> BinaryExpr<LhsT, RhsT&> const {
|
|
|
|
return BinaryExpr<LhsT, RhsT&>( m_lhs < rhs, m_lhs, "<", rhs );
|
|
|
|
}
|
|
|
|
template<typename RhsT>
|
|
|
|
auto operator >= ( RhsT&& rhs ) -> BinaryExpr<LhsT, RhsT&> const {
|
|
|
|
return BinaryExpr<LhsT, RhsT&>( m_lhs >= rhs, m_lhs, ">=", rhs );
|
|
|
|
}
|
|
|
|
template<typename RhsT>
|
|
|
|
auto operator <= ( RhsT&& rhs ) -> BinaryExpr<LhsT, RhsT&> const {
|
|
|
|
return BinaryExpr<LhsT, RhsT&>( m_lhs <= rhs, m_lhs, "<=", rhs );
|
|
|
|
}
|
|
|
|
|
|
|
|
auto makeUnaryExpr() const -> UnaryExpr<LhsT> {
|
|
|
|
return UnaryExpr<LhsT>( m_lhs );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void handleExpression( ITransientExpression const& expr );
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void handleExpression( ExprLhs<T> const& expr ) {
|
|
|
|
handleExpression( expr.makeUnaryExpr() );
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Decomposer {
|
|
|
|
template<typename T>
|
|
|
|
auto operator <= ( T& lhs ) -> ExprLhs<T&> {
|
|
|
|
return ExprLhs<T&>( lhs );
|
|
|
|
}
|
|
|
|
template<typename T>
|
|
|
|
auto operator <= ( T const& lhs ) -> ExprLhs<T const&> {
|
|
|
|
return ExprLhs<T const&>( lhs );
|
|
|
|
}
|
|
|
|
auto operator <=( bool value ) -> ExprLhs<bool> {
|
|
|
|
return ExprLhs<bool>( value );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-08-09 00:31:43 +02:00
|
|
|
// !TBD: this is just here temporarily
|
|
|
|
template<typename ArgT, typename MatcherT>
|
|
|
|
class MatchExpr : public ITransientExpression {
|
|
|
|
ArgT const& m_arg;
|
2017-08-09 01:44:30 +02:00
|
|
|
MatcherT m_matcher;
|
2017-08-09 00:31:43 +02:00
|
|
|
StringRef m_matcherString;
|
|
|
|
bool m_result;
|
|
|
|
public:
|
|
|
|
MatchExpr( ArgT const& arg, MatcherT const& matcher, StringRef matcherString )
|
|
|
|
: m_arg( arg ),
|
|
|
|
m_matcher( matcher ),
|
|
|
|
m_matcherString( matcherString ),
|
|
|
|
m_result( matcher.match( arg ) )
|
|
|
|
{}
|
|
|
|
|
|
|
|
auto isBinaryExpression() const -> bool override { return true; }
|
|
|
|
auto getResult() const -> bool override { return m_result; }
|
|
|
|
|
|
|
|
void streamReconstructedExpression( std::ostream &os ) const override {
|
|
|
|
auto matcherAsString = m_matcher.toString();
|
|
|
|
os << Catch::Detail::stringify( m_arg ) << ' ';
|
|
|
|
if( matcherAsString == Detail::unprintableString )
|
|
|
|
os << m_matcherString.c_str();
|
|
|
|
else
|
|
|
|
os << matcherAsString;
|
|
|
|
}
|
|
|
|
};
|
2017-08-09 01:44:30 +02:00
|
|
|
|
2017-08-09 00:31:43 +02:00
|
|
|
template<typename ArgT, typename MatcherT>
|
|
|
|
auto makeMatchExpr( ArgT const& arg, MatcherT const& matcher, StringRef matcherString ) -> MatchExpr<ArgT, MatcherT> {
|
|
|
|
return MatchExpr<ArgT, MatcherT>( arg, matcher, matcherString );
|
|
|
|
}
|
|
|
|
|
2017-08-08 18:53:01 +02:00
|
|
|
} // end namespace Catch
|
|
|
|
|
|
|
|
#endif // TWOBLUECUBES_CATCH_DECOMPOSER_H_INCLUDED
|