AFNetworking 3.0與服務(wù)端 自簽名證書 https雙向認(rèn)證

原創(chuàng)文章遷移至
https://blog.ixianshang.net/2019/08/08/AFNetworking%203.0%E4%B8%8E%E6%9C%8D%E5%8A%A1%E7%AB%AF%20%E8%87%AA%E7%AD%BE%E5%90%8D%E8%AF%81%E4%B9%A6%20https%E5%8F%8C%E5%90%91%E8%AE%A4%E8%AF%81/index.html

iOS網(wǎng)絡(luò)安全現(xiàn)在越來(lái)越重要崎苗,一個(gè)抓包工具(charles等)隨隨便便就能抓到你所請(qǐng)求的數(shù)據(jù)辩撑,這些數(shù)據(jù)如果是明碼的后果很嚴(yán)重,不是指明文窗市,可以通過(guò)這些數(shù)據(jù)來(lái)判定服務(wù)端部署的數(shù)據(jù)接口,更能夠嗅探到服務(wù)端的漏洞胳螟。

所以最近因公司業(yè)務(wù)及數(shù)據(jù)安全上的需要準(zhǔn)備使用https方式進(jìn)行數(shù)據(jù)請(qǐng)求枝笨。而且蘋果現(xiàn)在默認(rèn)是https方式請(qǐng)求层宫。

Apple CSR CER P12 mobileprovition 到底是什么

  • CSR(Certificate Signing Request)鑰匙串文件 為生成證書做基礎(chǔ),要生成CER證書必須要有CSR私鑰爽航,此私鑰包含了用戶自己的一些信息蚓让。這個(gè)文件是保存在我們的mac的(keychain)里面的, 此文件包含了(公鑰和私鑰)
  • CER 包含了開發(fā)者信息和公鑰
  • P12 它不僅包含CER的信息,還有私鑰信息讥珍,即: P12備份文件 = CER文件 + 私鑰历极;所以有了這個(gè)p12就再也不用擔(dān)心證書丟失了
  • mobileprovition 包含了上述所有內(nèi)容 Certificate && App ID && Device, 這個(gè)Provisioning Profile文件會(huì)在打包時(shí)嵌入到.ipa的包里。本人理解的是 可以真機(jī)調(diào)試的憑證衷佃。

https原理

HTTPS(全稱:Hyper Text Transfer Protocol over Secure Socket Layer)趟卸,是以安全為目標(biāo)的HTTP通道,簡(jiǎn)單講是HTTP的安全版氏义。即HTTP下加入SSL層锄列,HTTPS的安全基礎(chǔ)是SSL,因此加密的詳細(xì)內(nèi)容就需要SSL惯悠。 它是一個(gè)URI scheme(抽象標(biāo)識(shí)符體系)邻邮,句法類同http:體系。用于安全的HTTP數(shù)據(jù)傳輸克婶。https:URL表明它使用了HTTP筒严,但HTTPS存在不同于HTTP的默認(rèn)端口及一個(gè)加密/身份驗(yàn)證層(在HTTP與TCP之間)。這個(gè)系統(tǒng)的最初研發(fā)由網(wǎng)景公司(Netscape)進(jìn)行情萤,并內(nèi)置于其瀏覽器Netscape Navigator中鸭蛙,提供了身份驗(yàn)證與加密通訊方法。(摘自百度百科)

PKI(公鑰基礎(chǔ)設(shè)施)技術(shù)是HTTPS的基礎(chǔ)筋岛,PKI與非對(duì)稱密鑰加密技術(shù)密切相關(guān)规惰,包括消息摘要、數(shù)字簽名和加密服務(wù)泉蝌,而數(shù)字證書以及證書機(jī)構(gòu)(CA – Certificate Authority)是PKI中的重要概念。

看看這篇文章

雙向認(rèn)證原理圖:


httpsyuanlitu.png

需要準(zhǔn)備的自簽名證書

1.服務(wù)器私鑰 2.由CA簽發(fā)的含有服務(wù)器公鑰的數(shù)字證書 3.CA的數(shù)字證書揩晴。在雙向認(rèn)證的實(shí)踐中勋陪,通常服務(wù)器可以自己作為證書機(jī)構(gòu),并且由服務(wù)器CA簽發(fā)服務(wù)器證書和客戶端證書硫兰。

1.客戶端私鑰 2. 由CA簽發(fā)的含有客戶端公鑰的數(shù)字證書诅愚。為了避免中間人攻擊,客戶端還需要內(nèi)置服務(wù)器證書,用來(lái)驗(yàn)證所連接的服務(wù)器是否是指定的服務(wù)器违孝。

  • 服務(wù)端 .cer
  • 客戶端 .p12

