iOS开发WKWebView Cookie的读取与写入,与UIWebView的Cookie共享

一个群友说WKWebview的Cookie存取很鸡肋,之前以为和UIWebView一个原理,但是抱着对技术要有一颗严谨的心...我探究了一番,结果让我惊讶,还有就是让我感觉到WKWebView很鸡肋,很封闭,希望之后的iOS SDK版本更新会提供更多的操作接口与特性吧

NSHTTPCookieStorage和NSHttpCookie

NSHTTPCookieStorage 实现了一个管理Cookie的单例对象(只有一个实例),每个Cookie都是NSHTTPCookie类的实例,做为一个规则,Cookie在所有应用 之间共享并在不同进程之间保持同步。Session Cookie(一个isSessionOnly方法返回YES的Cookie)只能在单一进程中使用。

UIWebView Cookie

同一个应用,不同UIWebView之间的Cookie是自动同步的。并且可以被其他网络类访问比如NSURLConnection,AFNetworking。

它们都是保存在NSHTTPCookieStorage容器中。 当UIWebView加载一个URL的时候,在加载完成时候,Http Response,对Cookie进行写入,更新或者删除,结果更新Cookie到NSHTTPCookieStorage存储容器中。

WKWebView Cookie

NSURLCache和NSHTTPCookieStroage无法操作(WKWebView)WebCore进程的缓存和Cookie。

WKWebView实例将会忽略iOS11之前任何的默认网络存储器(NSURLCache, NSHTTPCookieStorage, NSCredentialStorage) 和一些标准的自定义网络请求类(NSURLProtocol,等等.),iOS11之后新增了WKHTTPCookieStore,与NSHTTPCookieStorage职能一致。

WKWebView实例不会把Cookie存入到App标准的的Cookie容器(NSHTTPCookieStorage)中,而是存在了NSHTTPCookieStorage中(iOS12以后),因为 NSURLSession/NSURLConnection等网络请求使用NSHTTPCookieStorage进行访问Cookie,所以不能访问WKWebView的Cookie,现象就是WKWebView存了Cookie,其他的网络类如NSURLSession/NSURLConnection却看不到。,

与Cookie相同的情况就是WKWebView的缓存,凭据等。WKWebView都拥有自己的私有存储,因此和标准Cocoa网络类兼容的不是那么好。

你也不能自定义requests(增加自己的http header,更改已经存在的header)使用自定义的 URL schemes等等,因为NSURLProtocol也是不支持WKWebView的。

http://stackoverflow.com/questions/24464397/how-can-i-retrieve-a-file-using-WKWebView

WKWebView Cookie 写入

使用传统的UIWebView时代的方法写入经过测试无效,我们盼望着苹果会在新的SDK版本中增加更多特性,但是问题是我们的App不仅仅是针对于新的iOS版本,所以老的版本还需要不同的方法解决问题。

JS注入1

WKUserContentController* userContentController = WKUserContentController.new;

WKUserScript * cookieScript = [[WKUserScript alloc] initWithSource: @"document.cookie ='TeskCookieKey1=TeskCookieValue1';document.cookie = 'TeskCookieKey2=TeskCookieValue2';"injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:NO];

[userContentController addUserScript:cookieScript];

WKWebViewConfiguration* webViewConfig = WKWebViewConfiguration.new;

webViewConfig.userContentController = userContentController;

WKWebView * webView = [[WKWebView alloc] initWithFrame:CGRectMake(/*set your values*/) configuration:webViewConfig];

JS注入2

- (void)webView:(WKWebView *)webView didFinishNavigation:(WKNavigation *)navigation {

    [webView evaluateJavaScript:@"document.cookie ='TeskCookieKey1=TeskCookieValue1';" completionHandler:^(id result, NSError *error) {

        //...

    }];

}

NSMutableURLRequest

NSMutableURLRequest *request= [NSMutableURLRequest requestWithURL:[NSURL URLWithString:@"http://dev.skyfox.org/cookie.php"]];

//[request setHTTPShouldHandleCookies:YES];

[request setValue:[NSString stringWithFormat:@"%@=%@",@"testwkcookie", @"testwkcookievalue"] forHTTPHeaderField:@"Cookie"];

注意:

JS注入的Cookie,比如PHP代码在Cookie容器中取是取不到的, javascript document.cookie能读取到,浏览器中也能看到。

NSMutableURLRequest 注入的PHP等动态语言直接能从$_COOKIE对象中获取到,但是js读取不到,浏览器也看不到

所以合理的办法让js,php,浏览器都能读取到相同的Cookie方法就是创建WebView的时候javascript注入Cookie,一开始发送NSMutableURLRequest请求的时候也要加上Cookie,并且保证两个地方的设置的cookie一致。


//初始化

WKUserContentController* userContentController = WKUserContentController.new;

WKUserScript * cookieScript = [[WKUserScript alloc] initWithSource: @"document.cookie ='TeskCookieKey3=TeskCookieValue3';"injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:NO];


[userContentController addUserScript:cookieScript];

WKWebViewConfiguration* webViewConfig = WKWebViewConfiguration.new;

