From 0b35a6a3dae1982935588174d79004bab32bcaff Mon Sep 17 00:00:00 2001 From: Peter Steinberger Date: Mon, 5 May 2014 10:43:24 +0200 Subject: [PATCH] Aspects 1.3.0. Optional, automatic hook de-registration. --- Aspects.h | 30 ++++--- Aspects.m | 89 ++++++++++++------- Aspects.podspec | 2 +- .../AspectsDemo.xcodeproj/project.pbxproj | 10 +++ AspectsDemo/AspectsDemo/AspectsAppDelegate.h | 2 +- AspectsDemo/AspectsDemo/AspectsAppDelegate.m | 21 +++-- .../AspectsDemo/AspectsViewController.h | 15 ++++ .../AspectsDemo/AspectsViewController.m | 34 +++++++ .../AspectsDemo/AspectsViewController.xib | 40 +++++++++ .../AspectsDemoTests/AspectsDemoTests.m | 81 ++++++++++++----- README.md | 9 +- 11 files changed, 255 insertions(+), 78 deletions(-) create mode 100644 AspectsDemo/AspectsDemo/AspectsViewController.h create mode 100644 AspectsDemo/AspectsDemo/AspectsViewController.m create mode 100644 AspectsDemo/AspectsDemo/AspectsViewController.xib diff --git a/Aspects.h b/Aspects.h index ddbb773..77c1340 100644 --- a/Aspects.h +++ b/Aspects.h @@ -7,10 +7,12 @@ #import -typedef NS_ENUM(NSUInteger, AspectPosition) { - AspectPositionBefore, /// Called before the original implementation. - AspectPositionInstead, /// Will replace the original implementation. - AspectPositionAfter /// Called after the original implementation. +typedef NS_OPTIONS(NSUInteger, AspectOptions) { + AspectPositionAfter = 0, /// Called after the original implementation (default) + AspectPositionInstead = 1, /// Will replace the original implementation. + AspectPositionBefore = 2, /// Called before the original implementation. + + AspectOptionAutomaticRemoval = 1 << 3 /// Will remove the hook after the first execution. }; /// Opaque Aspect Token that allows to deregister the hook. @@ -30,18 +32,19 @@ typedef NS_ENUM(NSUInteger, AspectPosition) { */ @interface NSObject (Aspects) -/// Adds a block of code before/instead/after the current `selector` for a specific object. +/// Adds a block of code before/instead/after the current `selector` for a specific class. /// If you choose `AspectPositionInstead`, the `arguments` array will contain the original invocation as last argument. +/// @note Hooking static methods is not supported. /// @return A token which allows to later deregister the aspect. -- (id)aspect_hookSelector:(SEL)selector - atPosition:(AspectPosition)position - withBlock:(void (^)(__unsafe_unretained id object, NSArray *arguments))block ++ (id)aspect_hookSelector:(SEL)selector + withOptions:(AspectOptions)options + usingBlock:(void (^)(id instance, NSArray *args))block error:(NSError **)error; -/// Hooks a selector class-wide. -+ (id)aspect_hookSelector:(SEL)selector - atPosition:(AspectPosition)position - withBlock:(void (^)(__unsafe_unretained id object, NSArray *arguments))block +/// Adds a block of code before/instead/after the current `selector` for a specific instance. +- (id)aspect_hookSelector:(SEL)selector + withOptions:(AspectOptions)options + usingBlock:(void (^)(id instance, NSArray *args))block error:(NSError **)error; @end @@ -49,7 +52,8 @@ typedef NS_ENUM(NSUInteger, AspectPosition) { typedef NS_ENUM(NSUInteger, AspectsErrorCode) { AspectsErrorSelectorBlacklisted, /// Selectors like release, retain, autorelease are blacklisted. - AspectsErrorSelectorDeallocPosition, /// When hooking dealloc, AspectPositionInstead is not allowed. + AspectsErrorDoesNotRespondToSelector, /// Selector could not be found. + AspectsErrorSelectorDeallocPosition, /// When hooking dealloc, only AspectPositionBefore is allowed. AspectsErrorSelectorAlreadyHookedInClassHierarchy, /// Statically hooking the same method in subclasses is not allowed. AspectsErrorFailedToAllocateClassPair, /// The runtime failed creating a class pair. AspectsErrorRemoveObjectAlreadyDeallocated = 100 /// (for removing) The object hooked is already deallocated. diff --git a/Aspects.m b/Aspects.m index ffedc96..9d31a91 100644 --- a/Aspects.m +++ b/Aspects.m @@ -12,18 +12,20 @@ #define AspectLog(...) //#define AspectLog(...) do { NSLog(__VA_ARGS__); }while(0) +#define AspectLogError(...) do { NSLog(__VA_ARGS__); }while(0) // Tracks a single aspect. @interface AspectIdentifier : NSObject -- (id)initWithSelector:(SEL)selector object:(id)object block:(id)block; +- (id)initWithSelector:(SEL)selector object:(id)object options:(AspectOptions)options block:(id)block; @property (nonatomic, assign) SEL selector; @property (nonatomic, strong) id block; @property (nonatomic, weak) id object; +@property (nonatomic, assign) AspectOptions options; @end // Tracks all aspects for an object/class. @interface AspectsContainer : NSObject -- (void)addAspect:(AspectIdentifier *)aspect atPosition:(AspectPosition)injectPosition; +- (void)addAspect:(AspectIdentifier *)aspect withOptions:(AspectOptions)injectPosition; - (BOOL)removeAspect:(id)aspect; - (BOOL)hasAspects; @property (atomic, copy) NSArray *beforeAspects; @@ -42,8 +44,12 @@ @interface NSInvocation (Aspects) - (NSArray *)aspects_arguments; @end +typedef void(^AspectBlock)(id instance, NSArray *arguments); + +#define AspectPositionFilter 0x07 + #define AspectError(errorCode, errorDescription) do { \ -AspectLog(@"Aspects: %@", errorDescription); \ +AspectLogError(@"Aspects: %@", errorDescription); \ if (error) { *error = [NSError errorWithDomain:AspectsErrorDomain code:errorCode userInfo:@{NSLocalizedDescriptionKey: errorDescription}]; }}while(0) NSString *const AspectsErrorDomain = @"AspectsErrorDomain"; @@ -55,34 +61,35 @@ @implementation NSObject (Aspects) /////////////////////////////////////////////////////////////////////////////////////////// #pragma mark - Public Aspects API -+ (id)aspect_hookSelector:(SEL)selector - atPosition:(AspectPosition)position - withBlock:(void (^)(__unsafe_unretained id object, NSArray *arguments))block - error:(NSError **)error { - return aspect_add((id)self, selector, position, block, error); ++ (id)aspect_hookSelector:(SEL)selector + withOptions:(AspectOptions)options + usingBlock:(AspectBlock)block + error:(NSError **)error { + return aspect_add((id)self, selector, options, block, error); } -- (id)aspect_hookSelector:(SEL)selector - atPosition:(AspectPosition)position - withBlock:(void (^)(__unsafe_unretained id object, NSArray *arguments))block - error:(NSError **)error { - return aspect_add(self, selector, position, block, error); +/// @return A token which allows to later deregister the aspect. +- (id)aspect_hookSelector:(SEL)selector + withOptions:(AspectOptions)options + usingBlock:(AspectBlock)block + error:(NSError **)error { + return aspect_add(self, selector, options, block, error); } /////////////////////////////////////////////////////////////////////////////////////////// #pragma mark - Private Helper -static id aspect_add(id self, SEL selector, AspectPosition position, void (^block)(__unsafe_unretained id object, NSArray *arguments), NSError **error) { +static id aspect_add(id self, SEL selector, AspectOptions options, AspectBlock block, NSError **error) { NSCParameterAssert(self); NSCParameterAssert(selector); NSCParameterAssert(block); __block AspectIdentifier *identifier = nil; aspect_performLocked(^{ - if (aspect_isSelectorAllowedAndTrack(self, selector, position, error)) { + if (aspect_isSelectorAllowedAndTrack(self, selector, options, error)) { AspectsContainer *aspectContainer = aspect_getContainerForObject(self, selector); - identifier = [[AspectIdentifier alloc] initWithSelector:selector object:self block:block];; - [aspectContainer addAspect:identifier atPosition:position]; + identifier = [[AspectIdentifier alloc] initWithSelector:selector object:self options:options block:block]; + [aspectContainer addAspect:identifier withOptions:options]; // Modify the class to allow message interception. aspect_prepareClassAndHookSelector(self, selector, error); @@ -337,7 +344,13 @@ static void aspect_undoSwizzleClassInPlace(Class klass) { #pragma mark - Aspect Invoke Point // This is a macro so we get a cleaner stack trace. -#define aspect_invoke(aspects, arguments) for (AspectIdentifier *aspect in aspects) {((void (^)(id, NSArray *))aspect.block)(self, arguments); } +#define aspect_invoke(aspects, arguments) \ +for (AspectIdentifier *aspect in aspects) {\ + ((void (^)(id, NSArray *))aspect.block)(self, arguments);\ + if (aspect.options & AspectOptionAutomaticRemoval) { \ + aspectsToRemove = [aspectsToRemove?:@[] arrayByAddingObject:aspect]; \ + } \ +} // This is the swizzled forwardInvocation: method. static void __ASPECTS_ARE_BEING_CALLED__(__unsafe_unretained NSObject *self, SEL selector, NSInvocation *invocation) { @@ -346,6 +359,7 @@ static void __ASPECTS_ARE_BEING_CALLED__(__unsafe_unretained NSObject *self, SEL SEL aliasSelector = aspect_aliasForSelector(invocation.selector); AspectsContainer *objectContainer = objc_getAssociatedObject(self, aliasSelector); AspectsContainer *classContainer = aspect_getContainerForClass(object_getClass(self), aliasSelector); + NSArray *aspectsToRemove = nil; // Before hooks. NSArray *arguments = nil; @@ -380,16 +394,16 @@ static void __ASPECTS_ARE_BEING_CALLED__(__unsafe_unretained NSObject *self, SEL // If no hooks are installed, call original implementation (usually to throw an exception) if (!respondsToAlias) { - SEL aspectsForwardInvocationSEL = NSSelectorFromString(AspectsForwardInvocationSelectorName); - if ([self respondsToSelector:aspectsForwardInvocationSEL]) { -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Warc-performSelector-leaks" - [self performSelector:aspectsForwardInvocationSEL withObject:invocation]; -#pragma clang diagnostic pop + SEL originalForwardInvocationSEL = NSSelectorFromString(AspectsForwardInvocationSelectorName); + if ([self respondsToSelector:originalForwardInvocationSEL]) { + ((void( *)(id, SEL, NSInvocation *))objc_msgSend)(self, originalForwardInvocationSEL, invocation); }else { [self doesNotRecognizeSelector:invocation.selector]; } } + + // Remove any hooks that are queued for deregistration. + [aspectsToRemove makeObjectsPerformSelector:@selector(remove)]; } #undef aspect_invoke @@ -437,7 +451,7 @@ static void aspect_destroyContainerForObject(id self, SEL selector) { return swizzledClassesDict; } -static BOOL aspect_isSelectorAllowedAndTrack(id self, SEL selector, AspectPosition position, NSError **error) { +static BOOL aspect_isSelectorAllowedAndTrack(NSObject *self, SEL selector, AspectOptions options, NSError **error) { static NSSet *disallowedSelectorList; static dispatch_once_t pred; dispatch_once(&pred, ^{ @@ -447,15 +461,22 @@ static BOOL aspect_isSelectorAllowedAndTrack(id self, SEL selector, AspectPositi // Check against the blacklist. NSString *selectorName = NSStringFromSelector(selector); if ([disallowedSelectorList containsObject:selectorName]) { - NSString *errorDescription = [NSString stringWithFormat:@"Selector `%@` is blacklisted.", selectorName]; + NSString *errorDescription = [NSString stringWithFormat:@"Selector %@ is blacklisted.", selectorName]; AspectError(AspectsErrorSelectorBlacklisted, errorDescription); return NO; } // Additional checks. - if ([selectorName isEqualToString:@"dealloc"] && position == AspectPositionInstead) { - NSString *errorDescription = @"dealloc can not be replaced. Use AspectPositionBefore."; - AspectError(AspectsErrorSelectorDeallocPosition, errorDescription); + AspectOptions position = options&AspectPositionFilter; + if ([selectorName isEqualToString:@"dealloc"] && position != AspectPositionBefore) { + NSString *errorDesc = @"AspectPositionBefore is the only valid position when hooking dealloc."; + AspectError(AspectsErrorSelectorDeallocPosition, errorDesc); + return NO; + } + + if (![self respondsToSelector:selector] && ![self.class instancesRespondToSelector:selector]) { + NSString *errorDesc = [NSString stringWithFormat:@"Unable to find selector -[%@ %@].", NSStringFromClass(self.class), selectorName]; + AspectError(AspectsErrorDoesNotRespondToSelector, errorDesc); return NO; } @@ -624,19 +645,20 @@ - (NSArray *)aspects_arguments { @implementation AspectIdentifier -- (id)initWithSelector:(SEL)selector object:(id)object block:(id)block { +- (id)initWithSelector:(SEL)selector object:(id)object options:(AspectOptions)options block:(id)block { NSCParameterAssert(block); NSCParameterAssert(selector); if (self = [super init]) { _selector = selector; _block = block; + _options = options; _object = object; // weak } return self; } - (NSString *)description { - return [NSString stringWithFormat:@"<%@: %p, SEL:%@ object:%@ block:%@>", self.class, self, NSStringFromSelector(self.selector), self.object, self.block]; + return [NSString stringWithFormat:@"<%@: %p, SEL:%@ object:%@ options:%tu block:%@>", self.class, self, NSStringFromSelector(self.selector), self.object, self.options, self.block]; } - (BOOL)remove { @@ -654,8 +676,9 @@ - (BOOL)hasAspects { return self.beforeAspects.count > 0 || self.insteadAspects.count > 0 || self.afterAspects.count > 0; } -- (void)addAspect:(AspectIdentifier *)aspect atPosition:(AspectPosition)injectPosition { - switch (injectPosition) { +- (void)addAspect:(AspectIdentifier *)aspect withOptions:(AspectOptions)options { + NSUInteger position = options&AspectPositionFilter; + switch (position) { case AspectPositionBefore: self.beforeAspects = [(self.beforeAspects ?:@[]) arrayByAddingObject:aspect]; break; case AspectPositionInstead: self.insteadAspects = [(self.insteadAspects?:@[]) arrayByAddingObject:aspect]; break; case AspectPositionAfter: self.afterAspects = [(self.afterAspects ?:@[]) arrayByAddingObject:aspect]; break; diff --git a/Aspects.podspec b/Aspects.podspec index 8f049cd..4548bb1 100644 --- a/Aspects.podspec +++ b/Aspects.podspec @@ -1,6 +1,6 @@ Pod::Spec.new do |s| s.name = "Aspects" - s.version = "1.2.0" + s.version = "1.3.0" s.summary = "Delightful, simple library for aspect oriented programming." s.homepage = "https://github.com/steipete/Aspects" s.license = { :type => 'MIT', :file => 'LICENSE' } diff --git a/AspectsDemo/AspectsDemo.xcodeproj/project.pbxproj b/AspectsDemo/AspectsDemo.xcodeproj/project.pbxproj index 76fcce9..9f9c17e 100644 --- a/AspectsDemo/AspectsDemo.xcodeproj/project.pbxproj +++ b/AspectsDemo/AspectsDemo.xcodeproj/project.pbxproj @@ -20,6 +20,8 @@ 78573F1819155A2E000D3B00 /* InfoPlist.strings in Resources */ = {isa = PBXBuildFile; fileRef = 78573F1619155A2E000D3B00 /* InfoPlist.strings */; }; 78573F1A19155A2E000D3B00 /* AspectsDemoTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 78573F1919155A2E000D3B00 /* AspectsDemoTests.m */; }; 78573F2519155A74000D3B00 /* Aspects.m in Sources */ = {isa = PBXBuildFile; fileRef = 78573F2319155A74000D3B00 /* Aspects.m */; }; + 78D7D77119177C8E002EB314 /* AspectsViewController.m in Sources */ = {isa = PBXBuildFile; fileRef = 78D7D76F19177C8E002EB314 /* AspectsViewController.m */; }; + 78D7D77219177C8E002EB314 /* AspectsViewController.xib in Resources */ = {isa = PBXBuildFile; fileRef = 78D7D77019177C8E002EB314 /* AspectsViewController.xib */; }; /* End PBXBuildFile section */ /* Begin PBXContainerItemProxy section */ @@ -51,6 +53,9 @@ 78573F1919155A2E000D3B00 /* AspectsDemoTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = AspectsDemoTests.m; sourceTree = ""; }; 78573F2319155A74000D3B00 /* Aspects.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = Aspects.m; path = ../../Aspects.m; sourceTree = ""; }; 78573F2419155A74000D3B00 /* Aspects.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = Aspects.h; path = ../../Aspects.h; sourceTree = ""; }; + 78D7D76E19177C8E002EB314 /* AspectsViewController.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = AspectsViewController.h; sourceTree = ""; }; + 78D7D76F19177C8E002EB314 /* AspectsViewController.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = AspectsViewController.m; sourceTree = ""; }; + 78D7D77019177C8E002EB314 /* AspectsViewController.xib */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.xib; path = AspectsViewController.xib; sourceTree = ""; }; /* End PBXFileReference section */ /* Begin PBXFrameworksBuildPhase section */ @@ -116,6 +121,9 @@ 78573F0419155A2E000D3B00 /* AspectsAppDelegate.m */, 78573F0619155A2E000D3B00 /* Images.xcassets */, 78573EFB19155A2E000D3B00 /* Supporting Files */, + 78D7D76E19177C8E002EB314 /* AspectsViewController.h */, + 78D7D76F19177C8E002EB314 /* AspectsViewController.m */, + 78D7D77019177C8E002EB314 /* AspectsViewController.xib */, ); path = AspectsDemo; sourceTree = ""; @@ -227,6 +235,7 @@ files = ( 78573EFF19155A2E000D3B00 /* InfoPlist.strings in Resources */, 78573F0719155A2E000D3B00 /* Images.xcassets in Resources */, + 78D7D77219177C8E002EB314 /* AspectsViewController.xib in Resources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -245,6 +254,7 @@ isa = PBXSourcesBuildPhase; buildActionMask = 2147483647; files = ( + 78D7D77119177C8E002EB314 /* AspectsViewController.m in Sources */, 78573F0519155A2E000D3B00 /* AspectsAppDelegate.m in Sources */, 78573F2519155A74000D3B00 /* Aspects.m in Sources */, 78573F0119155A2E000D3B00 /* main.m in Sources */, diff --git a/AspectsDemo/AspectsDemo/AspectsAppDelegate.h b/AspectsDemo/AspectsDemo/AspectsAppDelegate.h index 75137cf..d618e79 100644 --- a/AspectsDemo/AspectsDemo/AspectsAppDelegate.h +++ b/AspectsDemo/AspectsDemo/AspectsAppDelegate.h @@ -6,7 +6,7 @@ // Copyright (c) 2014 PSPDFKit GmbH. All rights reserved. // -#import +@import UIKit; @interface AspectsAppDelegate : UIResponder diff --git a/AspectsDemo/AspectsDemo/AspectsAppDelegate.m b/AspectsDemo/AspectsDemo/AspectsAppDelegate.m index e6ae4b6..1302f42 100644 --- a/AspectsDemo/AspectsDemo/AspectsAppDelegate.m +++ b/AspectsDemo/AspectsDemo/AspectsAppDelegate.m @@ -7,20 +7,29 @@ // #import "AspectsAppDelegate.h" +#import "AspectsViewController.h" #import "Aspects.h" @implementation AspectsAppDelegate - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { + AspectsViewController *aspectsController = [AspectsViewController new]; self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]]; - -// [self.window aspect_hookSelector:@selector(makeKeyAndVisible) atPosition:AspectPositionBefore withBlock:^(id object, NSArray *arguments) { -// NSLog(@"We're about to call -[UIWindow makeKeyAndVisible]."); -// }]; - - // Override point for customization after application launch. self.window.backgroundColor = [UIColor whiteColor]; + self.window.rootViewController = [[UINavigationController alloc] initWithRootViewController:aspectsController]; [self.window makeKeyAndVisible]; + + // Ignore hooks when we are testing. + if (!NSClassFromString(@"XCTestCase")) { + [aspectsController aspect_hookSelector:@selector(buttonPressed:) withOptions:0 usingBlock:^(id instance, NSArray *arguments) { + NSLog(@"Button was pressed by: %@", arguments.firstObject); + } error:NULL]; + + [aspectsController aspect_hookSelector:@selector(viewWillLayoutSubviews) withOptions:0 usingBlock:^(id instance, NSArray *arguments) { + NSLog(@"Controller is layouting!"); + } error:NULL]; + } + return YES; } diff --git a/AspectsDemo/AspectsDemo/AspectsViewController.h b/AspectsDemo/AspectsDemo/AspectsViewController.h new file mode 100644 index 0000000..ca11881 --- /dev/null +++ b/AspectsDemo/AspectsDemo/AspectsViewController.h @@ -0,0 +1,15 @@ +// +// AspectsViewController.h +// AspectsDemo +// +// Created by Peter Steinberger on 05/05/14. +// Copyright (c) 2014 PSPDFKit GmbH. All rights reserved. +// + +#import + +@interface AspectsViewController : UIViewController + +- (IBAction)buttonPressed:(id)sender; + +@end diff --git a/AspectsDemo/AspectsDemo/AspectsViewController.m b/AspectsDemo/AspectsDemo/AspectsViewController.m new file mode 100644 index 0000000..41d5d10 --- /dev/null +++ b/AspectsDemo/AspectsDemo/AspectsViewController.m @@ -0,0 +1,34 @@ +// +// AspectsViewController.m +// AspectsDemo +// +// Created by Peter Steinberger on 05/05/14. +// Copyright (c) 2014 PSPDFKit GmbH. All rights reserved. +// + +#import "AspectsViewController.h" +#import "Aspects.h" + +@implementation AspectsViewController + +- (IBAction)buttonPressed:(id)sender { + UIViewController *testController = [[UIImagePickerController alloc] init]; + + testController.modalPresentationStyle = UIModalPresentationFormSheet; + [self presentViewController:testController animated:YES completion:NULL]; + + // We are interested in being notified when the controller is being dismissed. + [testController aspect_hookSelector:@selector(viewWillDisappear:) withOptions:0 usingBlock:^(id instance, NSArray *arguments) { + UIViewController *controller = instance; + if (controller.isBeingDismissed || controller.isMovingFromParentViewController) { + [[[UIAlertView alloc] initWithTitle:@"Popped" message:@"Hello from Aspects" delegate:nil cancelButtonTitle:nil otherButtonTitles:@"Ok", nil] show]; + } + } error:NULL]; + + // Hooking dealloc is delicate, only AspectPositionBefore will work here. + [testController aspect_hookSelector:NSSelectorFromString(@"dealloc") withOptions:AspectPositionBefore usingBlock:^(id instance, NSArray *arguments) { + NSLog(@"Controller is about to be deallocated: %@", instance); + } error:NULL]; +} + +@end diff --git a/AspectsDemo/AspectsDemo/AspectsViewController.xib b/AspectsDemo/AspectsDemo/AspectsViewController.xib new file mode 100644 index 0000000..2e4af86 --- /dev/null +++ b/AspectsDemo/AspectsDemo/AspectsViewController.xib @@ -0,0 +1,40 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/AspectsDemo/AspectsDemoTests/AspectsDemoTests.m b/AspectsDemo/AspectsDemoTests/AspectsDemoTests.m index e5a6105..db15faa 100644 --- a/AspectsDemo/AspectsDemoTests/AspectsDemoTests.m +++ b/AspectsDemo/AspectsDemoTests/AspectsDemoTests.m @@ -73,7 +73,7 @@ - (void)testInsteadHook { UILabel *testLabel = [UILabel new]; testLabel.text = @"Default text"; XCTAssertEqualObjects(testLabel.text, @"Default text", @"Must match"); - id aspect = [testLabel aspect_hookSelector:@selector(text) atPosition:AspectPositionInstead withBlock:^(id object, NSArray *arguments) { + id aspect = [testLabel aspect_hookSelector:@selector(text) withOptions:AspectPositionInstead usingBlock:^(id instance, NSArray *arguments) { NSInvocation *invocation = arguments.lastObject; NSString *customText = @"Custom Text"; [invocation setReturnValue:&customText]; @@ -84,7 +84,7 @@ - (void)testInsteadHook { UILabel *testLabel2 = [UILabel new]; testLabel2.text = @"Default text2"; XCTAssertEqualObjects(testLabel2.text, @"Default text2", @"Must match"); - id aspect2 = [testLabel2 aspect_hookSelector:@selector(text) atPosition:AspectPositionInstead withBlock:^(id object, NSArray *arguments) { + id aspect2 = [testLabel2 aspect_hookSelector:@selector(text) withOptions:AspectPositionInstead usingBlock:^(id instance, NSArray *arguments) { NSInvocation *invocation = arguments.lastObject; NSString *customText = @"Custom Text2"; [invocation setReturnValue:&customText]; @@ -92,7 +92,7 @@ - (void)testInsteadHook { XCTAssertEqualObjects(testLabel2.text, @"Custom Text2", @"Must match"); // Globally override. - id globalAspect = [UILabel aspect_hookSelector:@selector(text) atPosition:AspectPositionInstead withBlock:^(id object, NSArray *arguments) { + id globalAspect = [UILabel aspect_hookSelector:@selector(text) withOptions:AspectPositionInstead usingBlock:^(id instance, NSArray *arguments) { NSInvocation *invocation = arguments.lastObject; NSString *customText = @"Global"; [invocation setReturnValue:&customText]; @@ -119,7 +119,7 @@ - (void)testAspectsCalledPerObject { TestClass *testClass = [TestClass new]; __block BOOL called = NO; - id aspect = [testClass aspect_hookSelector:@selector(testCall) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspect = [testClass aspect_hookSelector:@selector(testCall) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { called = YES; } error:NULL]; [testClass testCall]; @@ -139,13 +139,13 @@ - (void)testExecutionOrderAndMultipleRegistation { __block BOOL called_before = NO; __block BOOL called_after = NO; __block BOOL called_after2 = NO; - id aspect_before = [TestClass aspect_hookSelector:@selector(testCallAndExecuteBlock:) atPosition:AspectPositionBefore withBlock:^(id object, NSArray *arguments) { + id aspect_before = [TestClass aspect_hookSelector:@selector(testCallAndExecuteBlock:) withOptions:AspectPositionBefore usingBlock:^(id instance, NSArray *arguments) { called_before = YES; } error:NULL]; - id aspect_after = [TestClass aspect_hookSelector:@selector(testCallAndExecuteBlock:) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspect_after = [TestClass aspect_hookSelector:@selector(testCallAndExecuteBlock:) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { called_after2 = YES; } error:NULL]; - id aspect_after2 = [TestClass aspect_hookSelector:@selector(testCallAndExecuteBlock:) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspect_after2 = [TestClass aspect_hookSelector:@selector(testCallAndExecuteBlock:) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { called_after = YES; } error:NULL]; [testClass testCallAndExecuteBlock:^{ @@ -171,7 +171,7 @@ - (void)testExample { TestClass *testClass2 = [TestClass new]; __block BOOL testCallCalled = NO; - id aspectToken = [testClass aspect_hookSelector:@selector(testCall) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspectToken = [testClass aspect_hookSelector:@selector(testCall) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { testCallCalled = YES; } error:NULL]; @@ -185,7 +185,7 @@ - (void)testExample { - (void)testStructReturn { TestClass *testClass = [TestClass new]; CGRect rect = [testClass testThatReturnsAStruct]; - id aspect = [testClass aspect_hookSelector:@selector(testThatReturnsAStruct) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspect = [testClass aspect_hookSelector:@selector(testThatReturnsAStruct) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { } error:NULL]; CGRect rectHooked = [testClass testThatReturnsAStruct]; @@ -197,7 +197,7 @@ - (void)testHookReleaseIsNotAllowed { TestClass *testClass = [TestClass new]; __block BOOL testCallCalled = NO; - id aspectToken = [testClass aspect_hookSelector:NSSelectorFromString(@"release") atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspectToken = [testClass aspect_hookSelector:NSSelectorFromString(@"release") withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { testCallCalled = YES; } error:NULL]; XCTAssertNil(aspectToken, @"Token must be nil"); @@ -214,7 +214,7 @@ - (void)testDeallocHooking { TestClass *testClass = [TestClass new]; __block BOOL testCallCalled = NO; - __block id aspectToken = [testClass aspect_hookSelector:NSSelectorFromString(@"dealloc") atPosition:AspectPositionBefore withBlock:^(__unsafe_unretained id object, NSArray *arguments) { + __block id aspectToken = [testClass aspect_hookSelector:NSSelectorFromString(@"dealloc") withOptions:AspectPositionBefore usingBlock:^(id object, NSArray *arguments) { testCallCalled = YES; NSLog(@"called from dealloc"); } error:NULL]; @@ -230,7 +230,7 @@ - (void)testDeallocReplacing { NSError *error; __block BOOL deallocCalled = NO; - id aspectToken = [testClass aspect_hookSelector:NSSelectorFromString(@"dealloc") atPosition:AspectPositionInstead withBlock:^(__unsafe_unretained id object, NSArray *arguments) { + id aspectToken = [testClass aspect_hookSelector:NSSelectorFromString(@"dealloc") withOptions:AspectPositionInstead usingBlock:^(id object, NSArray *arguments) { deallocCalled = YES; NSLog(@"called from dealloc"); } error:&error]; @@ -241,6 +241,23 @@ - (void)testDeallocReplacing { XCTAssertFalse(deallocCalled, @"Dealloc-hook must not work."); } +- (void)testInvalidSelectorHooking { + TestClass *testClass = [TestClass new]; + NSError *error; + __block id aspectToken = [testClass aspect_hookSelector:NSSelectorFromString(@"fakeSelector") withOptions:AspectPositionBefore usingBlock:^(id object, NSArray *arguments) { + } error:&error]; + XCTAssertNil(aspectToken, @"Must return nil token."); + XCTAssertEqual(error.code, AspectsErrorDoesNotRespondToSelector, @"Error code must match"); +} + +- (void)testInvalidGlobalSelectorHooking { + NSError *error; + __block id aspectToken = [TestClass aspect_hookSelector:NSSelectorFromString(@"fakeSelector2") withOptions:AspectPositionBefore usingBlock:^(id object, NSArray *arguments) { + } error:&error]; + XCTAssertNil(aspectToken, @"Must return nil token."); + XCTAssertEqual(error.code, AspectsErrorDoesNotRespondToSelector, @"Error code must match"); +} + /////////////////////////////////////////////////////////////////////////////////////////// #pragma mark - Test Deregistration @@ -248,7 +265,7 @@ - (void)testInstanceTokenDeregistration { TestClass *testClass = [TestClass new]; __block BOOL testCallCalled = NO; - id aspectToken = [testClass aspect_hookSelector:@selector(testCall) atPosition:AspectPositionInstead withBlock:^(__unsafe_unretained id object, NSArray *arguments) { + id aspectToken = [testClass aspect_hookSelector:@selector(testCall) withOptions:AspectPositionInstead usingBlock:^(__unsafe_unretained id object, NSArray *arguments) { testCallCalled = YES; } error:NULL]; XCTAssertNotNil(aspectToken, @"Must return a token."); @@ -280,7 +297,7 @@ - (void)testGlobalTokenDeregistrationWithCustomForwardInvocation { } __block BOOL testCalled = NO; - id token = [TestWithCustomForwardInvocation aspect_hookSelector:@selector(test) atPosition:AspectPositionInstead withBlock:^(__unsafe_unretained id object, NSArray *arguments) { + id token = [TestWithCustomForwardInvocation aspect_hookSelector:@selector(test) withOptions:AspectPositionInstead usingBlock:^(__unsafe_unretained id object, NSArray *arguments) { testCalled = YES; } error:NULL]; XCTAssertNotNil(token, @"Must return a token."); @@ -322,7 +339,7 @@ - (void)testGlobalTokenDeregistration { } __block BOOL testCallCalled = NO; - id token = [TestClass aspect_hookSelector:@selector(testCall) atPosition:AspectPositionInstead withBlock:^(__unsafe_unretained id object, NSArray *arguments) { + id token = [TestClass aspect_hookSelector:@selector(testCall) withOptions:AspectPositionInstead usingBlock:^(__unsafe_unretained id object, NSArray *arguments) { testCallCalled = YES; } error:NULL]; XCTAssertNotNil(token, @"Must return a token."); @@ -359,7 +376,7 @@ - (void)testSimpleDeregistration { TestClass *testClass = [TestClass new]; __block BOOL called = NO; - id aspectToken = [TestClass aspect_hookSelector:@selector(testCall) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspectToken = [TestClass aspect_hookSelector:@selector(testCall) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { called = YES; } error:NULL]; [testClass testCall]; @@ -371,6 +388,24 @@ - (void)testSimpleDeregistration { XCTAssertFalse(called, @"Flag must have been NOT set."); } +- (void)testAutoDeregistration { + TestClass *testClass = [TestClass new]; + + __block BOOL testCallCalled = NO; + id aspectToken = [testClass aspect_hookSelector:@selector(testCall) withOptions:AspectPositionAfter|AspectOptionAutomaticRemoval usingBlock:^(id instance, NSArray *arguments) { + testCallCalled = YES; + } error:NULL]; + + [testClass testCall]; + XCTAssertTrue(testCallCalled, @"Must be set to YES"); + + testCallCalled = NO; + [testClass testCall]; + XCTAssertFalse(testCallCalled, @"Must be set to NO"); + + XCTAssertFalse([aspectToken remove], @"Must not able to deregister again"); +} + /////////////////////////////////////////////////////////////////////////////////////////// #pragma mark - Test KVO @@ -378,7 +413,7 @@ - (void)testKVOCoexistance { TestClass *testClass = [TestClass new]; __block BOOL hookCalled = NO; - id aspectToken = [testClass aspect_hookSelector:@selector(setString:) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { + id aspectToken = [testClass aspect_hookSelector:@selector(setString:) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { NSLog(@"Aspect hook!"); hookCalled = YES; } error:NULL]; @@ -407,7 +442,7 @@ - (void)testKVOCoexistance { // XCTAssertTrue(testClass.kvoTestCalled, @"KVO must work"); // // __block BOOL hookCalled = NO; -// [testClass aspect_hookSelector:@selector(setString:) atPosition:AspectPositionAfter withBlock:^(id object, NSArray *arguments) { +// [testClass aspect_hookSelector:@selector(setString:) withOptions:AspectPositionAfter usingBlock:^(id instance, NSArray *arguments) { // NSLog(@"Aspect hook!"); // hookCalled = YES; // }]; @@ -440,7 +475,7 @@ @implementation AspectsForwardInvocationTests - (void)testEnsureForwardInvocationIsCalled { TestWithCustomForwardInvocation *testClass = [TestWithCustomForwardInvocation new]; XCTAssertFalse(testClass.forwardInvocationCalled, @"Must have not called custom forwardInvocation:"); - id aspectToken = [TestWithCustomForwardInvocation aspect_hookSelector:@selector(test) atPosition:AspectPositionInstead withBlock:^(id object, NSArray *arguments) { + id aspectToken = [TestWithCustomForwardInvocation aspect_hookSelector:@selector(test) withOptions:AspectPositionInstead usingBlock:^(id instance, NSArray *arguments) { NSLog(@"Aspect hook called"); } error:NULL]; [testClass test]; @@ -483,11 +518,11 @@ @implementation AspectsSelectorTests //- (void)testSelectorMangling { // __block BOOL A_aspect_called = NO; // __block BOOL B_aspect_called = NO; -// [B aspect_hookSelector:@selector(foo) atPosition:AspectPositionBefore withBlock:^(id object, NSArray *arguments) { +// [B aspect_hookSelector:@selector(foo) withOptions:AspectPositionBefore usingBlock:^(id instance, NSArray *arguments) { // NSLog(@"before -[B foo]"); // B_aspect_called = YES; // }]; -// [A aspect_hookSelector:@selector(foo) atPosition:AspectPositionBefore withBlock:^(id object, NSArray *arguments) { +// [A aspect_hookSelector:@selector(foo) withOptions:AspectPositionBefore usingBlock:^(id instance, NSArray *arguments) { // NSLog(@"before -[A foo]"); // A_aspect_called = YES; // }]; @@ -503,13 +538,13 @@ @implementation AspectsSelectorTests - (void)testSelectorMangling2 { __block BOOL A_aspect_called = NO; __block BOOL B_aspect_called = NO; - id aspectToken1 = [A aspect_hookSelector:@selector(foo) atPosition:AspectPositionBefore withBlock:^(id object, NSArray *arguments) { + id aspectToken1 = [A aspect_hookSelector:@selector(foo) withOptions:AspectPositionBefore usingBlock:^(id instance, NSArray *arguments) { NSLog(@"before -[A foo]"); A_aspect_called = YES; } error:NULL]; XCTAssertNotNil(aspectToken1, @"Must return a token"); - id aspectToken2 = [B aspect_hookSelector:@selector(foo) atPosition:AspectPositionBefore withBlock:^(id object, NSArray *arguments) { + id aspectToken2 = [B aspect_hookSelector:@selector(foo) withOptions:AspectPositionBefore usingBlock:^(id instance, NSArray *arguments) { NSLog(@"before -[B foo]"); B_aspect_called = YES; } error:NULL]; diff --git a/README.md b/README.md index 8b93c56..4043446 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -Aspects v1.2.0 [![Build Status](https://travis-ci.org/steipete/Aspects.svg?branch=master)](https://travis-ci.org/steipete/Aspects) +Aspects v1.3.0 [![Build Status](https://travis-ci.org/steipete/Aspects.svg?branch=master)](https://travis-ci.org/steipete/Aspects) ============== Delightful, simple library for aspect oriented programming (AOP) by [@steipete](http://twitter.com/steipete). @@ -160,6 +160,13 @@ MIT licensed, Copyright (c) 2014 Peter Steinberger, steipete@gmail.com, [@steipe Release Notes ----------------- +Version 1.3.0 +- Add automatic deregistration. +- Checks if the selector exists before trying to hook. +- Improved dealloc hooking. (no more unsafe_unretained needed) +- Better examples. +- Always log errors. + Version 1.2.0 - Adds error parameter.