全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

iOS实用教程之Https双向认证详解

前言

年前的时候,关于苹果要强制https的传言四起,虽然结果只是一个“谣言”,但是很明显的这是迟早会到来的,间接上加速了各公司加紧上https的节奏,对于iOS客户端来说,上https需不需要改变一些东西取决于---------对,就是公司有没有钱。土豪公司直接买买买,iOS开发者只需要把http改成https完事。然而很不幸,我们在没钱的公司,选择了自签证书。虽然网上很多关于https的适配,然而很多都是已过时的,这里我们主要是讲一下https双向认证。

【证书选择】自签

【网络请求】原生NSURLSession或者AFNetworking3.0以上版本

【认证方式】双向认证

Https双向认证过程

先来了解一下双向认证的大体过程:(图片来自网络,如果是某位博主原创的请私信我)


下面我们一步步来实现

1、设置服务端证书

 NSString *certFilePath = [[NSBundle mainBundle] pathForResource:@"server" ofType:@"cer"];
 NSData *certData = [NSData dataWithContentsOfFile:certFilePath];
 NSSet *certSet = [NSSet setWithObject:certData];
 AFSecurityPolicy *policy = [AFSecurityPolicy policyWithPinningMode:AFSSLPinningModeCertificate withPinnedCertificates:certSet];
 policy.allowInvalidCertificates = YES;
 policy.validatesDomainName = NO;
 self.afnetworkingManager.securityPolicy = policy;

2、处理挑战

原生的NSURLSession是在

- (void)URLSession:(NSURLSession *)session didReceiveChallenge:(nonnull NSURLAuthenticationChallenge *)challenge completionHandler:(nonnull void (^)(NSURLSessionAuthChallengeDisposition, NSURLCredential * _Nullable))completionHandler

代理方法里面处理挑战的,再看看AFNetworking在该代理方法里处理的代码

 if (self.taskDidReceiveAuthenticationChallenge) {
  disposition = self.taskDidReceiveAuthenticationChallenge(session, task, challenge, &credential);
 } else {
  ...
 }

我们只需要给它传递一个处理的block

