blob: 546e2a3d8563a4a926e1322eac3bfb04a4b8ba25 [file] [log] [blame]
//
// Boost.Pointer Container
//
// Copyright Thorsten Ottosen 2003-2005. Use, modification and
// distribution is subject to 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)
//
// For more information, see http://www.boost.org/libs/ptr_container/
//
#include <boost/config.hpp>
#ifdef BOOST_MSVC
#pragma warning( disable: 4996 )
#endif
#include <boost/test/unit_test.hpp>
#include <boost/archive/text_oarchive.hpp>
#include <boost/archive/text_iarchive.hpp>
#include <boost/archive/xml_iarchive.hpp>
#include <boost/archive/xml_oarchive.hpp>
#include <boost/functional/hash.hpp>
#include <boost/ptr_container/ptr_container.hpp>
#include <boost/ptr_container/serialize_ptr_container.hpp>
#include <boost/serialization/export.hpp>
#include <boost/serialization/base_object.hpp>
#include <boost/serialization/utility.hpp>
#include <boost/serialization/string.hpp>
#include <fstream>
#include <string>
//
// serialization helper: we can't save a non-const object
//
template< class T >
inline T const& as_const( T const& r )
{
return r;
}
//
// used to customize tests for circular_buffer
//
template< class Cont >
struct set_capacity
{
void operator()( Cont& ) const
{ }
};
template<class T>
struct set_capacity< boost::ptr_circular_buffer<T> >
{
void operator()( boost::ptr_circular_buffer<T>& c ) const
{
c.set_capacity( 100u );
}
};
//
// class hierarchy
//
struct Base
{
friend class boost::serialization::access;
int i;
template< class Archive >
void serialize( Archive& ar, const unsigned int /*version*/ )
{
ar & boost::serialization::make_nvp( "i", i );
}
Base() : i(42)
{ }
Base( int i ) : i(i)
{ }
virtual ~Base()
{ }
};
inline bool operator<( const Base& l, const Base& r )
{
return l.i < r.i;
}
inline bool operator==( const Base& l, const Base& r )
{
return l.i == r.i;
}
inline std::size_t hash_value( const Base& b )
{
return boost::hash_value( b.i );
}
struct Derived : Base
{
int i2;
template< class Archive >
void serialize( Archive& ar, const unsigned int /*version*/ )
{
ar & boost::serialization::make_nvp( "Base",
boost::serialization::base_object<Base>( *this ) );
ar & boost::serialization::make_nvp( "i2", i2 );
}
Derived() : Base(42), i2(42)
{ }
explicit Derived( int i2 ) : Base(0), i2(i2)
{ }
};
BOOST_CLASS_EXPORT_GUID( Derived, "Derived" )
//
// test of containers
//
//
template< class C, class T >
void add( C& c, T* r, unsigned /*n*/ )
{
c.insert( c.end(), r );
}
template< class U, class T >
void add( boost::ptr_array<U,2>& c, T* r, unsigned n )
{
c.replace( n, r );
}
template< class Cont, class OArchive, class IArchive >
void test_serialization_helper()
{
Cont vec;
set_capacity<Cont>()( vec );
add( vec, new Base( -1 ), 0u );
add( vec, new Derived( 1 ), 1u );
BOOST_CHECK_EQUAL( vec.size(), 2u );
std::ofstream ofs("filename");
OArchive oa(ofs);
oa << boost::serialization::make_nvp( "container", as_const(vec) );
ofs.close();
std::ifstream ifs("filename", std::ios::binary);
IArchive ia(ifs);
Cont vec2;
ia >> boost::serialization::make_nvp( "container", vec2 );
ifs.close();
BOOST_CHECK_EQUAL( vec.size(), vec2.size() );
BOOST_CHECK_EQUAL( (*vec2.begin()).i, -1 );
BOOST_CHECK_EQUAL( (*++vec2.begin()).i, 0 );
typename Cont::iterator i = vec2.begin();
++i;
Derived* d = dynamic_cast<Derived*>( &*i );
BOOST_CHECK_EQUAL( d->i2, 1 );
}
template< class Cont, class OArchive, class IArchive >
void test_serialization_unordered_set_helper()
{
Cont vec;
set_capacity<Cont>()( vec );
add( vec, new Base( -1 ), 0u );
add( vec, new Derived( 1 ), 1u );
BOOST_CHECK_EQUAL( vec.size(), 2u );
std::ofstream ofs("filename");
OArchive oa(ofs);
oa << boost::serialization::make_nvp( "container", as_const(vec) );
ofs.close();
std::ifstream ifs("filename", std::ios::binary);
IArchive ia(ifs);
Cont vec2;
ia >> boost::serialization::make_nvp( "container", vec2 );
ifs.close();
BOOST_CHECK_EQUAL( vec.size(), vec2.size() );
BOOST_CHECK_EQUAL( (*vec2.begin()).i, 0 );
BOOST_CHECK_EQUAL( (*++vec2.begin()).i, -1 );
}
template< class Map, class OArchive, class IArchive >
void test_serialization_map_helper()
{
Map m;
std::string key1("key1"), key2("key2");
m.insert( key1, new Base( -1 ) );
m.insert( key2, new Derived( 1 ) );
BOOST_CHECK_EQUAL( m.size(), 2u );
std::ofstream ofs("filename");
OArchive oa(ofs);
oa << boost::serialization::make_nvp( "container", as_const(m) );
ofs.close();
std::ifstream ifs("filename", std::ios::binary);
IArchive ia(ifs);
Map m2;
ia >> boost::serialization::make_nvp( "container", m2 );
ifs.close();
BOOST_CHECK_EQUAL( m.size(), m2.size() );
BOOST_CHECK_EQUAL( m2.find(key1)->second->i, -1 );
BOOST_CHECK_EQUAL( m2.find(key2)->second->i, 0 );
typename Map::iterator i = m2.find(key2);
Derived* d = dynamic_cast<Derived*>( i->second );
BOOST_CHECK_EQUAL( d->i2, 1 );
}
//
// basic test of hierarchy
//
void test_hierarchy()
{
Base* p = new Derived();
std::ofstream ofs("filename");
boost::archive::text_oarchive oa(ofs);
oa << as_const(p);
ofs.close();
Base* d = 0;
std::ifstream ifs("filename", std::ios::binary);
boost::archive::text_iarchive ia(ifs);
ia >> d;
ifs.close();
BOOST_CHECK_EQUAL( p->i, d->i );
BOOST_CHECK( p != d );
BOOST_CHECK( dynamic_cast<Derived*>( d ) );
delete p;
delete d;
}
//
// test initializer
//
void test_serialization()
{
test_hierarchy();
test_serialization_helper< boost::ptr_deque<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive >();
test_serialization_helper< boost::ptr_list<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_helper< boost::ptr_vector<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_helper< boost::ptr_vector<Base>,
boost::archive::xml_oarchive,
boost::archive::xml_iarchive>();
test_serialization_helper< boost::ptr_circular_buffer<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_helper< boost::ptr_circular_buffer<Base>,
boost::archive::xml_oarchive,
boost::archive::xml_iarchive>();
test_serialization_helper< boost::ptr_array<Base,2>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_helper< boost::ptr_set<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_helper< boost::ptr_multiset<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_unordered_set_helper< boost::ptr_unordered_set<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_unordered_set_helper<boost::ptr_unordered_multiset<Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_map_helper< boost::ptr_map<std::string,Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_map_helper< boost::ptr_multimap<std::string,Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_map_helper< boost::ptr_map<std::string,Base>,
boost::archive::xml_oarchive,
boost::archive::xml_iarchive>();
test_serialization_map_helper< boost::ptr_multimap<std::string,Base>,
boost::archive::xml_oarchive,
boost::archive::xml_iarchive>();
test_serialization_map_helper< boost::ptr_unordered_map<std::string,Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_map_helper< boost::ptr_unordered_multimap<std::string,Base>,
boost::archive::text_oarchive,
boost::archive::text_iarchive>();
test_serialization_map_helper< boost::ptr_unordered_map<std::string,Base>,
boost::archive::xml_oarchive,
boost::archive::xml_iarchive>();
test_serialization_map_helper< boost::ptr_unordered_multimap<std::string,Base>,
boost::archive::xml_oarchive,
boost::archive::xml_iarchive>();
}
using boost::unit_test::test_suite;
test_suite* init_unit_test_suite( int argc, char* argv[] )
{
test_suite* test = BOOST_TEST_SUITE( "Pointer Container Test Suite" );
test->add( BOOST_TEST_CASE( &test_serialization ) );
return test;
}