webViewConfig.userContentController = userContentController;

WKWebView * webView = [[WKWebView alloc] initWithFrame:CGRectMake(/*set your values*/) configuration:webViewConfig];


//请求

NSMutableURLRequest *request= [NSMutableURLRequest requestWithURL:[NSURL URLWithString:@"http://dev.skyfox.org/cookie.php"]];

//[request setHTTPShouldHandleCookies:YES];

[request setValue:[NSString stringWithFormat:@"%@=%@",@"TeskCookieKey3", @"TeskCookieValue3"] forHTTPHeaderField:@"Cookie"];

[webView loadRequest:request];


WKWebsiteDataStore

WKWebsiteDataStore在iOS 9和OS X 10.11中引入,是一个新的API,它用于管理一个网站站点存储的数据,例如Cookies,它是你网页的 WKWebViewConfiguration上的一个可读写的属性。你可以根据类型或者时间来删除数据,例如Cookies和缓存,你可以用非持久性数 据存储来改变配置。

WKWebView Cookie 读取

1.http respone headerfields

因为cookie都存在http respone的headerfields,找到能获得respone的WKWebView回调,打印..

iOS12以后必须使用WKHTTPCookieStore获取allHeaderFields中已经读取不到cookie了

- (void)webView:(WKWebView *)webView decidePolicyForNavigationResponse:(WKNavigationResponse *)navigationResponse decisionHandler:(void (^)(WKNavigationResponsePolicy))decisionHandler{

    NSHTTPURLResponse *response = (NSHTTPURLResponse *)navigationResponse.response;

    NSArray *cookies =[NSHTTPCookie cookiesWithResponseHeaderFields:[response allHeaderFields] forURL:response.URL];

    //读取wkwebview中的cookie 方法1

    for (NSHTTPCookie *cookie in cookies) {

//        [[NSHTTPCookieStorage sharedHTTPCookieStorage] setCookie:cookie];

        NSLog(@"wkwebview中的cookie:%@", cookie);


    }

    //读取wkwebview中的cookie 方法2 读取Set-Cookie字段

    NSString *cookieString = [[response allHeaderFields] valueForKey:@"Set-Cookie"];

    NSLog(@"wkwebview中的cookie:%@", cookieString);


    //看看存入到了NSHTTPCookieStorage了没有

    NSHTTPCookieStorage *cookieJar2 = [NSHTTPCookieStorage sharedHTTPCookieStorage];

    for (NSHTTPCookie *cookie in cookieJar2.cookies) {

        NSLog(@"NSHTTPCookieStorage中的cookie%@", cookie);

    }


////iOS11也有这种获取方式,但是实测iOS11系统可以在response里面直接获取到,只有iOS12获取不到

   if (@available(iOS 12.0, *)) {

        WKHTTPCookieStore *cookieStore = webView.configuration.websiteDataStore.httpCookieStore;

        [cookieStore getAllCookies:^(NSArray* cookies) {

         }];

    }else {

        NSHTTPURLResponse *response = (NSHTTPURLResponse *)navigationResponse.response;

        NSArray *cookies =[NSHTTPCookie cookiesWithResponseHeaderFields:[response allHeaderFields] forURL:response.URL];

    }


    decisionHandler(WKNavigationResponsePolicyAllow);

}

结果确实读取到了Cookie,但是打印NSHTTPCookieStorage单例中Cookie发现并没有任何Cookie。


2.WKWebsiteDataStore iOS9

// 页面加载完成之后调用

- (void)webView:(WKWebView *)webView didFinishNavigation:(WKNavigation *)navigation{

    WKWebsiteDataStore *dataStore = [WKWebsiteDataStore defaultDataStore];

    [dataStore fetchDataRecordsOfTypes:[WKWebsiteDataStore allWebsiteDataTypes]

                     completionHandler:^(NSArray * __nonnull records) {

                         for (WKWebsiteDataRecord *record  in records)

                         {

                             NSLog(@"WKWebsiteDataRecord:%@",[record description]);

                         }

                     }];


}

WKHTTPCookieStore iOS11

在iOS11中,苹果新增加了用于WKWebview Cookie存取操作的WKHTTPCookieStore。

iOS11也有这种获取方式,但是实践测试后iOS11系统可以在response里面直接获取到,只有iOS12获取不到

if (@available(iOS 12.0, *)) {

        WKHTTPCookieStore *cookieStore = webView.configuration.websiteDataStore.httpCookieStore;

        [cookieStore getAllCookies:^(NSArray* cookies) {


        }];

    }

A WKHTTPCookieStore object allows managing the HTTP cookies associated with a particular WKWebsiteDataStore.

Demo地址:https://github.com/shaojiankui/iOS-Cookie

相关链接:

iOS HTTP网络请求Cookie的读取与写入(NSHTTPCookieStorage) - 天狐博客

iOS开发-打通UIWebView和WKWebView的Cookie 

http://stackoverflow.com/questions/24464397/how-can-i-retrieve-a-file-using-WKWebView

你可能感兴趣的:(iOS开发WKWebView Cookie的读取与写入,与UIWebView的Cookie共享)