使用AFNetworking 3.0

    NSString *certFilePath = [[NSBundle mainBundle] pathForResource:@"server" ofType:@"der"];
    NSData *certData = [NSData dataWithContentsOfFile:certFilePath];
    NSSet *certSet = [NSSet setWithObject:certData];
    AFSecurityPolicy *policy = [AFSecurityPolicy policyWithPinningMode:AFSSLPinningModeCertificate withPinnedCertificates:certSet];
    policy.allowInvalidCertificates = YES;
    policy.validatesDomainName = NO;
     
    _manager = [AFHTTPSessionManager manager];
    _manager.securityPolicy = policy;
    _manager.requestSerializer = [AFHTTPRequestSerializer serializer];
    _manager.responseSerializer = [AFHTTPResponseSerializer serializer];
    _manager.responseSerializer.acceptableContentTypes =  [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript",@"text/plain", nil];
    //關(guān)閉緩存避免干擾測(cè)試r
    _manager.requestSerializer.cachePolicy = NSURLRequestReloadIgnoringLocalCacheData;
    [_manager setSessionDidBecomeInvalidBlock:^(NSURLSession * _Nonnull session, NSError * _Nonnull error) {
        DLog(@"setSessionDidBecomeInvalidBlock");
    }];
//客服端請(qǐng)求驗(yàn)證 重寫 setSessionDidReceiveAuthenticationChallengeBlock 方法
__weak typeof(self)weakSelf = self;   
[_manager setSessionDidReceiveAuthenticationChallengeBlock:^NSURLSessionAuthChallengeDisposition(NSURLSession*session, NSURLAuthenticationChallenge *challenge, NSURLCredential *__autoreleasing*_credential) {
    NSURLSessionAuthChallengeDisposition disposition = NSURLSessionAuthChallengePerformDefaultHandling;
    __autoreleasing NSURLCredential *credential =nil;
    if([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust]) {
        if([weakSelf.manager.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;
        }
    } else {
        // client authentication
        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;
            }
        }
    }
    *_credential = credential;
    return disposition;
}];
+(BOOL)extractIdentity:(SecIdentityRef*)outIdentity andTrust:(SecTrustRef *)outTrust fromPKCS12Data:(NSData *)inPKCS12Data {
    OSStatus securityError = errSecSuccess;
    //client certificate password
    NSDictionary*optionsDictionary = [NSDictionary dictionaryWithObject:@"你的p12密碼"
                                                                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;
}

完畢

  • 使用MKNetworkKit 與 使用蘋果官方原生 代碼片段 詳見Demo

Demo

https://github.com/cuiwe000/HttpsDemo.git

參考文檔

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末刹前,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子雌桑,更是在濱河造成了極大的恐慌喇喉,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,311評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件校坑,死亡現(xiàn)場(chǎng)離奇詭異拣技,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)耍目,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門膏斤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人邪驮,你說(shuō)我怎么就攤上這事莫辨。” “怎么了毅访?”我有些...
    開封第一講書人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵沮榜,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我俺抽,道長(zhǎng)敞映,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任磷斧,我火速辦了婚禮振愿,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘弛饭。我一直安慰自己冕末,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開白布侣颂。 她就那樣靜靜地躺著档桃,像睡著了一般。 火紅的嫁衣襯著肌膚如雪憔晒。 梳的紋絲不亂的頭發(fā)上藻肄,一...
    開封第一講書人閱讀 49,031評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音拒担,去河邊找鬼嘹屯。 笑死,一個(gè)胖子當(dāng)著我的面吹牛从撼,可吹牛的內(nèi)容都是我干的州弟。 我是一名探鬼主播,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼婆翔!你這毒婦竟也來(lái)了拯杠?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤啃奴,失蹤者是張志新(化名)和其女友劉穎潭陪,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體纺腊,經(jīng)...
    沈念sama閱讀 43,466評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡畔咧,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評(píng)論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了揖膜。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片誓沸。...
    茶點(diǎn)故事閱讀 38,039評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖壹粟,靈堂內(nèi)的尸體忽然破棺而出拜隧,到底是詐尸還是另有隱情,我是刑警寧澤趁仙,帶...
    沈念sama閱讀 33,701評(píng)論 4 323
  • 正文 年R本政府宣布洪添,位于F島的核電站,受9級(jí)特大地震影響雀费,放射性物質(zhì)發(fā)生泄漏干奢。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評(píng)論 3 307
  • 文/蒙蒙 一盏袄、第九天 我趴在偏房一處隱蔽的房頂上張望忿峻。 院中可真熱鬧,春花似錦辕羽、人聲如沸逛尚。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)绰寞。三九已至,卻和暖如春铣口,著一層夾襖步出監(jiān)牢的瞬間滤钱,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工脑题, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留菩暗,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,497評(píng)論 2 354
  • 正文 我出身青樓旭蠕,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子掏熬,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容