catch2/include/internal/catch_objc.hpp

207 lines
7.3 KiB
C++
Raw Normal View History

2010-11-14 23:47:30 +01:00
/*
* 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
#include "catch_objc_arc.hpp"
2010-11-14 23:47:30 +01:00
#import <objc/runtime.h>
2012-02-10 09:28:37 +01:00
2010-11-14 23:47:30 +01:00
#include <string>
2011-04-26 09:32:40 +02:00
// NB. Any general catch headers included here must be included
// in catch.hpp first to make sure they are included by the single
// header for non obj-usage
2012-08-14 20:30:30 +02:00
#include "internal/catch_test_case_info.h"
2010-11-14 23:47:30 +01:00
2011-02-08 09:42:05 +01:00
///////////////////////////////////////////////////////////////////////////////
// This protocol is really only here for (self) documenting purposes, since
// all its methods are optional.
2010-11-16 08:00:08 +01:00
@protocol OcFixture
@optional
-(void) setUp;
-(void) tearDown;
@end
2012-05-16 16:07:11 +02:00
namespace Catch {
2012-08-14 09:38:22 +02:00
class OcMethod : public SharedImpl<ITestCase> {
2010-11-14 23:47:30 +01:00
public:
2012-05-16 16:07:11 +02:00
OcMethod( Class cls, SEL sel ) : m_cls( cls ), m_sel( sel ) {}
2012-05-16 16:07:11 +02:00
virtual void invoke() const {
2012-03-17 19:20:06 +01:00
id obj = [[m_cls alloc] init];
2012-03-17 19:20:06 +01:00
performOptionalSelector( obj, @selector(setUp) );
performOptionalSelector( obj, m_sel );
performOptionalSelector( obj, @selector(tearDown) );
2012-03-17 19:20:06 +01:00
arcSafeRelease( obj );
2010-11-14 23:47:30 +01:00
}
private:
2012-08-14 09:38:22 +02:00
virtual ~OcMethod() {}
2010-11-16 08:00:08 +01:00
Class m_cls;
2010-11-14 23:47:30 +01:00
SEL m_sel;
};
2012-05-16 16:07:11 +02:00
namespace Detail{
inline bool startsWith( std::string const& str, std::string const& sub ) {
2010-11-14 23:47:30 +01:00
return str.length() > sub.length() && str.substr( 0, sub.length() ) == sub;
}
inline std::string getAnnotation( Class cls,
std::string const& annotationName,
std::string const& testCaseName ) {
2010-11-14 23:47:30 +01:00
NSString* selStr = [[NSString alloc] initWithFormat:@"Catch_%s_%s", annotationName.c_str(), testCaseName.c_str()];
SEL sel = NSSelectorFromString( selStr );
2012-03-17 19:20:06 +01:00
arcSafeRelease( selStr );
id value = performOptionalSelector( cls, sel );
if( value )
return [(NSString*)value UTF8String];
2010-11-14 23:47:30 +01:00
return "";
}
2010-11-16 08:00:08 +01:00
}
2013-03-21 09:58:22 +01:00
2012-05-16 16:07:11 +02:00
inline size_t registerTestMethods() {
2010-11-16 08:00:08 +01:00
size_t noTestMethods = 0;
int noClasses = objc_getClassList( NULL, 0 );
Class* classes = (CATCH_UNSAFE_UNRETAINED Class *)malloc( sizeof(Class) * noClasses);
2012-03-17 19:20:06 +01:00
objc_getClassList( classes, noClasses );
2012-05-16 16:07:11 +02:00
for( int c = 0; c < noClasses; c++ ) {
2010-11-16 08:00:08 +01:00
Class cls = classes[c];
{
u_int count;
Method* methods = class_copyMethodList( cls, &count );
for( u_int m = 0; m < count ; m++ ) {
2010-11-16 08:00:08 +01:00
SEL selector = method_getName(methods[m]);
std::string methodName = sel_getName(selector);
2012-05-16 16:07:11 +02:00
if( Detail::startsWith( methodName, "Catch_TestCase_" ) ) {
2010-11-16 08:00:08 +01:00
std::string testCaseName = methodName.substr( 15 );
2012-03-17 19:20:06 +01:00
std::string name = Detail::getAnnotation( cls, "Name", testCaseName );
std::string desc = Detail::getAnnotation( cls, "Description", testCaseName );
2013-02-19 09:39:28 +01:00
const char* className = class_getName( cls );
2013-02-19 09:39:28 +01:00
getMutableRegistryHub().registerTest( makeTestCase( new OcMethod( cls, selector ), className, name.c_str(), desc.c_str(), SourceLineInfo() ) );
2010-11-16 08:00:08 +01:00
noTestMethods++;
}
}
free(methods);
2010-11-16 08:00:08 +01:00
}
}
return noTestMethods;
}
2012-05-16 16:07:11 +02:00
namespace Matchers {
namespace Impl {
namespace NSStringMatchers {
template<typename MatcherT>
struct StringHolder : MatcherImpl<MatcherT, NSString*>{
StringHolder( NSString* substr ) : m_substr( [substr copy] ){}
StringHolder( StringHolder const& other ) : m_substr( [other.m_substr copy] ){}
2012-05-16 16:07:11 +02:00
StringHolder() {
2012-03-17 19:20:06 +01:00
arcSafeRelease( m_substr );
}
NSString* m_substr;
};
struct Equals : StringHolder<Equals> {
2012-03-14 21:04:50 +01:00
Equals( NSString* substr ) : StringHolder( substr ){}
virtual bool match( ExpressionType const& str ) const {
return (str != nil || m_substr == nil ) &&
[str isEqualToString:m_substr];
2012-03-14 21:04:50 +01:00
}
virtual std::string toString() const {
return "equals string: \"" + Catch::toString( m_substr ) + "\"";
2012-03-14 21:04:50 +01:00
}
};
struct Contains : StringHolder<Contains> {
Contains( NSString* substr ) : StringHolder( substr ){}
virtual bool match( ExpressionType const& str ) const {
return (str != nil || m_substr == nil ) &&
[str rangeOfString:m_substr].location != NSNotFound;
}
virtual std::string toString() const {
return "contains string: \"" + Catch::toString( m_substr ) + "\"";
}
};
struct StartsWith : StringHolder<StartsWith> {
StartsWith( NSString* substr ) : StringHolder( substr ){}
virtual bool match( ExpressionType const& str ) const {
return (str != nil || m_substr == nil ) &&
[str rangeOfString:m_substr].location == 0;
}
virtual std::string toString() const {
return "starts with: \"" + Catch::toString( m_substr ) + "\"";
}
};
struct EndsWith : StringHolder<EndsWith> {
EndsWith( NSString* substr ) : StringHolder( substr ){}
virtual bool match( ExpressionType const& str ) const {
return (str != nil || m_substr == nil ) &&
[str rangeOfString:m_substr].location == [str length] - [m_substr length];
}
virtual std::string toString() const {
return "ends with: \"" + Catch::toString( m_substr ) + "\"";
}
};
} // namespace NSStringMatchers
} // namespace Impl
2012-03-14 21:04:50 +01:00
inline Impl::NSStringMatchers::Equals
Equals( NSString* substr ){ return Impl::NSStringMatchers::Equals( substr ); }
inline Impl::NSStringMatchers::Contains
Contains( NSString* substr ){ return Impl::NSStringMatchers::Contains( substr ); }
inline Impl::NSStringMatchers::StartsWith
StartsWith( NSString* substr ){ return Impl::NSStringMatchers::StartsWith( substr ); }
inline Impl::NSStringMatchers::EndsWith
EndsWith( NSString* substr ){ return Impl::NSStringMatchers::EndsWith( substr ); }
} // namespace Matchers
using namespace Matchers;
} // namespace Catch
2010-11-14 23:47:30 +01:00
2011-02-08 09:42:05 +01:00
///////////////////////////////////////////////////////////////////////////////
2010-11-14 23:47:30 +01:00
#define OC_TEST_CASE( name, desc )\
2012-03-17 19:20:06 +01:00
+(NSString*) INTERNAL_CATCH_UNIQUE_NAME( Catch_Name_test ) \
2010-11-14 23:47:30 +01:00
{\
2012-03-17 19:20:06 +01:00
return @ name; \
2010-11-14 23:47:30 +01:00
}\
2012-03-17 19:20:06 +01:00
+(NSString*) INTERNAL_CATCH_UNIQUE_NAME( Catch_Description_test ) \
2010-11-14 23:47:30 +01:00
{ \
2012-03-17 19:20:06 +01:00
return @ desc; \
2010-11-14 23:47:30 +01:00
} \
-(void) INTERNAL_CATCH_UNIQUE_NAME( Catch_TestCase_test )
2012-05-10 22:46:46 +02:00
#endif // TWOBLUECUBES_CATCH_OBJC_HPP_INCLUDED