OSSNetworking.m 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. //
  2. // OSSNetworking.m
  3. // oss_ios_sdk
  4. //
  5. // Created by zhouzhuo on 8/16/15.
  6. // Copyright (c) 2015 aliyun.com. All rights reserved.
  7. //
  8. #import "OSSDefine.h"
  9. #import "OSSNetworking.h"
  10. #import "OSSBolts.h"
  11. #import "OSSModel.h"
  12. #import "OSSUtil.h"
  13. #import "OSSLog.h"
  14. #import "OSSXMLDictionary.h"
  15. #import "OSSInputStreamHelper.h"
  16. #import "OSSNetworkingRequestDelegate.h"
  17. #import "OSSURLRequestRetryHandler.h"
  18. #import "OSSHttpResponseParser.h"
  19. @implementation OSSNetworkingConfiguration
  20. @end
  21. @implementation OSSNetworking
  22. - (instancetype)initWithConfiguration:(OSSNetworkingConfiguration *)configuration {
  23. if (self = [super init]) {
  24. self.configuration = configuration;
  25. NSURLSessionConfiguration * conf = nil;
  26. NSOperationQueue *delegateQueue = [NSOperationQueue new];
  27. if (configuration.enableBackgroundTransmitService) {
  28. conf = [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:self.configuration.backgroundSessionIdentifier];
  29. } else {
  30. conf = [NSURLSessionConfiguration defaultSessionConfiguration];
  31. }
  32. conf.URLCache = nil;
  33. if (configuration.timeoutIntervalForRequest > 0) {
  34. conf.timeoutIntervalForRequest = configuration.timeoutIntervalForRequest;
  35. }
  36. if (configuration.timeoutIntervalForResource > 0) {
  37. conf.timeoutIntervalForResource = configuration.timeoutIntervalForResource;
  38. }
  39. if (configuration.HTTPMaximumConnectionsPerHost > 0) {
  40. conf.HTTPMaximumConnectionsPerHost = configuration.HTTPMaximumConnectionsPerHost;
  41. }
  42. if (configuration.proxyHost && configuration.proxyPort) {
  43. // Create an NSURLSessionConfiguration that uses the proxy
  44. NSDictionary *proxyDict = @{
  45. @"HTTPEnable" : [NSNumber numberWithInt:1],
  46. (NSString *)kCFStreamPropertyHTTPProxyHost : configuration.proxyHost,
  47. (NSString *)kCFStreamPropertyHTTPProxyPort : configuration.proxyPort,
  48. @"HTTPSEnable" : [NSNumber numberWithInt:1],
  49. (NSString *)kCFStreamPropertyHTTPSProxyHost : configuration.proxyHost,
  50. (NSString *)kCFStreamPropertyHTTPSProxyPort : configuration.proxyPort,
  51. };
  52. conf.connectionProxyDictionary = proxyDict;
  53. }
  54. _session = [NSURLSession sessionWithConfiguration:conf
  55. delegate:self
  56. delegateQueue:delegateQueue];
  57. self.isUsingBackgroundSession = configuration.enableBackgroundTransmitService;
  58. _sessionDelagateManager = [OSSSyncMutableDictionary new];
  59. NSOperationQueue * operationQueue = [NSOperationQueue new];
  60. if (configuration.maxConcurrentRequestCount > 0) {
  61. operationQueue.maxConcurrentOperationCount = configuration.maxConcurrentRequestCount;
  62. }
  63. self.taskExecutor = [OSSExecutor executorWithOperationQueue: operationQueue];
  64. }
  65. return self;
  66. }
  67. - (OSSTask *)sendRequest:(OSSNetworkingRequestDelegate *)request {
  68. OSSLogVerbose(@"send request --------");
  69. if (self.configuration.proxyHost && self.configuration.proxyPort) {
  70. request.isAccessViaProxy = YES;
  71. }
  72. /* set maximum retry */
  73. request.retryHandler.maxRetryCount = self.configuration.maxRetryCount;
  74. OSSTaskCompletionSource * taskCompletionSource = [OSSTaskCompletionSource taskCompletionSource];
  75. __weak OSSNetworkingRequestDelegate *weakRequest= request;
  76. request.completionHandler = ^(id responseObject, NSError * error) {
  77. [weakRequest reset];
  78. // 1.判断是否出错,如果出错的话,直接设置错误信息
  79. if (error)
  80. {
  81. [taskCompletionSource setError:error];
  82. }else
  83. {
  84. [self checkForCrc64WithResult:responseObject
  85. requestDelegate:weakRequest
  86. taskCompletionSource:taskCompletionSource];
  87. }
  88. };
  89. [self dataTaskWithDelegate:request];
  90. return taskCompletionSource.task;
  91. }
  92. - (void)checkForCrc64WithResult:(nonnull id)response requestDelegate:(OSSNetworkingRequestDelegate *)delegate taskCompletionSource:(OSSTaskCompletionSource *)source
  93. {
  94. OSSResult *result = (OSSResult *)response;
  95. BOOL hasRange = [delegate.internalRequest valueForHTTPHeaderField:@"Range"] != nil;
  96. NSURLComponents *urlComponents = [NSURLComponents componentsWithURL:delegate.internalRequest.URL resolvingAgainstBaseURL:YES];
  97. BOOL hasXOSSProcess = [urlComponents.query containsString:@"x-oss-process"];
  98. BOOL enableCRC = delegate.crc64Verifiable;
  99. // 3.判断如果未开启crc校验,或者headerFields里面有Range字段或者参数表中存在
  100. // x-oss-process字段,都将不进行crc校验
  101. if (!enableCRC || hasRange || hasXOSSProcess)
  102. {
  103. [source setResult:response];
  104. }
  105. else
  106. {
  107. OSSLogVerbose(@"--- checkForCrc64WithResult --- ");
  108. // 如果服务端未返回crc信息,默认是成功的
  109. OSSLogVerbose(@"result.remoteCRC64ecma : %@",result.remoteCRC64ecma);
  110. OSSLogVerbose(@"if result.localCRC64ecma : %@",result.localCRC64ecma);
  111. if (!result.remoteCRC64ecma.oss_isNotEmpty)
  112. {
  113. [source setResult:response];
  114. return;
  115. }
  116. // getObject 操作的crc数值会在delegate.responseParser consumeHttpResponseBody 进行计算。
  117. // upload & put 操作在上传成功后再计算。
  118. // 如果用户设置onReceiveData block。无法计算localCRC64ecma
  119. if (!result.localCRC64ecma.oss_isNotEmpty)
  120. {
  121. OSSLogVerbose(@"delegate.uploadingFileURL : %@",delegate.uploadingFileURL);
  122. if (delegate.uploadingFileURL)
  123. {
  124. OSSInputStreamHelper *helper = [[OSSInputStreamHelper alloc] initWithURL:delegate.uploadingFileURL];
  125. [helper syncReadBuffers];
  126. if (helper.crc64 != 0) {
  127. result.localCRC64ecma = [NSString stringWithFormat:@"%llu",helper.crc64];
  128. }
  129. }
  130. else
  131. {
  132. result.localCRC64ecma = delegate.contentCRC;
  133. }
  134. OSSLogVerbose(@"finally result.localCRC64ecma : %@",result.localCRC64ecma);
  135. }
  136. // 针对append接口,需要多次计算crc值
  137. if ([delegate.lastCRC oss_isNotEmpty] && [result.localCRC64ecma oss_isNotEmpty])
  138. {
  139. uint64_t last_crc64,local_crc64;
  140. NSScanner *scanner = [NSScanner scannerWithString:delegate.lastCRC];
  141. [scanner scanUnsignedLongLong:&last_crc64];
  142. scanner = [NSScanner scannerWithString:result.localCRC64ecma];
  143. [scanner scanUnsignedLongLong:&local_crc64];
  144. NSURLComponents *urlComponents = [NSURLComponents componentsWithURL:delegate.internalRequest.URL resolvingAgainstBaseURL:YES];
  145. NSArray<NSString *> *params = [urlComponents.query componentsSeparatedByString:@"&"];
  146. __block NSString *positionValue;
  147. [params enumerateObjectsUsingBlock:^(NSString *obj, NSUInteger idx, BOOL * _Nonnull stop) {
  148. if ([obj rangeOfString:@"position="].location == 0)
  149. {
  150. *stop = YES;
  151. positionValue = [obj substringFromIndex:9];
  152. }
  153. }];
  154. uint64_t position = [positionValue longLongValue];
  155. NSString *next_append_position = [result.httpResponseHeaderFields objectForKey:@"x-oss-next-append-position"];
  156. uint64_t length = [next_append_position longLongValue] - position;
  157. uint64_t crc_local = [OSSUtil crc64ForCombineCRC1:last_crc64 CRC2:local_crc64 length:(size_t)length];
  158. result.localCRC64ecma = [NSString stringWithFormat:@"%llu",crc_local];
  159. OSSLogVerbose(@"crc_local: %llu, crc_remote: %@,last_position: %llu,nextAppendPosition: %llu,length: %llu",crc_local,result.remoteCRC64ecma,position,[next_append_position longLongValue],length);
  160. }
  161. //如果服务器和本机计算的crc值不一致,则报crc校验失败;否则,认为上传任务执行成功
  162. if (result.remoteCRC64ecma.oss_isNotEmpty && result.localCRC64ecma.oss_isNotEmpty)
  163. {
  164. if ([result.remoteCRC64ecma isEqualToString:result.localCRC64ecma])
  165. {
  166. [source setResult:response];
  167. }else
  168. {
  169. NSString *errorMessage = [NSString stringWithFormat:@"crc validation fails(local_crc64ecma: %@,remote_crc64ecma: %@)",result.localCRC64ecma,result.remoteCRC64ecma];
  170. NSError *crcError = [NSError errorWithDomain:OSSClientErrorDomain
  171. code:OSSClientErrorCodeInvalidCRC
  172. userInfo:@{OSSErrorMessageTOKEN:errorMessage}];
  173. [source setError:crcError];
  174. }
  175. }
  176. else
  177. {
  178. [source setResult:response];
  179. }
  180. }
  181. }
  182. - (void)dataTaskWithDelegate:(OSSNetworkingRequestDelegate *)requestDelegate {
  183. [[[[[OSSTask taskWithResult:nil] continueWithExecutor:self.taskExecutor withSuccessBlock:^id(OSSTask *task) {
  184. OSSLogVerbose(@"start to intercept request");
  185. for (id<OSSRequestInterceptor> interceptor in requestDelegate.interceptors) {
  186. task = [interceptor interceptRequestMessage:requestDelegate.allNeededMessage];
  187. if (task.error) {
  188. return task;
  189. }
  190. }
  191. return task;
  192. }] continueWithSuccessBlock:^id(OSSTask *task) {
  193. return [requestDelegate buildInternalHttpRequest];
  194. }] continueWithSuccessBlock:^id(OSSTask *task) {
  195. NSURLSessionDataTask * sessionTask = nil;
  196. if (self.configuration.timeoutIntervalForRequest > 0) {
  197. requestDelegate.internalRequest.timeoutInterval = self.configuration.timeoutIntervalForRequest;
  198. }
  199. if (requestDelegate.uploadingData) {
  200. [requestDelegate.internalRequest setHTTPBody:requestDelegate.uploadingData];
  201. sessionTask = [_session dataTaskWithRequest:requestDelegate.internalRequest];
  202. } else if (requestDelegate.uploadingFileURL) {
  203. sessionTask = [_session uploadTaskWithRequest:requestDelegate.internalRequest fromFile:requestDelegate.uploadingFileURL];
  204. requestDelegate.isBackgroundUploadFileTask = self.isUsingBackgroundSession;
  205. } else { // not upload request
  206. sessionTask = [_session dataTaskWithRequest:requestDelegate.internalRequest];
  207. }
  208. requestDelegate.currentSessionTask = sessionTask;
  209. requestDelegate.httpRequestNotSuccessResponseBody = [NSMutableData new];
  210. [self.sessionDelagateManager setObject:requestDelegate forKey:@(sessionTask.taskIdentifier)];
  211. if (requestDelegate.isRequestCancelled) {
  212. return [OSSTask taskWithError:[NSError errorWithDomain:OSSClientErrorDomain
  213. code:OSSClientErrorCodeTaskCancelled
  214. userInfo:nil]];
  215. }
  216. [sessionTask resume];
  217. return task;
  218. }] continueWithBlock:^id(OSSTask *task) {
  219. // if error occurs before created sessionTask
  220. if (task.error) {
  221. requestDelegate.completionHandler(nil, task.error);
  222. } else if (task.isFaulted) {
  223. requestDelegate.completionHandler(nil, [NSError errorWithDomain:OSSClientErrorDomain
  224. code:OSSClientErrorCodeExcpetionCatched
  225. userInfo:@{OSSErrorMessageTOKEN: [NSString stringWithFormat:@"Catch exception - %@", task.exception]}]);
  226. }
  227. return nil;
  228. }];
  229. }
  230. #pragma mark - NSURLSessionTaskDelegate Methods
  231. - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)sessionTask didCompleteWithError:(NSError *)error
  232. {
  233. if (error) {
  234. OSSLogError(@"%@,error: %@", NSStringFromSelector(_cmd), error);
  235. }
  236. OSSNetworkingRequestDelegate * delegate = [self.sessionDelagateManager objectForKey:@(sessionTask.taskIdentifier)];
  237. [self.sessionDelagateManager removeObjectForKey:@(sessionTask.taskIdentifier)];
  238. NSHTTPURLResponse * httpResponse = (NSHTTPURLResponse *)sessionTask.response;
  239. if (delegate == nil) {
  240. OSSLogVerbose(@"delegate: %@", delegate);
  241. /* if the background transfer service is enable, may recieve the previous task complete callback */
  242. /* for now, we ignore it */
  243. return ;
  244. }
  245. /* background upload task will not call back didRecieveResponse */
  246. if (delegate.isBackgroundUploadFileTask) {
  247. OSSLogVerbose(@"backgroud upload task did recieve response: %@", httpResponse);
  248. if (httpResponse.statusCode >= 200 && httpResponse.statusCode < 300 && httpResponse.statusCode != 203) {
  249. [delegate.responseParser consumeHttpResponse:httpResponse];
  250. } else {
  251. delegate.isHttpRequestNotSuccessResponse = YES;
  252. }
  253. }
  254. [[[[OSSTask taskWithResult:nil] continueWithSuccessBlock:^id(OSSTask * task) {
  255. if (!delegate.error) {
  256. delegate.error = error;
  257. }
  258. if (delegate.error) {
  259. OSSLogDebug(@"networking request completed with error: %@", error);
  260. if ([delegate.error.domain isEqualToString:NSURLErrorDomain] && delegate.error.code == NSURLErrorCancelled) {
  261. return [OSSTask taskWithError:[NSError errorWithDomain:OSSClientErrorDomain
  262. code:OSSClientErrorCodeTaskCancelled
  263. userInfo:[error userInfo]]];
  264. } else {
  265. NSMutableDictionary * userInfo = [NSMutableDictionary dictionaryWithDictionary:[error userInfo]];
  266. [userInfo setObject:[NSString stringWithFormat:@"%ld", (long)error.code] forKey:@"OriginErrorCode"];
  267. return [OSSTask taskWithError:[NSError errorWithDomain:OSSClientErrorDomain
  268. code:OSSClientErrorCodeNetworkError
  269. userInfo:userInfo]];
  270. }
  271. }
  272. return task;
  273. }] continueWithSuccessBlock:^id(OSSTask *task) {
  274. if (delegate.isHttpRequestNotSuccessResponse) {
  275. if (httpResponse.statusCode == 0) {
  276. return [OSSTask taskWithError:[NSError errorWithDomain:OSSClientErrorDomain
  277. code:OSSClientErrorCodeNetworkingFailWithResponseCode0
  278. userInfo:@{OSSErrorMessageTOKEN: @"Request failed, response code 0"}]];
  279. }
  280. NSString * notSuccessResponseBody = [[NSString alloc] initWithData:delegate.httpRequestNotSuccessResponseBody encoding:NSUTF8StringEncoding];
  281. OSSLogError(@"http error response: %@", notSuccessResponseBody);
  282. NSDictionary * dict = [NSDictionary oss_dictionaryWithXMLString:notSuccessResponseBody];
  283. return [OSSTask taskWithError:[NSError errorWithDomain:OSSServerErrorDomain
  284. code:(-1 * httpResponse.statusCode)
  285. userInfo:dict]];
  286. }
  287. return task;
  288. }] continueWithBlock:^id(OSSTask *task) {
  289. if (task.error) {
  290. OSSNetworkingRetryType retryType = [delegate.retryHandler shouldRetry:delegate.currentRetryCount
  291. requestDelegate:delegate
  292. response:httpResponse
  293. error:task.error];
  294. OSSLogVerbose(@"current retry count: %u, retry type: %d", delegate.currentRetryCount, (int)retryType);
  295. switch (retryType) {
  296. case OSSNetworkingRetryTypeShouldNotRetry: {
  297. delegate.completionHandler(nil, task.error);
  298. return nil;
  299. }
  300. case OSSNetworkingRetryTypeShouldCorrectClockSkewAndRetry: {
  301. /* correct clock skew */
  302. NSString * dateStr = [[httpResponse allHeaderFields] objectForKey:@"Date"];
  303. if ([dateStr length] > 0) {
  304. NSDate * serverTime = [NSDate oss_dateFromString:dateStr];
  305. NSDate * deviceTime = [NSDate date];
  306. NSTimeInterval skewTime = [deviceTime timeIntervalSinceDate:serverTime];
  307. [NSDate oss_setClockSkew:skewTime];
  308. } else if (!error) {
  309. // The response header does not have the 'Date' field.
  310. // This should not happen.
  311. OSSLogError(@"Date header does not exist, unable to fix the clock skew");
  312. }
  313. [delegate.interceptors insertObject:[OSSTimeSkewedFixingInterceptor new] atIndex:0];
  314. break;
  315. }
  316. default:
  317. break;
  318. }
  319. /* now, should retry */
  320. NSTimeInterval suspendTime = [delegate.retryHandler timeIntervalForRetry:delegate.currentRetryCount retryType:retryType];
  321. delegate.currentRetryCount++;
  322. [NSThread sleepForTimeInterval:suspendTime];
  323. if(delegate.retryCallback){
  324. delegate.retryCallback();
  325. }
  326. /* retry recursively */
  327. [delegate reset];
  328. [self dataTaskWithDelegate:delegate];
  329. } else {
  330. delegate.completionHandler([delegate.responseParser constructResultObject], nil);
  331. }
  332. return nil;
  333. }];
  334. }
  335. - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didSendBodyData:(int64_t)bytesSent totalBytesSent:(int64_t)totalBytesSent totalBytesExpectedToSend:(int64_t)totalBytesExpectedToSend
  336. {
  337. OSSNetworkingRequestDelegate * delegate = [self.sessionDelagateManager objectForKey:@(task.taskIdentifier)];
  338. if (delegate.uploadProgress)
  339. {
  340. delegate.uploadProgress(bytesSent, totalBytesSent, totalBytesExpectedToSend);
  341. }
  342. }
  343. - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge completionHandler:(void (^)(NSURLSessionAuthChallengeDisposition disposition, NSURLCredential * __nullable credential))completionHandler
  344. {
  345. if (!challenge) {
  346. return;
  347. }
  348. NSURLSessionAuthChallengeDisposition disposition = NSURLSessionAuthChallengePerformDefaultHandling;
  349. NSURLCredential *credential = nil;
  350. /*
  351. * Gets the host name
  352. */
  353. NSString * host = [[task.currentRequest allHTTPHeaderFields] objectForKey:@"Host"];
  354. if (!host) {
  355. host = task.currentRequest.URL.host;
  356. }
  357. if ([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust]) {
  358. if ([self evaluateServerTrust:challenge.protectionSpace.serverTrust forDomain:host]) {
  359. disposition = NSURLSessionAuthChallengeUseCredential;
  360. credential = [NSURLCredential credentialForTrust:challenge.protectionSpace.serverTrust];
  361. }
  362. } else {
  363. disposition = NSURLSessionAuthChallengePerformDefaultHandling;
  364. }
  365. // Uses the default evaluation for other challenges.
  366. completionHandler(disposition,credential);
  367. }
  368. - (void)URLSessionDidFinishEventsForBackgroundURLSession:(NSURLSession *)session
  369. {
  370. }
  371. - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task
  372. willPerformHTTPRedirection:(NSHTTPURLResponse *)response
  373. newRequest:(NSURLRequest *)request
  374. completionHandler:(void (^)(NSURLRequest * _Nullable))completionHandler {
  375. if (self.configuration.enableFollowRedirects) {
  376. completionHandler(request);
  377. } else {
  378. completionHandler(nil);
  379. }
  380. }
  381. #pragma mark - NSURLSessionDataDelegate Methods
  382. - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
  383. {
  384. /* background upload task will not call back didRecieveResponse */
  385. OSSLogVerbose(@"%@,response: %@", NSStringFromSelector(_cmd), response);
  386. OSSNetworkingRequestDelegate * delegate = [self.sessionDelagateManager objectForKey:@(dataTask.taskIdentifier)];
  387. NSHTTPURLResponse * httpResponse = (NSHTTPURLResponse *)response;
  388. if (httpResponse.statusCode >= 200 && httpResponse.statusCode < 300 && httpResponse.statusCode != 203) {
  389. [delegate.responseParser consumeHttpResponse:httpResponse];
  390. } else {
  391. delegate.isHttpRequestNotSuccessResponse = YES;
  392. }
  393. completionHandler(NSURLSessionResponseAllow);
  394. }
  395. - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
  396. {
  397. OSSNetworkingRequestDelegate * delegate = [self.sessionDelagateManager objectForKey:@(dataTask.taskIdentifier)];
  398. /* background upload task will not call back didRecieveResponse.
  399. so if we recieve response data after background uploading file,
  400. we consider it as error response message since a successful uploading request will not response any data */
  401. if (delegate.isBackgroundUploadFileTask)
  402. {
  403. //判断当前的statuscode是否成功
  404. NSHTTPURLResponse * httpResponse = (NSHTTPURLResponse *)dataTask.response;
  405. if (httpResponse.statusCode >= 200 && httpResponse.statusCode < 300 && httpResponse.statusCode != 203) {
  406. [delegate.responseParser consumeHttpResponse:httpResponse];
  407. delegate.isHttpRequestNotSuccessResponse = NO;
  408. }else
  409. {
  410. delegate.isHttpRequestNotSuccessResponse = YES;
  411. }
  412. }
  413. if (delegate.isHttpRequestNotSuccessResponse) {
  414. [delegate.httpRequestNotSuccessResponseBody appendData:data];
  415. }
  416. else {
  417. if (delegate.onRecieveData) {
  418. delegate.onRecieveData(data);
  419. } else {
  420. OSSTask * consumeTask = [delegate.responseParser consumeHttpResponseBody:data];
  421. if (consumeTask.error) {
  422. OSSLogError(@"consume data error: %@", consumeTask.error);
  423. delegate.error = consumeTask.error;
  424. [dataTask cancel];
  425. }
  426. }
  427. }
  428. if (!delegate.isHttpRequestNotSuccessResponse && delegate.downloadProgress) {
  429. int64_t bytesWritten = [data length];
  430. delegate.payloadTotalBytesWritten += bytesWritten;
  431. int64_t totalBytesExpectedToWrite = dataTask.response.expectedContentLength;
  432. delegate.downloadProgress(bytesWritten, delegate.payloadTotalBytesWritten, totalBytesExpectedToWrite);
  433. }
  434. }
  435. #pragma mark - Private Methods
  436. - (BOOL)evaluateServerTrust:(SecTrustRef)serverTrust forDomain:(NSString *)domain {
  437. /*
  438. * Creates the policies for certificate verification.
  439. */
  440. NSMutableArray *policies = [NSMutableArray array];
  441. if (domain) {
  442. [policies addObject:(__bridge_transfer id)SecPolicyCreateSSL(true, (__bridge CFStringRef)domain)];
  443. } else {
  444. [policies addObject:(__bridge_transfer id)SecPolicyCreateBasicX509()];
  445. }
  446. /*
  447. * Sets the policies to server's certificate
  448. */
  449. SecTrustSetPolicies(serverTrust, (__bridge CFArrayRef)policies);
  450. /*
  451. * Evaulates if the current serverTrust is trustable.
  452. * It's officially suggested that the serverTrust could be passed when result = kSecTrustResultUnspecified or kSecTrustResultProceed.
  453. * For more information checks out https://developer.apple.com/library/ios/technotes/tn2232/_index.html
  454. * For detail information about SecTrustResultType, checks out SecTrust.h
  455. */
  456. SecTrustResultType result;
  457. SecTrustEvaluate(serverTrust, &result);
  458. return (result == kSecTrustResultUnspecified || result == kSecTrustResultProceed);
  459. }
  460. @end