Boost logo

Boost-Commit :

From: emil_at_[hidden]
Date: 2008-04-10 23:51:07


Author: emildotchevski
Date: 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
New Revision: 44164
URL: http://svn.boost.org/trac/boost/changeset/44164

Log:
to_string fixes
Added:
   trunk/boost/exception/detail/is_output_streamable.hpp (contents, props changed)
   trunk/libs/exception/test/is_output_streamable_test.cpp (contents, props changed)
   trunk/libs/exception/test/to_string_fail.cpp (contents, props changed)
   trunk/libs/exception/test/to_string_stub_test.cpp (contents, props changed)
Text files modified:
   trunk/boost/exception/to_string.hpp | 60 +++++++++++++++++++++++++++-----
   trunk/boost/exception/to_string_stub.hpp | 58 +------------------------------
   trunk/libs/exception/test/Jamfile.v2 | 8 +++
   trunk/libs/exception/test/to_string_test.cpp | 73 +++++++++------------------------------
   4 files changed, 78 insertions(+), 121 deletions(-)

Added: trunk/boost/exception/detail/is_output_streamable.hpp
==============================================================================
--- (empty file)
+++ trunk/boost/exception/detail/is_output_streamable.hpp 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -0,0 +1,50 @@
+//Copyright (c) 2006-2008 Emil Dotchevski and Reverge Studios, Inc.
+
+//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 UUID_898984B4076411DD973EDFA055D89593
+#define UUID_898984B4076411DD973EDFA055D89593
+
+#include <ostream>
+
+namespace
+boost
+ {
+ namespace
+ exception_detail
+ {
+ template <bool>
+ struct
+ is_output_streamable_dispatch
+ {
+ enum e { value=1 };
+ };
+
+ template <>
+ struct
+ is_output_streamable_dispatch<false>
+ {
+ enum e { value=0 };
+ };
+
+ template <class T,class CharT,class Traits>
+ char operator<<( std::basic_ostream<CharT,Traits> &, T const & );
+
+ template <class T,class CharT,class Traits>
+ struct
+ is_output_streamable_impl
+ {
+ enum e { value=is_output_streamable_dispatch<1!=sizeof((*(std::basic_ostream<CharT,Traits>*)0)<<(*(T*)0))>::value };
+ };
+ }
+
+ template <class T, class CharT=char, class Traits=std::char_traits<CharT> >
+ struct
+ is_output_streamable
+ {
+ enum e { value=exception_detail::is_output_streamable_impl<T,CharT,Traits>::value };
+ };
+ }
+
+#endif

Modified: trunk/boost/exception/to_string.hpp
==============================================================================
--- trunk/boost/exception/to_string.hpp (original)
+++ trunk/boost/exception/to_string.hpp 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -6,22 +6,62 @@
 #ifndef UUID_7E48761AD92811DC9011477D56D89593
 #define UUID_7E48761AD92811DC9011477D56D89593
 
+#include <boost/utility/enable_if.hpp>
+#include <boost/exception/detail/is_output_streamable.hpp>
 #include <sstream>
