36

不幸的是,有许多适用于 Android 的 cookie 管理器。的 cookieHttpURLConnection由 维护java.net.CookieManager, 的 cookieWebView由 维护android.webkit.CookieManager。这些 cookie 存储库是独立的,需要手动同步。

我的应用同时使用HttpURLConnections和显示WebViews(它是原生 HTML 混合)。自然,我希望两者都共享所有 cookie - 所以我将有一个透明的会话。

进一步来说:

  1. 当在 HttpURLConnection 中设置/更改 cookie 时,我希望 WebViews 也能看到此更改。
  2. 在 WebView 中设置/更改 cookie 时,我希望下一个 HttpURLConnections 也能看到此更改。

简单地说 - 我正在寻找双向同步。或者更好的是,让它们都使用相同的 cookie 存储库。您可以假设两者同时处于活动状态(例如在不同的选项卡上)。

问题:

  1. 有没有办法让两者都使用相同的 cookie 存储库?

  2. 如果不是,建议进行手动同步的做法是什么?我应该什么时候同步以及如何同步?

相关问题:这个问题解决了一个类似的问题,但只实现了单向同步(HttpURLConnection -> WebView)。

迄今为止我最好的想法:我真的很想避免手动同步,所以我试着思考如何让两者使用同一个存储库。也许我可以创建自己的核心处理程序来扩展java.net.CookieManager. 我将使用java.net.CookieHandler.setDefault(). 它的实现将是android.webkit.CookieManager处理程序实例的代理(对于每个函数,我将简单地访问 webkit 管理器)。

4

1 回答 1

66

我已经实现了我自己的想法。它实际上很酷。我创建了自己的实现,java.net.CookieManager它将所有请求转发到 WebView 的 webkit android.webkit.CookieManager。这意味着不需要同步,并且 HttpURLConnection 使用与 WebView相同的cookie 存储。

WebkitCookieManagerProxy 类:

import java.io.IOException;
import java.net.CookieManager;
import java.net.CookiePolicy;
import java.net.CookieStore;
import java.net.URI;
import java.util.Arrays;
import java.util.List;
import java.util.Map;

public class WebkitCookieManagerProxy extends CookieManager 
{
    private android.webkit.CookieManager webkitCookieManager;

    public WebkitCookieManagerProxy()
    {
        this(null, null);
    }

    public WebkitCookieManagerProxy(CookieStore store, CookiePolicy cookiePolicy)
    {
        super(null, cookiePolicy);

        this.webkitCookieManager = android.webkit.CookieManager.getInstance();
    }

    @Override
    public void put(URI uri, Map<String, List<String>> responseHeaders) throws IOException 
    {
        // make sure our args are valid
        if ((uri == null) || (responseHeaders == null)) return;

        // save our url once
        String url = uri.toString();

        // go over the headers
        for (String headerKey : responseHeaders.keySet()) 
        {
            // ignore headers which aren't cookie related
            if ((headerKey == null) || !(headerKey.equalsIgnoreCase("Set-Cookie2") || headerKey.equalsIgnoreCase("Set-Cookie"))) continue;

            // process each of the headers
            for (String headerValue : responseHeaders.get(headerKey))
            {
                this.webkitCookieManager.setCookie(url, headerValue);
            }
        }
    }

    @Override
    public Map<String, List<String>> get(URI uri, Map<String, List<String>> requestHeaders) throws IOException 
    {
        // make sure our args are valid
        if ((uri == null) || (requestHeaders == null)) throw new IllegalArgumentException("Argument is null");

        // save our url once
        String url = uri.toString();

        // prepare our response
        Map<String, List<String>> res = new java.util.HashMap<String, List<String>>();

        // get the cookie
        String cookie = this.webkitCookieManager.getCookie(url);

        // return it
        if (cookie != null) res.put("Cookie", Arrays.asList(cookie));
        return res;
    }

    @Override
    public CookieStore getCookieStore() 
    {
        // we don't want anyone to work with this cookie store directly
        throw new UnsupportedOperationException();
    }
}

并通过在您的应用程序初始化中执行此操作来使用它:

android.webkit.CookieSyncManager.createInstance(appContext);
// unrelated, just make sure cookies are generally allowed
android.webkit.CookieManager.getInstance().setAcceptCookie(true);

// magic starts here
WebkitCookieManagerProxy coreCookieManager = new WebkitCookieManagerProxy(null, java.net.CookiePolicy.ACCEPT_ALL);
java.net.CookieHandler.setDefault(coreCookieManager);

测试

我最初的测试表明这运行良好。我看到了 WebView 和 HttpURLConnection 之间共享的 cookie。我希望我不会遇到任何问题。如果您尝试此操作并发现任何问题,请发表评论。

于 2013-08-05T15:12:00.120 回答