我的WKWebView cookie處理方案

Demo點這里

背景

正在維護的App使用cookie來維護用戶登錄狀態(tài)嗅钻、App版本系統(tǒng)晾腔、語言等狀態(tài)信息。

在UIWebView時代啊犬,可以通過NSHTTPCookieStorage單例很直接的管理客戶端cookie灼擂。UIWebView的cookie數(shù)據(jù)會自動和NSHTTPCookieStorage進行同步。然而WKWebView的cookie維護一直為人詬觉至。只要你維護過相關(guān)業(yè)務(wù)剔应,不同iOS版本上出現(xiàn)的各種cookie的問題一定讓你頭疼過。

這個Demo是目前項目中使用的cookie管理方案语御,方案來回折騰了好幾個月峻贮,雖然不是很完整,但是基本滿足當(dāng)前項目需求应闯。

方案

初始化時WKWebsiteDataStore使用[WKWebsiteDataStore defaultDataStore]纤控,WKProcessPool使用全局單例。

//  webView初始化
- (WKWebView *)wkWebView{
    if (!_wkWebView) {
        WKWebViewConfiguration * config = [[WKWebViewConfiguration alloc]init];
        config.allowsInlineMediaPlayback = YES;
        config.selectionGranularity = YES;
        config.processPool = [WebViewCookieUtil sharedProcessPool];
        config.userContentController = self.userContentController;
        
        _wkWebView = [[WKWebView alloc] initWithFrame:self.view.bounds configuration:config];
        _wkWebView.translatesAutoresizingMaskIntoConstraints = NO;
        _wkWebView.navigationDelegate = self;
        _wkWebView.UIDelegate = self;
        [_wkWebView addObserver:self forKeyPath:NSStringFromSelector(@selector(estimatedProgress)) options:0 context:nil];
        _wkWebView.opaque = NO;
        if (@available(iOS 11.0, *)) {
            _wkWebView.scrollView.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever;
        }
    }
    return _wkWebView;
}

通過NSHTTPCookieStorage維護客戶端cookie碉纺,當(dāng)cookie更新時首先更新NSHTTPCookieStorage中船万,然后在不同的iOS版本使用不同的方式同步到WKWebView中。

+ (void)clientCookieDidUpdate:(NSDictionary *)cookieDict toRemove:(NSArray *)toRemove {
    //  客戶端cookie更新時調(diào)用骨田,比如用戶登錄狀態(tài)改變
    NSHTTPCookieStorage *httpCookieStorage = [NSHTTPCookieStorage sharedHTTPCookieStorage];
    // 移除Cookie
    NSArray<NSHTTPCookie *> *oldCookies = httpCookieStorage.cookies;
    [oldCookies enumerateObjectsUsingBlock:^(NSHTTPCookie * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        for (NSString *name in toRemove) {
            if ([obj.domain isEqualToString:name] && [obj.domain isEqualToString:domainForThisApp]) {
                [httpCookieStorage deleteCookie:obj];
            }
        }
    }];
    
    // Write new Cookie to storage.
    NSArray *cookieObjectArr = [self cookieObjectsFromCookieDict:cookieDict];
    for (NSHTTPCookie *cookie in cookieObjectArr) {
        [httpCookieStorage setCookie:cookie];
    }
    
    if (@available(iOS 11.0, *)) {
        WKHTTPCookieStore *wkCookieStore = [WKWebsiteDataStore defaultDataStore].httpCookieStore;
        [wkCookieStore getAllCookies:^(NSArray<NSHTTPCookie *> * _Nonnull currentCookies) {
            dispatch_group_t cleanGroup = dispatch_group_create();
            for (NSHTTPCookie *cookie in currentCookies) {
                for (NSString *name in toRemove) {
                    if ([cookie.domain isEqualToString:domainForThisApp] && [cookie.name isEqualToString:name]) {
                        dispatch_group_enter(cleanGroup);
                        [wkCookieStore deleteCookie:cookie completionHandler:^{
                            dispatch_group_leave(cleanGroup);
                        }];
                    }
                }
            }
            dispatch_group_notify(cleanGroup, dispatch_get_main_queue(), ^{
                for (NSHTTPCookie *cookie in cookieObjectArr) {
                    [wkCookieStore setCookie:cookie completionHandler:nil];
                }
            });
        }];
    } else {
        //  adjust userScript and reload webView if needed
    }
}

iOS11以上

管理方式如上段代碼所示耿导,在cookie更新之后直接對[WKWebsiteDataStore defaultDataStore]進行cookie設(shè)置即可。

潛在問題:
Cookie更新緩慢或無法更新态贤。
當(dāng)存在非視圖結(jié)構(gòu)中的WKWebView同時加載網(wǎng)頁時舱呻,WKHttpCookieStorage的異步API回調(diào)可能會被阻塞導(dǎo)致cookie無法及時更新或者完全無法更新。
目前觀察該問題出現(xiàn)在iOS11.3~iOS12.2的系統(tǒng)悠汽。如果遇到相同問題可以首先排查是否有多個webView同時加載。
如果解問題難以發(fā)現(xiàn)或者解決成本較高,可以犧牲性能使用nonPersistentDataStore來暫時規(guī)避這個問題

