mirror of
https://github.com/mat007/turtle.git
synced 2026-06-22 12:13:43 +00:00
git-svn-id: https://svn.code.sf.net/p/turtle/code/trunk@476 860be788-9bd5-4423-9f1e-828f051e677b
311 lines
7.8 KiB
C++
311 lines
7.8 KiB
C++
// http://turtle.sourceforge.net
|
|
//
|
|
// Copyright Mathieu Champlon 2011
|
|
//
|
|
// 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 MOCK_LOG_HPP_INCLUDED
|
|
#define MOCK_LOG_HPP_INCLUDED
|
|
|
|
#include <boost/utility/enable_if.hpp>
|
|
#include <boost/detail/container_fwd.hpp>
|
|
#include <boost/function_types/is_callable_builtin.hpp>
|
|
#include <boost/utility/addressof.hpp>
|
|
#include <ostream>
|
|
#include <memory>
|
|
|
|
namespace boost
|
|
{
|
|
template< typename T > class shared_ptr;
|
|
template< typename T > class weak_ptr;
|
|
template< typename T > class reference_wrapper;
|
|
|
|
namespace phoenix
|
|
{
|
|
template< typename T > struct actor;
|
|
}
|
|
namespace lambda
|
|
{
|
|
template< typename T > class lambda_functor;
|
|
}
|
|
namespace assign_detail
|
|
{
|
|
template< typename T > class generic_list;
|
|
}
|
|
}
|
|
|
|
namespace mock
|
|
{
|
|
struct stream
|
|
{
|
|
explicit stream( std::ostream& s )
|
|
: s_( &s )
|
|
{}
|
|
std::ostream* s_;
|
|
};
|
|
|
|
#ifdef MOCK_USE_CONVERSIONS
|
|
|
|
namespace detail3
|
|
{
|
|
struct sink
|
|
{
|
|
template< typename T >
|
|
sink( const T& )
|
|
{}
|
|
};
|
|
|
|
inline std::ostream& operator<<( std::ostream& s, const sink& )
|
|
{
|
|
return s << '?';
|
|
}
|
|
|
|
struct holder
|
|
{
|
|
virtual ~holder()
|
|
{}
|
|
virtual void serialize( std::ostream& s ) const = 0;
|
|
};
|
|
|
|
template< typename T >
|
|
struct holder_imp : holder
|
|
{
|
|
explicit holder_imp( const T& t )
|
|
: t_( boost::addressof( t ) )
|
|
{}
|
|
virtual void serialize( std::ostream& s ) const
|
|
{
|
|
// if an error about an ambiguous conversion is generated by the
|
|
// line below the solution is to add a serialization operator to a
|
|
// mock::stream for T
|
|
s << *t_;
|
|
}
|
|
const T* t_;
|
|
};
|
|
|
|
struct data
|
|
{
|
|
template< typename T >
|
|
data( const T& t )
|
|
: h_( new holder_imp< T >( t ) )
|
|
{}
|
|
~data()
|
|
{
|
|
delete h_;
|
|
}
|
|
holder* h_;
|
|
};
|
|
}
|
|
|
|
inline stream& operator<<( stream& s, const detail3::data& d )
|
|
{
|
|
d.h_->serialize( *s.s_ );
|
|
return s;
|
|
}
|
|
|
|
#else // MOCK_USE_CONVERSIONS
|
|
|
|
namespace detail3
|
|
{
|
|
template< typename S, typename T >
|
|
S& operator<<( S &s, const T& )
|
|
{
|
|
return s << '?';
|
|
}
|
|
}
|
|
|
|
template< typename T >
|
|
stream& operator<<( stream& s, const T& t )
|
|
{
|
|
using namespace detail3;
|
|
*s.s_ << t;
|
|
return s;
|
|
}
|
|
|
|
#endif // MOCK_USE_CONVERSIONS
|
|
|
|
namespace detail2
|
|
{
|
|
template< typename T >
|
|
void serialize( stream& s, const T& t )
|
|
{
|
|
// if an error about an ambiguous conversion is generated by the
|
|
// line below the solution is to add a serialization operator to a
|
|
// mock::stream for T
|
|
s << t;
|
|
}
|
|
inline void serialize( stream& s, bool b )
|
|
{
|
|
s << (b ? "true" : "false");
|
|
}
|
|
template< typename C, typename T, typename A >
|
|
void serialize( stream& s, const std::basic_string< C, T, A >& str )
|
|
{
|
|
s << '"' << str << '"';
|
|
}
|
|
inline void serialize( stream& s, const char* const str )
|
|
{
|
|
s << '"' << str << '"';
|
|
}
|
|
inline void serialize( stream& s, unsigned char c )
|
|
{
|
|
s << static_cast< int >( c );
|
|
}
|
|
|
|
template< typename T >
|
|
struct formatter
|
|
{
|
|
explicit formatter( const T& t )
|
|
: t_( boost::addressof( t ) )
|
|
{}
|
|
void serialize( stream& s ) const
|
|
{
|
|
mock::detail2::serialize( s, *t_ );
|
|
}
|
|
const T* t_;
|
|
};
|
|
|
|
template< typename T >
|
|
stream& operator<<( stream& s, const formatter< T >& ser )
|
|
{
|
|
ser.serialize( s );
|
|
return s;
|
|
}
|
|
|
|
template< typename T >
|
|
std::ostream& operator<<( std::ostream& s, const formatter< T >& ser )
|
|
{
|
|
stream ss( s );
|
|
ser.serialize( ss );
|
|
return s;
|
|
}
|
|
}
|
|
|
|
template< typename T >
|
|
detail2::formatter< T > format( const T& t )
|
|
{
|
|
return detail2::formatter< T >( t );
|
|
}
|
|
|
|
template< typename T >
|
|
stream& operator<<( stream& s, const std::auto_ptr< T >& t )
|
|
{
|
|
return s << mock::format( t.get() );
|
|
}
|
|
template< typename T1, typename T2 >
|
|
stream& operator<<( stream& s, const std::pair< T1, T2 >& p )
|
|
{
|
|
return s << '(' << mock::format( p.first )
|
|
<< ',' << mock::format( p.second ) << ')';
|
|
}
|
|
|
|
namespace detail
|
|
{
|
|
template< typename T >
|
|
void serialize( stream& s, const T& begin, const T& end )
|
|
{
|
|
s << '(';
|
|
for( T it = begin; it != end; ++it )
|
|
s << (it == begin ? "" : ",") << mock::format( *it );
|
|
s << ')';
|
|
}
|
|
}
|
|
|
|
template< typename T, typename A >
|
|
stream& operator<<( stream& s, const std::deque< T, A >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename T, typename A >
|
|
stream& operator<<( stream& s, const std::list< T, A >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename T, typename A >
|
|
stream& operator<<( stream& s, const std::vector< T, A >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename K, typename T, typename C, typename A >
|
|
stream& operator<<( stream& s, const std::map< K, T, C, A >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename K, typename T, typename C, typename A >
|
|
stream& operator<<( stream& s, const std::multimap< K, T, C, A >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename T, typename C, typename A >
|
|
stream& operator<<( stream& s, const std::set< T, C, A >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename T, typename C, typename A >
|
|
stream& operator<<( stream& s, const std::multiset< T, C, A >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename T >
|
|
stream& operator<<( stream& s,
|
|
const boost::assign_detail::generic_list< T >& t )
|
|
{
|
|
mock::detail::serialize( s, t.begin(), t.end() );
|
|
return s;
|
|
}
|
|
template< typename T >
|
|
stream& operator<<( stream& s, boost::reference_wrapper< T > t )
|
|
{
|
|
return s << mock::format( t.get() );
|
|
}
|
|
template< typename T >
|
|
stream& operator<<( stream& s, boost::shared_ptr< T > t )
|
|
{
|
|
return s << mock::format( t.get() );
|
|
}
|
|
template< typename T >
|
|
stream& operator<<( stream& s, boost::weak_ptr< T > t )
|
|
{
|
|
return s << mock::format( t.lock() );
|
|
}
|
|
template< typename T >
|
|
stream& operator<<( stream& s, boost::lambda::lambda_functor< T > )
|
|
{
|
|
return s << '?';
|
|
}
|
|
template< typename T >
|
|
stream& operator<<( stream& s, boost::phoenix::actor< T > )
|
|
{
|
|
return s << '?';
|
|
}
|
|
template< typename T >
|
|
BOOST_DEDUCED_TYPENAME boost::enable_if<
|
|
boost::function_types::is_callable_builtin< T >,
|
|
stream&
|
|
>::type
|
|
operator<<( stream& s, T* )
|
|
{
|
|
return s << '?';
|
|
}
|
|
template< typename T >
|
|
BOOST_DEDUCED_TYPENAME boost::disable_if<
|
|
boost::function_types::is_callable_builtin< T >,
|
|
stream&
|
|
>::type
|
|
operator<<( stream& s, T* t )
|
|
{
|
|
*s.s_ << t;
|
|
return s;
|
|
}
|
|
} // mock
|
|
|
|
#endif // MOCK_LOG_HPP_INCLUDED
|