Alamofire 安全认证ServerTrustPolicy

前言

在互联网迅速发展的年代,基本上天天都在跟网络打交道。那么,在网络的通讯中怎么保证信息的安全性呢?这篇文章,我们就来讲讲,Alamofire作为iOS开发中一个非常优秀的网络请求相关的第三方库,它的安全策略是怎么设计和使用的。

HTTPS简介

在切入正题之前,先来简单的了解一下HTTPS相关知识,方便对后面内容的理解。如果你已经了解了,可以直接跳过这一段。

为什么使用HTTPS

在以前,我们用的更多的是HTTP,那么是什么原因苹果公司也主推我们使用HTTPS这个更安全请求方式的呢?HTTP存在的问题:

  1. 通讯使用明文,内容容易被窃听
  2. 不验证通讯双方的身份,容易被伪装
  3. 无法验证数据完整性,可能会被篡改数据

使用 HTTPS 通信机制可以有效地防止这些问题。HTTPS 并非是应用层的一种新协议,只是HTTP通信接口部分用 SSLTLS 协议代替而已。通常 HTTP 直接跟 TCP 通信,当使用 SSL 时,则变成先和 SSL 通信,再由 SSLTCP 通信了。简言之,HTTPS 就是身披 SSL 协议这层外壳的 HTTP

HTTP+数据加密+身份认证+数据完整性保护=HTTPS

HTTPS加密方式

HTTPS 采用混合加密机制,就是共享秘钥加密(对称加密)和公开密钥加密(非对称加密)两者并用的混合加密机制。在交换密钥环节使用公开密钥加密的方式,之后建立通讯交换报文阶段则使用共享密钥加密。公开密钥加密比共享密钥加密更加安全,那为什么不全用公开密钥加密?因为与共享秘钥加密相比,处理起来更复杂、处理速度慢。HTTPS结合了两种加密方式的优劣来实现一种混合加密的流程。如图所示:

HTTPS验证和加密流程

HTTPS有单向认证和双向认证,原理基本差不多,这里就讲一下单向认证的整个流程,先看一张图:

  1. 客户端发起HTTPS请求:客户端向服务端发送SSL协议版本号、加密算法种类、随机数等信息。
  2. 服务端的配置:采用HTTPS协议的服务器必须要有一套数字证书,可以自己制作,也可以向组织申请。区别就是自己颁发的证书需要客户端验证通过,才可以继续访问,而使用受信任的公司申请的证书可以直接通过。这套证书其实就是一对公钥和私钥。
  3. 传送证书:这个证书其实就是公钥,只是包含了很多信息,如证书的颁发机构,过期时间等等。同时服务端也会向客户端发送SSL协议版本号、加密算法种类、随机数等信息
  4. 客户端解析证书:这部分工作是由客户端的TLS来完成的,首先会验证公钥是否有效,比如颁发机构,过期时间等等,如果发现异常,则会抛出一个警告,提示证书存在问题。如果证书没有问题,那么就生成一个随机值,然后用证书对该随机值进行加密。
  5. 传送加密信息:这部分传送的是用证书加密后的随机值,目的就是让服务端得到这个随机值,以后客户端和服务端的通信就可以通过这个随机值来进行加密解密了。
  6. 服务段解密信息:服务端用私钥解密后,得到了客户端传过来的随机值(私钥),然后把内容通过该值进行对称加密。
  7. 传输加密后的信息:这部分信息是服务段用私钥加密后的信息,可以在客户端被还原。
  8. 客户端解密信息:客户端用之前生成的私钥解密服务段传过来的信息,于是获取了解密后的内容。整个过程第三方即使监听到了数据,也束手无策。

Alamofire的安全认证

证书验证模式

如果使用的是自签证书需要我们进行安全认证,如果是CA机构颁发的证书是不需要我们写安全认证的相关代码的。
举个Alamofire发起HTTPS请求的栗子🌰:

let serverTrustPlolicies: [String: ServerTrustPolicy] = [
    hostUrl: .pinCertificates(
        certificates: ServerTrustPolicy.certificates(),
        validateCertificateChain: true,
        validateHost: true)
]
self.sessionManager = SessionManager(serverTrustPolicyManager: ServerTrustPolicyManager(policies: serverTrustPlolicies))
self.sessionManager?.request(urlString).response { (defaultResponse) in
    print(defaultResponse)
}
  • 使用起来也是非常的简单,跟HTTP相比,只是在初始化SessionManager的时候传入了一个ServerTrustPolicyManager对象,它是证书信任策略的管理者。
  • 初始化ServerTrustPolicyManager对象的时候,传入了一个[String: ServerTrustPolicy]类型的集合作为参数并保存,key是主机地址,value是验证模式。
