forked from catchorg/Catch2
		
	
		
			
				
	
	
		
			184 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  *  catch_objc.hpp
 | |
|  *  Catch
 | |
|  *
 | |
|  *  Created by Phil on 14/11/2010.
 | |
|  *  Copyright 2010 Two Blue Cubes Ltd. All rights reserved.
 | |
|  *
 | |
|  *  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 TWOBLUECUBES_CATCH_OBJC_HPP_INCLUDED
 | |
| #define TWOBLUECUBES_CATCH_OBJC_HPP_INCLUDED
 | |
| 
 | |
| #import <objc/runtime.h>
 | |
| #include <string>
 | |
| 
 | |
| #include "internal/catch_test_case_info.hpp"
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // This protocol is really only here for (self) documenting purposes, since
 | |
| // all its methods are optional.
 | |
| @protocol OcFixture
 | |
| 
 | |
| @optional
 | |
| 
 | |
| -(void) setUp;
 | |
| -(void) tearDown;
 | |
| 
 | |
| @end
 | |
| 
 | |
| namespace Catch 
 | |
| {
 | |
|     class OcMethod : public ITestCase
 | |
|     {
 | |
|     public:
 | |
|         ///////////////////////////////////////////////////////////////////////
 | |
|         OcMethod
 | |
|         (
 | |
|             Class cls, 
 | |
|             SEL sel
 | |
|         )
 | |
|         :   m_cls( cls ), 
 | |
|             m_sel( sel )
 | |
|         {
 | |
|         }
 | |
|         
 | |
|         ///////////////////////////////////////////////////////////////////////
 | |
|         virtual void invoke
 | |
|         ()
 | |
|         const
 | |
|         {
 | |
|             id obj = class_createInstance( m_cls, 0 );
 | |
|             obj = [obj init];
 | |
|             
 | |
|             if( [obj respondsToSelector: @selector(setUp) ] )
 | |
|                 [obj performSelector: @selector(setUp)];
 | |
| 
 | |
|             if( [obj respondsToSelector: m_sel] )
 | |
|                 [obj performSelector: m_sel];
 | |
| 
 | |
|             if( [obj respondsToSelector: @selector(tearDown) ] )
 | |
|                 [obj performSelector: @selector(tearDown)];
 | |
|             
 | |
|             [obj release];
 | |
|         }
 | |
|         
 | |
|         ///////////////////////////////////////////////////////////////////////
 | |
|         virtual ITestCase* clone
 | |
|         ()
 | |
|         const
 | |
|         {
 | |
|             return new OcMethod( m_cls, m_sel );
 | |
|         }
 | |
|         
 | |
|         ///////////////////////////////////////////////////////////////////////
 | |
|         virtual bool operator == 
 | |
|         (
 | |
|             const ITestCase& other
 | |
|         )
 | |
|         const
 | |
|         {
 | |
|             const OcMethod* ocmOther = dynamic_cast<const OcMethod*> ( &other );
 | |
|             return ocmOther && ocmOther->m_sel == m_sel;
 | |
|         }
 | |
|         
 | |
|         ///////////////////////////////////////////////////////////////////////
 | |
|         virtual bool operator < 
 | |
|         (
 | |
|             const ITestCase& other
 | |
|         )
 | |
|         const
 | |
|         {
 | |
|             const OcMethod* ocmOther = dynamic_cast<const OcMethod*> ( &other );
 | |
|             return ocmOther && ocmOther->m_sel < m_sel;
 | |
|         }
 | |
|         
 | |
|     private:
 | |
|         Class m_cls;
 | |
|         SEL m_sel;
 | |
|     };
 | |
|     
 | |
|     namespace Detail
 | |
|     {
 | |
|     
 | |
|         ///////////////////////////////////////////////////////////////////////
 | |
|         inline bool startsWith
 | |
|         (
 | |
|             const std::string& str, 
 | |
|             const std::string& sub
 | |
|         )
 | |
|         {
 | |
|             return str.length() > sub.length() && str.substr( 0, sub.length() ) == sub;
 | |
|         }
 | |
|         
 | |
|         ///////////////////////////////////////////////////////////////////////
 | |
|         inline const char* getAnnotation
 | |
|         (
 | |
|             Class cls, 
 | |
|             const std::string& annotationName, 
 | |
|             const std::string& testCaseName
 | |
|         )
 | |
|         {
 | |
|             NSString* selStr = [[NSString alloc] initWithFormat:@"Catch_%s_%s", annotationName.c_str(), testCaseName.c_str()];
 | |
|             SEL sel = NSSelectorFromString( selStr );
 | |
|             [selStr release];
 | |
|             if( [cls respondsToSelector: sel] )
 | |
|                 return (const char*)[cls performSelector: sel];
 | |
|             return "";
 | |
|         }        
 | |
|     }
 | |
|     
 | |
|     ///////////////////////////////////////////////////////////////////////////
 | |
|     inline size_t registerTestMethods
 | |
|     ()
 | |
|     {
 | |
|         size_t noTestMethods = 0;
 | |
|         int noClasses = objc_getClassList( NULL, 0 );
 | |
|         
 | |
|         std::vector<Class> classes( noClasses );
 | |
|         objc_getClassList( &classes[0], noClasses );
 | |
|         
 | |
|         for( int c = 0; c < noClasses; c++ )
 | |
|         {
 | |
|             Class cls = classes[c];
 | |
|             {
 | |
|                 u_int count;
 | |
|                 Method* methods = class_copyMethodList( cls, &count );
 | |
|                 for( int m = 0; m < count ; m++ )
 | |
|                 {
 | |
|                     SEL selector = method_getName(methods[m]);
 | |
|                     std::string methodName = sel_getName(selector);
 | |
|                     if( Detail::startsWith( methodName, "Catch_TestCase_" ) )
 | |
|                     {
 | |
|                         std::string testCaseName = methodName.substr( 15 );
 | |
|                         const char* name = Detail::getAnnotation( cls, "Name", testCaseName );
 | |
|                         const char* desc = Detail::getAnnotation( cls, "Description", testCaseName );
 | |
|                         
 | |
|                         Hub::getTestCaseRegistry().registerTest( TestCaseInfo( new OcMethod( cls, selector ), name, desc, "", 0 ) );
 | |
|                         noTestMethods++;
 | |
|                         
 | |
|                     }
 | |
|                 }
 | |
|                 free(methods);              
 | |
|             }
 | |
|         }
 | |
|         return noTestMethods;
 | |
|     }  
 | |
| }
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| #define OC_TEST_CASE( name, desc )\
 | |
| +(const char*) INTERNAL_CATCH_UNIQUE_NAME( Catch_Name_test ) \
 | |
| {\
 | |
| return name; \
 | |
| }\
 | |
| +(const char*) INTERNAL_CATCH_UNIQUE_NAME( Catch_Description_test ) \
 | |
| { \
 | |
| return desc; \
 | |
| } \
 | |
| -(void) INTERNAL_CATCH_UNIQUE_NAME( Catch_TestCase_test )
 | |
| 
 | |
| #endif // TWOBLUECUBES_CATCH_OBJC_HPP_INCLUDED
 |