[self.afnetworkingManager setSessionDidReceiveAuthenticationChallengeBlock:^NSURLSessionAuthChallengeDisposition(NSURLSession*session, NSURLAuthenticationChallenge *challenge, NSURLCredential *__autoreleasing*_credential) {
  ...
}

根据传来的challenge生成disposition(应对挑战的方式)和credential(客户端生成的挑战证书)

3、服务端认证

当challenge的认证方法为NSURLAuthenticationMethodServerTrust时,需要客户端认证服务端证书

//评估服务端安全性
if([weakSelf.afnetworkingManager.securityPolicy evaluateServerTrust:challenge.protectionSpace.serverTrust forDomain:challenge.protectionSpace.host]) {
    //创建凭据
    credential = [NSURLCredential credentialForTrust:challenge.protectionSpace.serverTrust];
    if(credential) {
     disposition =NSURLSessionAuthChallengeUseCredential;
    } else {
     disposition =NSURLSessionAuthChallengePerformDefaultHandling;
    }
   } else {
    disposition = NSURLSessionAuthChallengeCancelAuthenticationChallenge;
   }

4、客户端认证

认证完服务端后,需要认证客户端

由于是双向认证,这一步是必不可省的

   SecIdentityRef identity = NULL;
   SecTrustRef trust = NULL;
   NSString *p12 = [[NSBundle mainBundle] pathForResource:@"client"ofType:@"p12"];
   NSFileManager *fileManager =[NSFileManager defaultManager];

   if(![fileManager fileExistsAtPath:p12])
   {
    NSLog(@"client.p12:not exist");
   }
   else
   {
    NSData *PKCS12Data = [NSData dataWithContentsOfFile:p12];

    if ([[weakSelf class]extractIdentity:&identity andTrust:&trust fromPKCS12Data:PKCS12Data])
    {
     SecCertificateRef certificate = NULL;
     SecIdentityCopyCertificate(identity, &certificate);
     const void*certs[] = {certificate};
     CFArrayRef certArray =CFArrayCreate(kCFAllocatorDefault, certs,1,NULL);
     credential =[NSURLCredential credentialWithIdentity:identity certificates:(__bridge NSArray*)certArray persistence:NSURLCredentialPersistencePermanent];
     disposition =NSURLSessionAuthChallengeUseCredential;
    }
   }
+ (BOOL)extractIdentity:(SecIdentityRef*)outIdentity andTrust:(SecTrustRef *)outTrust fromPKCS12Data:(NSData *)inPKCS12Data {
 OSStatus securityError = errSecSuccess;
 //client certificate password
 NSDictionary*optionsDictionary = [NSDictionary dictionaryWithObject:@"your p12 file pwd"
                 forKey:(__bridge id)kSecImportExportPassphrase];

 CFArrayRef items = CFArrayCreate(NULL, 0, 0, NULL);
 securityError = SecPKCS12Import((__bridge CFDataRef)inPKCS12Data,(__bridge CFDictionaryRef)optionsDictionary,&items);

 if(securityError == 0) {
  CFDictionaryRef myIdentityAndTrust =CFArrayGetValueAtIndex(items,0);
  const void*tempIdentity =NULL;
  tempIdentity= CFDictionaryGetValue (myIdentityAndTrust,kSecImportItemIdentity);
  *outIdentity = (SecIdentityRef)tempIdentity;
  const void*tempTrust =NULL;
  tempTrust = CFDictionaryGetValue(myIdentityAndTrust,kSecImportItemTrust);
  *outTrust = (SecTrustRef)tempTrust;
 } else {
  NSLog(@"Failedwith error code %d",(int)securityError);
  return NO;
 }
 return YES;
}

原生NSURLSession双向认证

在原生的代理方法里面认证就行,代码基本和AFNetworking的一致,注意最后需要调用

 completionHandler(NSURLSessionAuthChallengeUseCredential, credential);

来执行回调操作

关于UIWebView的Https双向认证

网上的资料大体上有几种解决方法

1:跳过Https认证(这还能跳过?没试过,不太靠谱)

2:中断原有的请求步骤,将request拿出来,下载完整的HTML代码,让webView加载该代码(在单页面展示的情况下基本满足使用,但是在部分标签不是独立跳转https路径的时候,将出现无法加载的情况,不是很好用)

- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType {
 NSString * urlString = [request.URL absoluteString];
 if ([urlString containsString:URL_API_BASE]) {
  [[SUHTTPOperationManager manager]REQUEST:request progress:nil handler:^(BOOL isSucc, id responseObject, NSError *error) {
   NSString * htmlString = [[NSString alloc] initWithData:responseObject encoding:NSUTF8StringEncoding];
   BASE_INFO_FUN(@"下载HTML完毕");
   [self loadHTMLString:htmlString baseURL:nil];
  }];
  return NO;
 }
 return YES;
}

3、中断原有的请求步骤,将request拿出来,完成鉴权认证之后,再让webView重新请求该request(这种方式理论上好像可以,我试过,没有成功,可能我打开的方式不正确)
4、或许,您有更好的解决方案 - -

关于代码

网上很多https双向认证的代码,基本是一样的,这里我们直接拿来用就可以,前提是我们不能单纯copy,而是在理解其实现的基础上,整合到工程中,遇到问题解决思路清晰,而不是一脸懵逼。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。


# https  # 双向认证  # ios  # iOS9苹果将原http协议改成了https协议的方法  # 详解iOS开发 - 用AFNetworking实现https单向验证  # 双向验证  # IOS开发 支持https请求以及ssl证书配置详解  # iOS适配https证书问题(AFNetworking3.0为例)  # iOS之Https自签名证书认证及数据请求的封装原理  # 服务端  # 客户端  # 拿出来  # 跳过  # 网上  # 都是  # 这是  # 加载  # 是在  # 不太  # 不需要  # 基础上  # 只需  # 上有  # 一脸  # 要把  # 就行  # 几种  # 只需要  # 跳转 


相关文章: 南平网站制作公司,2025年南平市事业单位报名时间?  ,有什么在线背英语单词效率比较高的网站?  建站中国必看指南:CMS建站系统+手机网站搭建核心技巧解析  如何在Windows环境下新建FTP站点并设置权限?  高端建站如何打造兼具美学与转化的品牌官网?  建站一年半SEO优化实战指南:核心词挖掘与长尾流量提升策略  表情包在线制作网站免费,表情包怎么弄?  定制建站方案优化指南:企业官网开发与建站费用解析  javascript中对象的定义、使用以及对象和原型链操作小结  建站之星各版本价格是多少?  教学论文网站制作软件有哪些,写论文用什么软件 ?  TestNG的testng.xml配置文件怎么写  如何用西部建站助手快速创建专业网站?  如何快速完成中国万网建站详细流程?  如何快速查询域名建站关键信息?  制作营销网站公司,淘特是干什么用的?  如何在阿里云ECS服务器部署织梦CMS网站?  香港服务器选型指南:免备案配置与高效建站方案解析  北京的网站制作公司有哪些,哪个视频网站最好?  如何在IIS中新建站点并解决端口绑定冲突?  Java解压缩zip - 解压缩多个文件或文件夹实例  如何快速查询网站的真实建站时间?  如何配置FTP站点权限与安全设置?  如何通过服务器快速搭建网站?完整步骤解析  微课制作网站有哪些,微课网怎么进?  制作网站的过程怎么写,用凡科建站如何制作自己的网站?  网站海报制作教学视频教程,有什么免费的高清可商用图片网站,用于海报设计?  制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?  常州自助建站:操作简便模板丰富,企业个人快速搭建网站  如何在Ubuntu系统下快速搭建WordPress个人网站?  如何在橙子建站中快速调整背景颜色?  保定网站制作方案定制,保定招聘的渠道有哪些?找工作的人一般都去哪里看招聘信息?  建站主机选购指南:核心配置优化与品牌推荐方案  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?  如何在阿里云完成域名注册与建站?  建站之星五站合一营销型网站搭建攻略,流量入口全覆盖优化指南  一键制作网站软件下载安装,一键自动采集网页文档制作步骤?  建站之星后台密码遗忘或太弱?如何重置与强化?  百度网页制作网站有哪些,谁能告诉我百度网站是怎么联系?  如何选择CMS系统实现快速建站与SEO优化?  如何自己制作一个网站链接,如何制作一个企业网站,建设网站的基本步骤有哪些?  建站主机功能解析:服务器选择与快速搭建指南  广州营销型建站服务商推荐:技术优势与SEO优化解析  道歉网站制作流程,世纪佳缘致歉小吴事件,相亲网站身份信息伪造该如何稽查?  大同网页,大同瑞慈医院官网?  北京营销型网站制作公司,可以用python做一个营销推广网站吗?  头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?  赚钱网站制作软件,建一个网站怎样才能赚钱?是如何盈利的?  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  北京网站制作公司哪家好一点,北京租房网站有哪些? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。