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

Posted

技术标签:

【中文标题】如何使用 NSURLProtocol 在 iOS 中拦截网络调用(REST)【英文标题】:How to intercept networking calls (REST) in iOS using NSURLProtocol 【发布时间】:2014-02-04 03:55:39 【问题描述】:

我需要拦截客户端将在应用程序中进行的网络调用,同时调用将成功完成。 我发现一个解决方案是实现 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

网络视图控制器:

@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] URL: 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] 数据:请求 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 “请求超时。” 用户信息=0x8a5cb10 NSErrorFailingURLStringKey=http://requestb.in/19kcum21, NSErrorFailingURLKey=http://requestb.in/19kcum21, NSLocalizedDescription=请求超时。, NSUnderlyingError=0x8a5c220 "请求超时。"

编辑:

在jqyao回答和阅读文章之后,我学到了一些关于继承NSURLProtocol的话题。但是很遗憾我仍然无法拦截HTTP,也就是说我希望原始请求能够正常继续。

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

这是我在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] URL: 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 响应:'消息':'JSON_MESSAGE' 2014-02-04 17:27:22.260 InterceptionHttp[419:1303] 拦截请求停止加载

【问题讨论】:

【参考方案1】:

我建议你看看这篇文章http://robnapier.net/offline-uiwebview-nsurlprotocol 和实现https://github.com/rnapier/RNCachingURLProtocol。您多久看到一次超时问题?

【讨论】:

我已经阅读了这篇文章并遵循了一些代码,最终我发现自定义协议将在 ObjC 工作时以反向注册顺序执行。我尝试调用 URLProtocol 选择器,现在它到达 AFNetworking 报告响应,这是请求的数据!检查更新的答案。 RNCachingURLProtocol.m 是一个很好的拦截 http 的例子。在您的情况下,您需要添加代码 NSURLConnection *connection = [NSURLConnection connectionWithRequest:connectionRequest delegate:self]; [self setConnection:connection]; 以将连接转发到较低级别的网络堆栈。然后在每个委托函数中,您使用 NSURLProtocolClient 将数据传回给应用程序。 谢谢你的帮助,你能告诉我你的意思吗?在此处检查我的实施文件并根据您的建议进行任何更改。 gist.github.com/gtasko/8822394 按要点回答。 我复制了 RNCachingURLProtocol 但 POST 无法提供数据。虽然现在的反应是正确的!【参考方案2】:

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

相关类是https://github.com/square/PonyDebugger/blob/master/ObjC/PonyDebugger/PDNetworkDomainController.m

希望对其他看起来相似的人有所帮助。

【讨论】:

以上是关于如何使用 NSURLProtocol 在 iOS 中拦截网络调用(REST)的主要内容,如果未能解决你的问题,请参考以下文章

iOS Swift 如何告诉 UIView 控制器我们通过 NSURLProtocol 对象接收到信息

IOS网络篇1之截取本地URL请求(NSURLProtocol)

iOS - 将 AFNetworking 与自定义 NSURLProtocol 类一起使用

NSURLProtocol & swift - ios7 中的错误

iOS 开发之— NSURLProtocol

iOS H5容器的一些探究:iOS 下的黑魔法 NSURLProtocol