public enum ServerTrustPolicy {
    case performDefaultEvaluation(validateHost: Bool)
    case performRevokedEvaluation(validateHost: Bool, revocationFlags: CFOptionFlags)
    case pinCertificates(certificates: [SecCertificate], validateCertificateChain: Bool, validateHost: Bool)
    case pinPublicKeys(publicKeys: [SecKey], validateCertificateChain: Bool, validateHost: Bool)
    case disableEvaluation
    case customEvaluation((_ serverTrust: SecTrust, _ host: String) -> Bool)
}
  • Alamofire 安全认证策略的六种模式,其中最常用的有这三种:.pinCertificates 证书验证模式、.pinPublicKeys 公钥验证模式和 .disableEvaluation 不验证模式。
    • .performDefaultEvaluation 默认策略,只有合法证书才能通过验证
    • .performRevokedEvaluation 对注销证书做的一种额外设置
    • .pinCertificates 证书验证模式,代表客户端会将服务器返回的证书和本地保存的证书中的 所有内容 全部进行校验,如果正确,才继续执行。
    • .pinPublicKeys 公钥验证模式,代表客户端会将服务器返回的证书和本地保存的证书中的 PublicKey部分 进行校验,如果正确,才继续执行。
    • .disableEvaluation 该选项下验证一直都是通过的,无条件信任。
    • .customEvaluation 自定义验证,需要返回一个布尔类型的结果。
  • 前面的例子就是使用的.pinCertificates证书验证模式。它有三个关联值:
    • 参数1:certificates代表的是证书
    • 参数2:validateCertificateChain代表是否验证证书链
    • 参数3:validateHost代表是否验证子地址
  • 那么,我们怎么找到项目中的证书呢?Alamofire为我们提供了一个方法ServerTrustPolicy.certificates()方便使用。
public static func certificates(in bundle: Bundle = Bundle.main) -> [SecCertificate] {
    var certificates: [SecCertificate] = []

    let paths = Set([".cer", ".CER", ".crt", ".CRT", ".der", ".DER"].map { fileExtension in
        bundle.paths(forResourcesOfType: fileExtension, inDirectory: nil)
    }.joined())

    for path in paths {
        if
            let certificateData = try? Data(contentsOf: URL(fileURLWithPath: path)) as CFData,
            let certificate = SecCertificateCreateWithData(nil, certificateData)
        {
            certificates.append(certificate)
        }
    }

    return certificates
}
  • 默认是在Bundle.main中查找,我们也可以指定存放证书的Bundle来查找。
  • 了解了验证模式之后,来看看验证的流程。当发起请求之后,会回调URLSessionTaskDelegate的下面这个方法。
open func urlSession(
    _ session: URLSession,
    task: URLSessionTask,
    didReceive challenge: URLAuthenticationChallenge,
    completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void)
{
    guard taskDidReceiveChallengeWithCompletion == nil else {
        taskDidReceiveChallengeWithCompletion?(session, task, challenge, completionHandler)
        return
    }

    if let taskDidReceiveChallenge = taskDidReceiveChallenge {
        let result = taskDidReceiveChallenge(session, task, challenge)
        completionHandler(result.0, result.1)
    } else if let delegate = self[task]?.delegate {
        delegate.urlSession(
            session,
            task: task,
            didReceive: challenge,
            completionHandler: completionHandler
        )
    } else {
        urlSession(session, didReceive: challenge, completionHandler: completionHandler)
    }
}
  • 如果taskDidReceiveChallengeWithCompletion有值的话,直接回调这个闭包,这就说明我们可以自己实现验证的逻辑。可见Alamofire是非常的友好的,既提供了常规实现方式,也支持开发者自定义实现。

  • 继续跟踪代码进入到delegate.urlSession方法里面

  • 然后会执行红框所示的方法,获取的是前面设置的验证模式,然后执行evaluate方法

public func evaluate(_ serverTrust: SecTrust, forHost host: String) -> Bool {
    var serverTrustIsValid = false

    switch self {
    // 省略无法代码.....
    case let .pinCertificates(pinnedCertificates, validateCertificateChain, validateHost):
        if validateCertificateChain {
            let policy = SecPolicyCreateSSL(true, validateHost ? host as CFString : nil)
            SecTrustSetPolicies(serverTrust, policy)

            SecTrustSetAnchorCertificates(serverTrust, pinnedCertificates as CFArray)
            SecTrustSetAnchorCertificatesOnly(serverTrust, true)

            serverTrustIsValid = trustIsValid(serverTrust)
        } else {
            let serverCertificatesDataArray = certificateData(for: serverTrust)
            let pinnedCertificatesDataArray = certificateData(for: pinnedCertificates)

            outerLoop: for serverCertificateData in serverCertificatesDataArray {
                for pinnedCertificateData in pinnedCertificatesDataArray {
                    if serverCertificateData == pinnedCertificateData {
                        serverTrustIsValid = true
                        break outerLoop
                    }
                }
            }
        }
    // 省略无法代码.....
    return serverTrustIsValid
}
  • 如果需要验证证书链的话,会执行if代码块,大概步骤如下:
    • SecPolicyCreateSSL:创建策略,是否验证host
    • SecTrustSetPolicies:为待验证的对象设置策略
    • trustIsValid:进行验证,成功返回YES
  • 如果需要验证证书链的话,会先把服务器证书和自己的证书处理成证书二进制数组,然后进行对比。
  • 如果验证通过就可以开始正常请求数据,验证失败就会终止请求,抛出异常。

公钥验证模式

  • 公钥验证模式跟证书验证模式使用起来差不多,使用 .pinPublicKeys 这个枚举值,第一个参数传公钥,其他参数和证书验证模式一样。Alamofire同样为我们提供了获取公钥的方法ServerTrustPolicy.publicKeys(),直接调用这个方法即可。
  • 那么什么时候使用公钥验证模式呢?使用公钥验证模式的好处是:只要公钥不变,就可以一直使用,不用更新证书,不用担心证书过期了。

总结

这篇文章大概聊了一下HTTPS的加密流程,以及通过如何Alamofire请求HTTPS的接口和网络挑战流程的分析。当然,如果想彻底明白HTTPS的实现方式,还需继续学习。推荐一本书《图解HTTP》,讲的非常通俗易懂,文章的几张图片也是来源于这本书。


有问题或者建议和意见,欢迎大家评论或者私信。
喜欢的朋友可以点下关注和喜欢,后续会持续更新文章。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容