iOS10方案

iOS10或以下系統(tǒng)還沒有提供httpCookieStorage茬射,我們需要使用WKUserScript注入JS代碼的方式進行cookie更新吆倦。

//  創(chuàng)建userContentController時添加UserScript
- (WKUserContentController *)userContentController {
    if (!_userContentController) {
        WKUserContentController *userContentController = [WKUserContentController new];
        if (@available(iOS 11.0, *)) {
            [userContentController addUserScript:[WebViewCookieUtil cookieScriptForIOS10AndEarlier]];
        }
        _userContentController = userContentController;
    }
    return _userContentController;
}

NSHTTPCookieStorage中的cookie轉(zhuǎn)換成JS設(shè)置語句

//  WKCookieUtil
+ (WKUserScript *)cookieScriptForIOS10AndEarlier {
    NSMutableString *temp = @"".mutableCopy;
    NSHTTPCookieStorage *httpCookieStorage = [NSHTTPCookieStorage sharedHTTPCookieStorage];
    [httpCookieStorage.cookies enumerateObjectsUsingBlock:^(NSHTTPCookie * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        if (![obj.domain isEqualToString:domainForThisApp]) {
            return ;
        }
        NSString *foo = [NSString stringWithFormat:@"%@=%@;domain=%@;path=/",obj.name, obj.value, domainForThisApp];
        [temp appendFormat:@"document.cookie = '%@';\n", foo];
    }];
    WKUserScript * cookieScript = [[WKUserScript alloc] initWithSource:[temp copy] injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:NO];
    return cookieScript;
}

即使已經(jīng)添加WKUserScript,首個請求仍然需要通過設(shè)置request的httpHeader來帶上cookie信息。

- (void)loadURLWithCachePolicy:(NSURLRequestCachePolicy)cachePolicy {
   //   ...
    if (@available(iOS 11.0, *)) {
        //  ...
    } else {
        NSString *cookieStr = [WebViewCookieUtil cookieStringForFirstRequestIOS10AndEarlier];
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:self.URL cachePolicy:cachePolicy timeoutInterval:10];
        [request addValue:cookieStr forHTTPHeaderField:@"Cookie"];
        [self.wkWebView loadRequest:request];
    }
}
//  WKCookieUtil
+ (NSString *)cookieStringForFirstRequestIOS10AndEarlier {
    NSHTTPCookieStorage *httpCookieStorage = [NSHTTPCookieStorage sharedHTTPCookieStorage];
    
    return [self cookieStringForCurrentDomain:httpCookieStorage.cookies];
}

+ (NSString *)cookieStringForCurrentDomain:(NSArray<NSHTTPCookie *> *)cookies {
    NSMutableString *temp = @"".mutableCopy;
    [temp appendFormat:@"domain=%@; path=%@; ", domainForThisApp, @"/"];
    [cookies enumerateObjectsUsingBlock:^(NSHTTPCookie * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        if (![obj.domain isEqualToString:domainForThisApp]) {
            return ;
        }
        [temp appendFormat:@"%@=%@; ", obj.name, obj.value];
    }];
    return [temp copy];
}

補充:
如果當(dāng)前webView使用過程中需要更新cookie,必須刪除之前的WKUserScript,重新添加新的WKUserScript爵憎,然后reload整個頁面。

參考資料

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末婚瓜,一起剝皮案震驚了整個濱河市宝鼓,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌巴刻,老刑警劉巖愚铡,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異胡陪,居然都是意外死亡沥寥,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門柠座,熙熙樓的掌柜王于貴愁眉苦臉地迎上來邑雅,“玉大人,你說我怎么就攤上這事妈经』匆埃” “怎么了?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵吹泡,是天一觀的道長骤星。 經(jīng)常有香客問我,道長爆哑,這世上最難降的妖魔是什么洞难? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮揭朝,結(jié)果婚禮上队贱,老公的妹妹穿的比我還像新娘。我一直安慰自己萝勤,他們只是感情好露筒,可當(dāng)我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著敌卓,像睡著了一般。 火紅的嫁衣襯著肌膚如雪伶氢。 梳的紋絲不亂的頭發(fā)上趟径,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天瘪吏,我揣著相機與錄音,去河邊找鬼蜗巧。 笑死掌眠,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的幕屹。 我是一名探鬼主播蓝丙,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼望拖!你這毒婦竟也來了渺尘?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤说敏,失蹤者是張志新(化名)和其女友劉穎鸥跟,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體盔沫,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡医咨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了架诞。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拟淮。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖谴忧,靈堂內(nèi)的尸體忽然破棺而出很泊,到底是詐尸還是另有隱情,我是刑警寧澤俏蛮,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布撑蚌,位于F島的核電站,受9級特大地震影響搏屑,放射性物質(zhì)發(fā)生泄漏争涌。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一辣恋、第九天 我趴在偏房一處隱蔽的房頂上張望亮垫。 院中可真熱鬧,春花似錦伟骨、人聲如沸饮潦。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽继蜡。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間稀并,已是汗流浹背仅颇。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留碘举,地道東北人忘瓦。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像引颈,于是被迫代替她去往敵國和親耕皮。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,864評論 2 354

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