如何使用NSURLProtocol拦截iOS中的网络调用(REST)

时间:2014-01-15 15:29:35

标签: ios networking afnetworking intercept nsurlprotocol

我需要在呼叫成功完成时拦截客户端在应用程序中进行的网络呼叫。 我发现一个解决方案是实现NSURLProtocol抽象类并注册到应用程序。

这似乎解决了我的问题,但后来我撞墙,请求超时。 我使用AFNetworking最新的1.x版本,因为我需要支持iOS 5.0。

所以现在我需要找到如何延长初始请求的超时或者另一个更适合我问题的解决方案。

这就是我所做的。 NSURLProtocol实现:

@implementation SLKURLProtocol

+ (NSURLRequest*) canonicalRequestForRequest:(NSURLRequest *)request
    {
    return request;
    }

+ (BOOL) canInitWithRequest:(NSURLRequest *)request
    {
    return [[[request URL] scheme] isEqualToString:@"http"];
    }

- (void) startLoading
    {
    dispatch_async(kBgQueue,
        ^{
        NSLog(@"Intercept Request startLoading");
        NSURLRequest* request = self.request;
        NSLog(@"URL: %@", request.URL);
        NSLog(@"HTTP Method: %@", request.HTTPMethod);
        NSLog(@"Data: %@", [NSString stringWithUTF8String:[request.HTTPBody bytes]]);
        });
    }

- (void) stopLoading
    {
    dispatch_async(kBgQueue,
        ^{
        NSLog(@"Intercept Request stopLoading");
        });
    }

@end

网络ViewController:

@implementation SLKViewController

- (void) viewDidLoad
    {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    [NSURLProtocol registerClass:[SLKURLProtocol class]];
    [self initRequest];
    }

- (void) initRequest
    {
    NSMutableURLRequest* urlRequest = [[NSMutableURLRequest alloc] initWithURL:[NSURL URLWithString:@"http://requestb.in/19kcum21"]
            cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:10.0f];
    [urlRequest setHTTPMethod:@"POST"];

    NSMutableData* postData = [NSMutableData data];
    [postData appendData:[@"REQUEST AFNETWORKING" dataUsingEncoding:NSUTF8StringEncoding]];
    [urlRequest setHTTPBody:postData];
    AFHTTPRequestOperation* operation = [[AFHTTPRequestOperation alloc] initWithRequest:urlRequest];

    [operation setCompletionBlockWithSuccess:
        ^(AFHTTPRequestOperation *operation, id responseObject)
        {
        NSString* serverResponse = [operation responseString];
        NSLog(@"AFNetworking Response: %@", serverResponse);
        } failure:
            ^(AFHTTPRequestOperation *operation, NSError *error)
        {
        NSLog(@"AFNetworking Error: %@", error.description);
        }];

    [operation start];
    }

@end

这是日志输出:

  

2014-01-15 17:07:19.518 InterceptionHttp [510:1303]拦截请求   startLoading 2014-01-15 17:07:19.520 InterceptionHttp [510:1303]网址:   http://requestb.in/19kcum21 2014-01-15 17:07:19.521   InterceptionHttp [510:1303] HTTP方法:POST 2014-01-15 17:07:19.522   InterceptionHttp [510:1303]数据:REQUEST AFNETWORKING 2014-01-15   17:07:29.522 InterceptionHttp [510:400b]拦截请求stopLoading   2014-01-15 17:07:29.522 InterceptionHttp [510:70b] AFNetworking错误:   错误域= NSURLErrorDomain代码= -1001“请求超时。”   的UserInfo = 0x8a5cb10   {NSErrorFailingURLStringKey = http://requestb.in/19kcum21,   NSErrorFailingURLKey = http://requestb.in/19kcum21,   NSLocalizedDescription =请求超时。,   NSUnderlyingError = 0x8a5c220“请求超时。”}

修改

在jqyao回答并阅读文章后,我学到了一些关于子类化NSURLProtocol的主题。但不幸的是,我仍然无法拦截HTTP,这意味着我希望原始请求能够正常继续。

同时,如果我添加3行调用URLProtocol委托选择器,原始请求将继续,但是响应数据而不是服务器。

以下是我在startLoading方法中添加的内容。

[[self client] URLProtocol:self didReceiveResponse:[[NSURLResponse alloc] init] cacheStoragePolicy:NSURLCacheStorageNotAllowed];
[[self client] URLProtocol:self didLoadData:data];
[[self client] URLProtocolDidFinishLoading:self];

这是我的输出日志。

  

2014-02-04 17:27:22.253 InterceptionHttp [419:3a03]拦截请求   startLoading 2014-02-04 17:27:22.254 InterceptionHttp [419:3a03]网址:   http://requestb.in/tsvc14ts 2014-02-04 17:27:22.255   InterceptionHttp [419:3a03] HTTP方法:POST 2014-02-04 17:27:22.255   InterceptionHttp [419:3a03]数据:{'message':'JSON_MESSAGE'}   2014-02-04 17:27:22.258 InterceptionHttp [419:70b] AFNetworking   回复:{'message':'JSON_MESSAGE'} 2014-02-04 17:27:22.260   InterceptionHttp [419:1303]拦截请求stopLoading

2 个答案:

答案 0 :(得分:4)

我建议您查看一下这篇文章http://robnapier.net/offline-uiwebview-nsurlprotocol和实施https://github.com/rnapier/RNCachingURLProtocol。您经常看到超时问题吗?

答案 1 :(得分:3)

好的,我从这个非常酷的网络和数据调试组件https://github.com/square/PonyDebugger中借用了一些代码,并且工作非常顺利。

关于课程是https://github.com/square/PonyDebugger/blob/master/ObjC/PonyDebugger/PDNetworkDomainController.m

希望有助于其他人看到类似的东西。