diff --git a/Nocilla.xcodeproj/project.pbxproj b/Nocilla.xcodeproj/project.pbxproj index 15c1ae2c..e4d03e4d 100644 --- a/Nocilla.xcodeproj/project.pbxproj +++ b/Nocilla.xcodeproj/project.pbxproj @@ -171,7 +171,7 @@ A085B83B15E30749007D33C1 /* Foundation.framework */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = wrapper.framework; name = Foundation.framework; path = System/Library/Frameworks/Foundation.framework; sourceTree = SDKROOT; }; A085B83F15E30749007D33C1 /* Nocilla-Prefix.pch */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Nocilla-Prefix.pch"; sourceTree = ""; }; A085B84015E30749007D33C1 /* Nocilla.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = Nocilla.h; sourceTree = ""; }; - A085B84915E30749007D33C1 /* NocillaTests.octest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; name = NocillaTests.octest; path = NocillaTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; + A085B84915E30749007D33C1 /* NocillaTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = NocillaTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; A085B85415E30749007D33C1 /* NocillaTests-Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = "NocillaTests-Info.plist"; sourceTree = ""; }; A085B85615E30749007D33C1 /* en */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = en; path = en.lproj/InfoPlist.strings; sourceTree = ""; }; A085B85915E30749007D33C1 /* AFNetworkingStubbingSpec.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; name = AFNetworkingStubbingSpec.m; path = Hooks/NSURLRequest/AFNetworkingStubbingSpec.m; sourceTree = ""; }; @@ -405,7 +405,7 @@ isa = PBXGroup; children = ( A085B83815E30749007D33C1 /* libNocilla.a */, - A085B84915E30749007D33C1 /* NocillaTests.octest */, + A085B84915E30749007D33C1 /* NocillaTests.xctest */, ); name = Products; sourceTree = ""; @@ -600,7 +600,7 @@ ); name = NocillaTests; productName = NocillaTests; - productReference = A085B84915E30749007D33C1 /* NocillaTests.octest */; + productReference = A085B84915E30749007D33C1 /* NocillaTests.xctest */; productType = "com.apple.product-type.bundle.unit-test"; }; /* End PBXNativeTarget section */ diff --git a/Nocilla.xcodeproj/xcshareddata/xcschemes/Nocilla.xcscheme b/Nocilla.xcodeproj/xcshareddata/xcschemes/Nocilla.xcscheme index 7279bb02..a041d40c 100644 --- a/Nocilla.xcodeproj/xcshareddata/xcschemes/Nocilla.xcscheme +++ b/Nocilla.xcodeproj/xcshareddata/xcschemes/Nocilla.xcscheme @@ -29,7 +29,7 @@ @@ -47,7 +47,7 @@ @@ -63,6 +63,15 @@ ignoresPersistentStateOnLaunch = "NO" debugDocumentVersioning = "YES" allowLocationSimulation = "YES"> + + + + +#import "LSNocilla.h" @interface LSHTTPStubURLProtocol : NSURLProtocol ++ (void)registerNocilla:(LSNocilla *)nocilla; ++ (Class)randomSubclass; + @end diff --git a/Nocilla/Hooks/NSURLRequest/LSHTTPStubURLProtocol.m b/Nocilla/Hooks/NSURLRequest/LSHTTPStubURLProtocol.m index 4af21ce6..3bfb13ea 100644 --- a/Nocilla/Hooks/NSURLRequest/LSHTTPStubURLProtocol.m +++ b/Nocilla/Hooks/NSURLRequest/LSHTTPStubURLProtocol.m @@ -3,6 +3,8 @@ #import "NSURLRequest+LSHTTPRequest.h" #import "LSStubRequest.h" #import "NSURLRequest+DSL.h" +#import +#import @interface NSHTTPURLResponse(UndocumentedInitializer) - (id)initWithURL:(NSURL*)URL statusCode:(NSInteger)statusCode headerFields:(NSDictionary*)headerFields requestTime:(double)requestTime; @@ -10,6 +12,33 @@ - (id)initWithURL:(NSURL*)URL statusCode:(NSInteger)statusCode headerFields:(NSD @implementation LSHTTPStubURLProtocol +static NSMapTable *nocillaWeakMapTable; + ++ (void)load +{ + if (!nocillaWeakMapTable) { + nocillaWeakMapTable = [NSMapTable mapTableWithKeyOptions:NSPointerFunctionsCopyIn + valueOptions:NSPointerFunctionsWeakMemory]; + } +} + ++ (Class)randomSubclass +{ + Class baseClass = [self class]; + NSString * className = NSStringFromClass(baseClass); + + NSString * subclassName = [NSString stringWithFormat:@"%@%d", className, arc4random()]; + Class subclass = NSClassFromString(subclassName); + + if (subclass == nil) { + subclass = objc_allocateClassPair(baseClass, [subclassName UTF8String], 0); + if (subclass != nil) { + objc_registerClassPair(subclass); + } + } + return subclass; +} + + (BOOL)canInitWithRequest:(NSURLRequest *)request { return [@[ @"http", @"https" ] containsObject:request.URL.scheme]; } @@ -17,15 +46,27 @@ + (BOOL)canInitWithRequest:(NSURLRequest *)request { + (NSURLRequest *)canonicalRequestForRequest:(NSURLRequest *)request { return request; } + + (BOOL)requestIsCacheEquivalent:(NSURLRequest *)a toRequest:(NSURLRequest *)b { return NO; } ++ (void)registerNocilla:(LSNocilla *)nocilla +{ + NSString *className = NSStringFromClass([self class]); + [nocillaWeakMapTable setObject:nocilla forKey:className]; +} + ++ (LSNocilla *)nocilla +{ + return [nocillaWeakMapTable objectForKey:(NSStringFromClass([self class]))]; +} + - (void)startLoading { NSURLRequest* request = [self request]; id client = [self client]; - LSStubResponse* stubbedResponse = [[LSNocilla sharedInstance] responseForRequest:request]; + LSStubResponse* stubbedResponse = [[[self class] nocilla] responseForRequest:request]; if (stubbedResponse.shouldFail) { [client URLProtocol:self didFailWithError:stubbedResponse.error]; diff --git a/Nocilla/Hooks/NSURLRequest/LSNSURLHook.h b/Nocilla/Hooks/NSURLRequest/LSNSURLHook.h index abba2cbf..e17836b7 100644 --- a/Nocilla/Hooks/NSURLRequest/LSNSURLHook.h +++ b/Nocilla/Hooks/NSURLRequest/LSNSURLHook.h @@ -2,4 +2,6 @@ @interface LSNSURLHook : LSHTTPClientHook +@property(strong, nonatomic) Class urlProtocolClass; + @end diff --git a/Nocilla/Hooks/NSURLRequest/LSNSURLHook.m b/Nocilla/Hooks/NSURLRequest/LSNSURLHook.m index 77d70200..c7c6da99 100644 --- a/Nocilla/Hooks/NSURLRequest/LSNSURLHook.m +++ b/Nocilla/Hooks/NSURLRequest/LSNSURLHook.m @@ -4,11 +4,11 @@ @implementation LSNSURLHook - (void)load { - [NSURLProtocol registerClass:[LSHTTPStubURLProtocol class]]; + [NSURLProtocol registerClass:self.urlProtocolClass]; } - (void)unload { - [NSURLProtocol unregisterClass:[LSHTTPStubURLProtocol class]]; + [NSURLProtocol unregisterClass:self.urlProtocolClass]; } @end diff --git a/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.h b/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.h index aa6ec8c4..1f9e74e6 100644 --- a/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.h +++ b/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.h @@ -12,4 +12,6 @@ @interface LSNSURLSessionHook : LSHTTPClientHook +@property(strong, nonatomic) Class urlProtocolClass; + @end diff --git a/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.m b/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.m index ab584f59..6b528342 100644 --- a/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.m +++ b/Nocilla/Hooks/NSURLSession/LSNSURLSessionHook.m @@ -10,6 +10,10 @@ #import "LSHTTPStubURLProtocol.h" #import +@interface LSNSURLSessionHook () + +@end + @implementation LSNSURLSessionHook - (void)load { @@ -23,7 +27,6 @@ - (void)unload { } - (void)swizzleSelector:(SEL)selector fromClass:(Class)original toClass:(Class)stub { - Method originalMethod = class_getInstanceMethod(original, selector); Method stubMethod = class_getInstanceMethod(stub, selector); if (!originalMethod || !stubMethod) { @@ -33,8 +36,8 @@ - (void)swizzleSelector:(SEL)selector fromClass:(Class)original toClass:(Class)s } - (NSArray *)protocolClasses { - return @[[LSHTTPStubURLProtocol class]]; + Class stubUrlProtocolClass = self.urlProtocolClass; + return @[stubUrlProtocolClass]; } - @end diff --git a/Nocilla/LSNocilla.h b/Nocilla/LSNocilla.h index aa85901e..9ee8fa02 100644 --- a/Nocilla/LSNocilla.h +++ b/Nocilla/LSNocilla.h @@ -2,17 +2,23 @@ #import "Nocilla.h" @class LSStubRequest; +@class LSStubRequestDSL; @class LSStubResponse; @class LSHTTPClientHook; @protocol LSHTTPRequest; +@protocol LSMatcheable; extern NSString * const LSUnexpectedRequest; +typedef LSStubRequestDSL *(^StubRequestMethod)(NSString *method, id url); + @interface LSNocilla : NSObject + (LSNocilla *)sharedInstance; @property (nonatomic, strong, readonly) NSArray *stubbedRequests; @property (nonatomic, assign, readonly, getter = isStarted) BOOL started; +@property (nonatomic, strong, readonly) StubRequestMethod stubRequest; + - (void)start; - (void)stop; @@ -22,4 +28,5 @@ extern NSString * const LSUnexpectedRequest; - (void)registerHook:(LSHTTPClientHook *)hook; - (LSStubResponse *)responseForRequest:(id)request; + @end diff --git a/Nocilla/LSNocilla.m b/Nocilla/LSNocilla.m index 78ce6f5c..abb48cf7 100644 --- a/Nocilla/LSNocilla.m +++ b/Nocilla/LSNocilla.m @@ -1,10 +1,13 @@ #import "LSNocilla.h" #import "LSNSURLHook.h" #import "LSStubRequest.h" +#import "LSStubRequestDSL.h" +#import "LSStubResponseDSL.h" #import "LSHTTPRequestDSLRepresentation.h" #import "LSASIHTTPRequestHook.h" #import "LSNSURLSessionHook.h" #import "LSASIHTTPRequestHook.h" +#import "LSHTTPStubURLProtocol.h" NSString * const LSUnexpectedRequest = @"Unexpected Request"; @@ -34,9 +37,18 @@ - (id)init { if (self) { _mutableRequests = [NSMutableArray array]; _hooks = [NSMutableArray array]; - [self registerHook:[[LSNSURLHook alloc] init]]; + + Class stubUrlProtocolClass = [LSHTTPStubURLProtocol randomSubclass]; + [stubUrlProtocolClass registerNocilla:self]; + + LSNSURLHook *urlHook = [[LSNSURLHook alloc] init]; + urlHook.urlProtocolClass = stubUrlProtocolClass; + + [self registerHook:urlHook]; if (NSClassFromString(@"NSURLSession") != nil) { - [self registerHook:[[LSNSURLSessionHook alloc] init]]; + LSNSURLSessionHook *sessionHook = [[LSNSURLSessionHook alloc] init]; + sessionHook.urlProtocolClass = stubUrlProtocolClass; + [self registerHook:sessionHook]; } [self registerHook:[[LSASIHTTPRequestHook alloc] init]]; } @@ -69,7 +81,7 @@ - (void)clearStubs { } - (LSStubResponse *)responseForRequest:(id)actualRequest { - NSArray* requests = [LSNocilla sharedInstance].stubbedRequests; + NSArray* requests = self.stubbedRequests; for(LSStubRequest *someStubbedRequest in requests) { if ([someStubbedRequest matchesRequest:actualRequest]) { @@ -95,6 +107,17 @@ - (BOOL)hookWasRegistered:(LSHTTPClientHook *)aHook { } return NO; } + +- ( LSStubRequestDSL *(^)(NSString *method, id url))stubRequest +{ + return ^(NSString *method, id url){ + LSStubRequest *request = [[LSStubRequest alloc] initWithMethod:method urlMatcher:url.matcher]; + LSStubRequestDSL *dsl = [[LSStubRequestDSL alloc] initWithRequest:request]; + [self addStubbedRequest:request]; + return dsl; + }; +} + #pragma mark - Private - (void)loadHooks { for (LSHTTPClientHook *hook in self.hooks) {