-#include <string>
 
 namespace
 boost
     {
- namespace
- {
+ namespace
+ exception_detail
+ {
+ template <bool>
+ struct
+ has_to_string_dispatch
+ {
+ enum e { value=1 };
+ };
+
+ template <>
+ struct
+ has_to_string_dispatch<false>
+ {
+ enum e { value=0 };
+ };
+
+ template <class T>
+ std::string
+ to_string( T const & x, typename enable_if< is_output_streamable<T> >::type * = 0 )
+ {
+ std::ostringstream out;
+ out << x;
+ return out.str();
+ }
+
         template <class T>
- std::string
- to_string( T const & x )
- {
- std::ostringstream out;
- out << x;
- return out.str();
- }
+ char to_string( T const &, typename disable_if< is_output_streamable<T> >::type * = 0 );
+
+ template <class T>
+ struct
+ has_to_string_impl
+ {
+ enum e { value=has_to_string_dispatch<1!=sizeof(to_string(*(T*)0))>::value };
+ };
+ }
+
+ template <class T>
+ struct
+ has_to_string
+ {
+ enum e { value=exception_detail::has_to_string_impl<T>::value };
+ };
+
+ template <class T>
+ std::string
+ to_string( T const & x, typename enable_if< is_output_streamable<T> >::type * = 0 )
+ {
+ return exception_detail::to_string(x);
         }
     }
 

Modified: trunk/boost/exception/to_string_stub.hpp
==============================================================================
--- trunk/boost/exception/to_string_stub.hpp (original)
+++ trunk/boost/exception/to_string_stub.hpp 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -16,53 +16,6 @@
     namespace
     exception_detail
         {
- template <bool ShiftLeftAvailable>
- struct shift_left_dispatcher;
-
- template <>
- struct
- shift_left_dispatcher<true>
- {
- template <class T,class CharT,class Traits,class Stub>
- static
- void
- convert( std::basic_ostream<CharT,Traits> & out, T const & x, Stub )
- {
- out << x;
- }
- };
-
- template <>
- struct
- shift_left_dispatcher<false>
- {
- template <class T,class CharT,class Traits,class Stub>
- static
- void
- convert( std::basic_ostream<CharT,Traits> & out, T const & x, Stub s )
- {
- out << s(x);
- }
- };
-
- namespace
- shift_left_dispatch
- {
- template <class T,class CharT,class Traits>
- char operator<<( std::basic_ostream<CharT,Traits> &, T );
-
- template <class T,class CharT,class Traits,class Stub>
- void
- dispatch( std::basic_ostream<CharT,Traits> & out, T const & x, Stub s )
- {
- shift_left_dispatcher<1!=sizeof(out<<x)>::convert(out,x,s);
- }
- }
- }
-
- namespace
- exception_detail
- {
         template <bool ToStringAvailable>
         struct
         to_string_dispatcher
@@ -82,26 +35,21 @@
             {
             template <class T,class Stub>
             static
- std::string
+ std::string
             convert( T const & x, Stub s )
                 {
- std::ostringstream out;
- shift_left_dispatch::dispatch(out,x,s);
- return out.str();
+ return s(x);
                 }
             };
 
         namespace
         to_string_dispatch
             {
- template <class T>
- char to_string( T );
-
             template <class T,class Stub>
             std::string
             dispatch( T const & x, Stub s )
                 {
- return to_string_dispatcher<1!=sizeof(to_string(x))>::convert(x,s);
+ return to_string_dispatcher<has_to_string<T>::value>::convert(x,s);
                 }
             }
 

Modified: trunk/libs/exception/test/Jamfile.v2
==============================================================================
--- trunk/libs/exception/test/Jamfile.v2 (original)
+++ trunk/libs/exception/test/Jamfile.v2 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -7,9 +7,15 @@
 
 import testing ;
 
+#to_string
+run is_output_streamable_test.cpp ;
+run to_string_test.cpp ;
+run to_string_stub_test.cpp ;
+compile-fail to_string_fail.cpp ;
+
+#exception
 run cloning_test.cpp ;
 run unknown_exception_test.cpp ;
-run to_string_test.cpp ;
 run exception_test.cpp ;
 run boost_error_info_test.cpp ;
 run enable_error_info_test.cpp helper1.cpp ;

Added: trunk/libs/exception/test/is_output_streamable_test.cpp
==============================================================================
--- (empty file)
+++ trunk/libs/exception/test/is_output_streamable_test.cpp 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -0,0 +1,41 @@
+//Copyright (c) 2006-2008 Emil Dotchevski and Reverge Studios, Inc.
+
+//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)
+
+#include <boost/exception/detail/is_output_streamable.hpp>
+#include <boost/detail/lightweight_test.hpp>
+
+namespace
+n1
+ {
+ class
+ c1
+ {
+ };
+ }
+
+namespace
+n2
+ {
+ class
+ c2
+ {
+ };
+
+ std::ostream &
+ operator<<( std::ostream & s, c2 const & )
+ {
+ s << "c2";
+ return s;
+ }
+ }
+
+int
+main()
+ {
+ BOOST_TEST( !boost::is_output_streamable<n1::c1>::value );
+ BOOST_TEST( boost::is_output_streamable<n2::c2>::value );
+ BOOST_TEST( boost::is_output_streamable<int>::value );
+ return boost::report_errors();
+ }

Added: trunk/libs/exception/test/to_string_fail.cpp
==============================================================================
--- (empty file)
+++ trunk/libs/exception/test/to_string_fail.cpp 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -0,0 +1,23 @@
+//Copyright (c) 2006-2008 Emil Dotchevski and Reverge Studios, Inc.
+
+//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)
+
+#include <boost/exception/to_string.hpp>
+
+namespace
+n1
+ {
+ struct
+ c1
+ {
+ };
+ }
+
+int
+tester()
+ {
+ using namespace boost;
+ (void) to_string(n1::c1());
+ return 1;
+ }

Added: trunk/libs/exception/test/to_string_stub_test.cpp
==============================================================================
--- (empty file)
+++ trunk/libs/exception/test/to_string_stub_test.cpp 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -0,0 +1,98 @@
+//Copyright (c) 2006-2008 Emil Dotchevski and Reverge Studios, Inc.
+
+//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)
+
+#include <boost/exception/to_string_stub.hpp>
+#include <boost/detail/lightweight_test.hpp>
+#include <iostream>
+
+namespace
+n1
+ {
+ class
+ c1
+ {
+ };
+
+ std::string
+ to_string( c1 const & )
+ {
+ return "c1";
+ }
+ }
+
+namespace
+n2
+ {
+ class
+ c2
+ {
+ };
+
+ std::ostream &
+ operator<<( std::ostream & s, c2 const & )
+ {
+ s << "c2";
+ return s;
+ }
+ }
+
+namespace
+n3
+ {
+ class
+ c3
+ {
+ };
+
+ std::ostream &
+ operator<<( std::ostream & s, c3 const & )
+ {
+ s << "bad";
+ return s;
+ }
+
+ std::string
+ to_string( c3 const & )
+ {
+ return "c3";
+ }
+ }
+
+namespace
+boost
+ {
+ class
+ to_string_tester
+ {
+ };
+ }
+
+template <class T>
+struct
+my_stub
+ {
+ std::string
+ operator()( T const & )
+ {
+ return "stub";
+ }
+ };
+
+int
+main()
+ {
+ using namespace boost;
+ BOOST_TEST( to_string(42)=="42" );
+ BOOST_TEST( to_string(n1::c1())=="c1" );
+ BOOST_TEST( to_string(n2::c2())=="c2" );
+ BOOST_TEST( to_string(n3::c3())=="c3" );
+ BOOST_TEST( to_string_stub(42)=="42" );
+ BOOST_TEST( to_string_stub(n1::c1())=="c1" );
+ BOOST_TEST( to_string_stub(n2::c2())=="c2" );
+ BOOST_TEST( to_string_stub(n3::c3())=="c3" );
+ BOOST_TEST( to_string_stub(to_string_tester(),my_stub<to_string_tester>())=="stub" );
+ BOOST_TEST( !to_string_stub(to_string_tester()).empty() );
+ return boost::report_errors();
+ }

Modified: trunk/libs/exception/test/to_string_test.cpp
==============================================================================
--- trunk/libs/exception/test/to_string_test.cpp (original)
+++ trunk/libs/exception/test/to_string_test.cpp 2008-04-10 23:51:06 EDT (Thu, 10 Apr 2008)
@@ -3,95 +3,58 @@
 //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)
 
-#include <boost/exception/to_string_stub.hpp>
+#include <boost/exception/to_string.hpp>
 #include <boost/detail/lightweight_test.hpp>
-#include <iostream>
 
 namespace
 n1
     {
- class
+ struct
     c1
         {
         };
-
- std::string
- to_string( c1 const & )
- {
- return "c1";
- }
     }
 
 namespace
 n2
     {
- class
+ struct
     c2
         {
         };
 
- std::ostream &
- operator<<( std::ostream & s, c2 const & )
+ std::string
+ to_string( c2 const & )
         {
- s << "c2";
- return s;
+ return "c2";
         }
     }
 
 namespace
 n3
     {
- class
+ struct
     c3
         {
         };
 
- std::ostream &
- operator<<( std::ostream & s, c3 const & )
- {
- s << "bad";
- return s;
- }
-
- std::string
- to_string( c3 const & )
+ std::ostream &
+ operator<<( std::ostream & s, c3 const & )
         {
- return "c3";
+ return s << "c3";
         }
     }
 
-namespace
-boost
- {
- class
- to_string_tester
- {
- };
- }
-
-template <class T>
-struct
-my_stub
- {
- std::string
- operator()( T const & )
- {
- return "stub";
- }
- };
-
 int
 main()
     {
- using namespace boost;
- BOOST_TEST( to_string(5)=="5" );
- BOOST_TEST( to_string(n1::c1())=="c1" );
- BOOST_TEST( to_string(n2::c2())=="c2" );
- BOOST_TEST( to_string(n3::c3())=="c3" );
- BOOST_TEST( to_string_stub(5)=="5" );
- BOOST_TEST( to_string_stub(n1::c1())=="c1" );
- BOOST_TEST( to_string_stub(n2::c2())=="c2" );
- BOOST_TEST( to_string_stub(n3::c3())=="c3" );
- BOOST_TEST( to_string_stub(to_string_tester(),my_stub<to_string_tester>())=="stub" );
+ using namespace boost;
+ BOOST_TEST( !has_to_string<n1::c1>::value );
+ BOOST_TEST( has_to_string<n2::c2>::value );
+ BOOST_TEST( has_to_string<n3::c3>::value );
+ BOOST_TEST( has_to_string<int>::value );
+ BOOST_TEST( "c2"==to_string(n2::c2()) );
+ BOOST_TEST( "c3"==to_string(n3::c3()) );
+ BOOST_TEST( "42"==to_string(42) );
     return boost::report_errors();
     }


Boost-Commit list run by bdawes at acm.org, david.abrahams at rcn.com, gregod at cs.rpi.edu, cpdaniel at pacbell.net, john at johnmaddock.co.uk