// (C) Copyright Gennadiy Rozental 2001-2008. // 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) // See http://www.boost.org/libs/test for the library home page. // // File : $RCSfile$ // // Version : $Revision$ // // Description : defines test_unit, test_case, test_case_results, test_suite and test_tree_visitor // *************************************************************************** #ifndef BOOST_TEST_UNIT_TEST_SUITE_IMPL_HPP_071894GER #define BOOST_TEST_UNIT_TEST_SUITE_IMPL_HPP_071894GER // Boost.Test #include #include #include #include #include #include #include // Boost #include #include #include #include #include // STL #include // for typeid #include // for std::string #include // for std::list #include // for std::vector #include //____________________________________________________________________________// namespace boost { namespace unit_test { // ************************************************************************** // // ************** test_unit ************** // // ************************************************************************** // class BOOST_TEST_DECL test_unit { public: enum { type = tut_any }; // Constructor test_unit( const_string tu_name, test_unit_type t ); // dependencies management void depends_on( test_unit* tu ); bool check_dependencies() const; // Public r/o properties typedef BOOST_READONLY_PROPERTY(test_unit_id,(framework_impl)) id_t; typedef BOOST_READONLY_PROPERTY(test_unit_id,(test_suite)) parent_id_t; readonly_property p_type; // type for this test unit readonly_property p_type_name; // "case"/"suite" id_t p_id; // unique id for this test unit parent_id_t p_parent_id; // parent test suite id // Public r/w properties readwrite_property p_name; // name for this test unit readwrite_property p_timeout; // timeout for the test unit execution readwrite_property p_expected_failures; // number of expected failures in this test unit mutable readwrite_property p_enabled; // enabled status for this unit void increase_exp_fail( unsigned num ); protected: ~test_unit(); private: // Data members std::list m_dependencies; }; // ************************************************************************** // // ************** test_case_generator ************** // // ************************************************************************** // class BOOST_TEST_DECL test_unit_generator { public: virtual test_unit* next() const = 0; protected: BOOST_TEST_PROTECTED_VIRTUAL ~test_unit_generator() {} }; // ************************************************************************** // // ************** test_case ************** // // ************************************************************************** // class BOOST_TEST_DECL test_case : public test_unit { public: enum { type = tut_case }; // Constructor test_case( const_string tc_name, callback0<> const& test_func ); // Access methods callback0<> const& test_func() const { return m_test_func; } private: friend class framework_impl; ~test_case() {} // BOOST_MSVC <= 1200 have problems with callback as property // Data members callback0<> m_test_func; }; // ************************************************************************** // // ************** test_suite ************** // // ************************************************************************** // class BOOST_TEST_DECL test_suite : public test_unit { public: enum { type = tut_suite }; // Constructor explicit test_suite( const_string ts_name ); // test unit list management void add( test_unit* tu, counter_t expected_failures = 0, unsigned timeout = 0 ); void add( test_unit_generator const& gen, unsigned timeout = 0 ); void remove( test_unit_id id ); // access methods test_unit_id get( const_string tu_name ) const; std::size_t size() const { return m_members.size(); } protected: friend BOOST_TEST_DECL void traverse_test_tree( test_suite const&, test_tree_visitor& ); friend class framework_impl; virtual ~test_suite() {} // Data members std::vector m_members; }; // ************************************************************************** // // ************** master_test_suite ************** // // ************************************************************************** // class BOOST_TEST_DECL master_test_suite_t : public test_suite { public: master_test_suite_t() : test_suite( "Master Test Suite" ) , argc( 0 ) , argv( 0 ) {} // Data members int argc; char** argv; }; // ************************************************************************** // // ************** test_tree_visitor ************** // // ************************************************************************** // class BOOST_TEST_DECL test_tree_visitor { public: // test tree visitor interface virtual void visit( test_case const& ) {} virtual bool test_suite_start( test_suite const& ) { return true; } virtual void test_suite_finish( test_suite const& ) {} protected: BOOST_TEST_PROTECTED_VIRTUAL ~test_tree_visitor() {} }; // ************************************************************************** // // ************** traverse_test_tree ************** // // ************************************************************************** // BOOST_TEST_DECL void traverse_test_tree( test_case const&, test_tree_visitor& ); BOOST_TEST_DECL void traverse_test_tree( test_suite const&, test_tree_visitor& ); BOOST_TEST_DECL void traverse_test_tree( test_unit_id , test_tree_visitor& ); //____________________________________________________________________________// inline void traverse_test_tree( test_unit const& tu, test_tree_visitor& V ) { if( tu.p_type == tut_case ) traverse_test_tree( static_cast( tu ), V ); else traverse_test_tree( static_cast( tu ), V ); } //____________________________________________________________________________// // ************************************************************************** // // ************** test_case_counter ************** // // ************************************************************************** // class test_case_counter : public test_tree_visitor { public: // Constructor test_case_counter() : p_count( 0 ) {} BOOST_READONLY_PROPERTY( counter_t, (test_case_counter)) p_count; private: // test tree visitor interface virtual void visit( test_case const& ); virtual bool test_suite_start( test_suite const& ts ) { return ts.p_enabled; } }; // ************************************************************************** // // ************** test_being_aborted ************** // // ************************************************************************** // struct BOOST_TEST_DECL test_being_aborted {}; // ************************************************************************** // // ************** object generators ************** // // ************************************************************************** // namespace ut_detail { BOOST_TEST_DECL std::string normalize_test_case_name( const_string tu_name ); template struct user_tc_method_invoker { typedef void (UserTestCase::*TestMethod )(); user_tc_method_invoker( shared_ptr inst, TestMethod test_method ) : m_inst( inst ), m_test_method( test_method ) {} void operator()() { ((*m_inst).*m_test_method)(); } shared_ptr m_inst; TestMethod m_test_method; }; } // namespace ut_detail //____________________________________________________________________________// inline test_case* make_test_case( callback0<> const& test_func, const_string tc_name ) { return new test_case( ut_detail::normalize_test_case_name( tc_name ), test_func ); } //____________________________________________________________________________// template inline test_case* make_test_case( void (UserTestCase::* test_method )(), const_string tc_name, boost::shared_ptr user_test_case ) { return new test_case( ut_detail::normalize_test_case_name( tc_name ), ut_detail::user_tc_method_invoker( user_test_case, test_method ) ); } //____________________________________________________________________________// // ************************************************************************** // // ************** auto_test_unit_registrar ************** // // ************************************************************************** // namespace ut_detail { struct BOOST_TEST_DECL auto_test_unit_registrar { // Constructors auto_test_unit_registrar( test_case* tc, counter_t exp_fail ); explicit auto_test_unit_registrar( const_string ts_name ); explicit auto_test_unit_registrar( test_unit_generator const& tc_gen ); explicit auto_test_unit_registrar( int ); private: static std::list& curr_ts_store(); }; //____________________________________________________________________________// template struct auto_tc_exp_fail { auto_tc_exp_fail() : m_value( 0 ) {} explicit auto_tc_exp_fail( unsigned v ) : m_value( v ) { instance() = this; } static auto_tc_exp_fail*& instance() { static auto_tc_exp_fail inst; static auto_tc_exp_fail* inst_ptr = &inst; return inst_ptr; } unsigned value() const { return m_value; } private: // Data members unsigned m_value; }; //____________________________________________________________________________// } // namespace ut_detail // ************************************************************************** // // ************** global_fixture ************** // // ************************************************************************** // class BOOST_TEST_DECL global_fixture : public test_observer { public: // Constructor global_fixture(); }; //____________________________________________________________________________// namespace ut_detail { template struct global_fixture_impl : public global_fixture { // Constructor global_fixture_impl(): m_fixure( 0 ) {} // test observer interface virtual void test_start( counter_t ) { m_fixure = new F; } virtual void test_finish() { delete m_fixure; m_fixure = 0; } virtual void test_aborted() { delete m_fixure; m_fixure = 0; } private: // Data members F* m_fixure; }; // ************************************************************************** // // ************** test_case_template_invoker ************** // // ************************************************************************** // template class test_case_template_invoker { public: void operator()() { TestCaseTemplate::run( (boost::type*)0 ); } }; // ************************************************************************** // // ************** generate_test_case_4_type ************** // // ************************************************************************** // template struct generate_test_case_4_type { explicit generate_test_case_4_type( const_string tc_name, Generator& G ) : m_test_case_name( tc_name ) , m_holder( G ) {} template void operator()( mpl::identity ) { std::string full_name; assign_op( full_name, m_test_case_name, 0 ); full_name += '<'; full_name += typeid(TestType).name(); if( boost::is_const::value ) full_name += " const"; full_name += '>'; m_holder.m_test_cases.push_back( new test_case( full_name, test_case_template_invoker() ) ); } private: // Data members const_string m_test_case_name; Generator& m_holder; }; // ************************************************************************** // // ************** test_case_template ************** // // ************************************************************************** // template class template_test_case_gen : public test_unit_generator { public: // Constructor template_test_case_gen( const_string tc_name ) { typedef generate_test_case_4_type, TestCaseTemplate > single_test_gen; mpl::for_each >( single_test_gen( tc_name, *this ) ); } virtual test_unit* next() const { if( m_test_cases.empty() ) return 0; test_unit* res = m_test_cases.front(); m_test_cases.pop_front(); return res; } // Data members mutable std::list m_test_cases; }; //____________________________________________________________________________// } // namespace ut_detail } // unit_test } // namespace boost #include #endif // BOOST_TEST_UNIT_TEST_SUITE_IMPL_HPP_071894GER