//----------------------------------------------------------------------------- // boost-libs variant/test/recursive_variant_test.cpp source file // See http://www.boost.org for updates, documentation, and revision history. //----------------------------------------------------------------------------- // // Copyright (c) 2003 Eric Friedman, Itay Maman // Copyright (c) 2013 Antony Polukhin // // 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) // This file is used in two test cases: // // 1) recursive_variant_test.cpp that tests recursive usage of variant // // 2) variant_noexcept_test that tests Boost.Variant ability to compile // and work with disabled exceptions #ifdef BOOST_NO_EXCEPTIONS // `boost/test/minimal.hpp` cannot work with exceptions disabled, // so we need the following workarounds for that case: namespace boost { int exit_success = 0; } int test_main(int , char* []); int main( int argc, char* argv[] ) { return test_main(argc, argv); } #include #define BOOST_CHECK(exp) if (!(exp)) exit(EXIT_FAILURE) #else // BOOST_NO_EXCEPTIONS # include "boost/test/minimal.hpp" #endif // BOOST_NO_EXCEPTIONS #include "boost/variant.hpp" #include "boost/mpl/vector.hpp" #include "boost/mpl/copy.hpp" #include #include #include #include struct vector_printer : boost::static_visitor { template std::string operator()(const std::vector& vec) const { std::ostringstream ost; ost << "( "; typename std::vector::const_iterator it = vec.begin(); for (; it != vec.end(); ++it) ost << boost::apply_visitor( vector_printer(), *it ); ost << ") "; return ost.str(); } template std::string operator()(const T& operand) const { std::ostringstream ost; ost << operand << ' '; return ost.str(); } }; void test_recursive_variant() { typedef boost::make_recursive_variant< int , std::vector >::type var1_t; std::vector vec1; vec1.push_back(3); vec1.push_back(5); vec1.push_back(vec1); vec1.push_back(7); var1_t var1(vec1); std::string result1( boost::apply_visitor( vector_printer(), var1 ) ); std::cout << "result1: " << result1 << '\n'; BOOST_CHECK(result1 == "( 3 5 ( 3 5 ) 7 ) "); typedef boost::make_recursive_variant< boost::variant , std::vector >::type var2_t; std::vector vec2; vec2.push_back(boost::variant(3)); vec2.push_back(boost::variant(3.5)); vec2.push_back(vec2); vec2.push_back(boost::variant(7)); var2_t var2(vec2); std::string result2( boost::apply_visitor( vector_printer(), var2 ) ); std::cout << "result2: " << result2 << '\n'; BOOST_CHECK(result2 == "( 3 3.5 ( 3 3.5 ) 7 ) "); typedef boost::make_recursive_variant< int , std::vector< boost::variant< double , std::vector > > >::type var3_t; typedef boost::variant > var4_t; std::vector vec3; vec3.push_back(3); vec3.push_back(5); std::vector vec4; vec4.push_back(3.5); vec4.push_back(vec3); vec3.push_back(vec4); vec3.push_back(7); var4_t var4(vec3); std::string result3( boost::apply_visitor( vector_printer(), var4 ) ); std::cout << "result2: " << result3 << '\n'; BOOST_CHECK(result3 == "( 3 5 ( 3.5 ( 3 5 ) ) 7 ) "); typedef boost::make_recursive_variant< double, std::vector >::type var5_t; std::vector vec5; vec5.push_back(3.5); vec5.push_back(vec1); vec5.push_back(17.25); std::string result5( vector_printer()(vec5) ); std::cout << "result5: " << result5 << '\n'; BOOST_CHECK(result5 == "( 3.5 ( 3 5 ( 3 5 ) 7 ) 17.25 ) "); typedef boost::make_recursive_variant< int, std::map >::type var6_t; var6_t var6; } void test_recursive_variant_over() { typedef boost::make_recursive_variant_over< boost::mpl::vector< int , std::vector > >::type var1_t; std::vector vec1; vec1.push_back(3); vec1.push_back(5); vec1.push_back(vec1); vec1.push_back(7); var1_t var1(vec1); std::string result1( boost::apply_visitor( vector_printer(), var1 ) ); std::cout << "result1: " << result1 << '\n'; BOOST_CHECK(result1 == "( 3 5 ( 3 5 ) 7 ) "); typedef boost::make_recursive_variant_over< boost::mpl::vector< boost::make_variant_over >::type , std::vector > >::type var2_t; std::vector vec2; vec2.push_back(boost::variant(3)); vec2.push_back(boost::variant(3.5)); vec2.push_back(vec2); vec2.push_back(boost::variant(7)); var2_t var2(vec2); std::string result2( boost::apply_visitor( vector_printer(), var2 ) ); std::cout << "result2: " << result2 << '\n'; BOOST_CHECK(result2 == "( 3 3.5 ( 3 3.5 ) 7 ) "); typedef boost::make_recursive_variant_over< boost::mpl::vector< int , std::vector< boost::make_variant_over< boost::mpl::vector< double , std::vector > >::type > > >::type var3_t; typedef boost::make_variant_over< boost::mpl::copy< boost::mpl::vector< double , std::vector > >::type >::type var4_t; std::vector vec3; vec3.push_back(3); vec3.push_back(5); std::vector vec4; vec4.push_back(3.5); vec4.push_back(vec3); vec3.push_back(vec4); vec3.push_back(7); var4_t var3(vec3); std::string result3( boost::apply_visitor( vector_printer(), var3 ) ); std::cout << "result2: " << result3 << '\n'; BOOST_CHECK(result3 == "( 3 5 ( 3.5 ( 3 5 ) ) 7 ) "); typedef boost::make_recursive_variant_over< boost::mpl::vector< double , std::vector > >::type var5_t; std::vector vec5; vec5.push_back(3.5); vec5.push_back(vec1); vec5.push_back(17.25); std::string result5( vector_printer()(vec5) ); std::cout << "result5: " << result5 << '\n'; BOOST_CHECK(result5 == "( 3.5 ( 3 5 ( 3 5 ) 7 ) 17.25 ) "); } int test_main(int , char* []) { test_recursive_variant(); test_recursive_variant_over(); return boost::exit_success; }