Android混合应用程序+Volley、Cookies和WebView

3
我已经花费超过15个小时研究了这个问题:我正在构建一个混合应用程序,使用Volley框架(本机视图)进行登录。一旦成功登录,我会获取响应头部,提取Cookie并将其保存到我的sharedprefs中。在本机主屏幕上,包含多个Webview链接,如何将从登录接收到的Cookie传递到Webview中?网络上90%的答案使用CookieSyncManager,但它已被弃用。我尝试了java.net.CookieManager也没有成功。以下是我的代码。
  @Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    mWebView = new WebView(this);

    setContentView(mWebView);

    cookies = pref.getString(Const.COOKIE_KEY,"null");
    userID = pref.getString(Const.USER_ID_KEY,"null");
    mUrl = Const.PERFORMANCE_WEBVIEW_LINK + Const.USER_ID;
    String cookieText = Const.COOKIE_KEY + "=" + cookies;

    //Approach A Environment for the Cookies
    //Does not work
    cookieSync = CookieSyncManager.createInstance(this);
    cookieManager = CookieManager.getInstance();
    cookieManager.removeSessionCookie();
    cookieManager.setCookie(mUrl, cookieText);
    cookieSync.sync();

    SystemClock.sleep(10000);

    /*APPROACH B, sending the Cookies with header
      ##Did not Work##
    final Map<String, String> headers = new HashMap<>();

    Log.d("cookie", cookieText);
    headers.put("Cookie",cookieText);
    */

    if(cookies.equals("null") || userID.equals("null")) {
        Toast.makeText(PerformanceWebview.this, "Error", Toast.LENGTH_SHORT).show();
        //Logging Out
        Intent intent = new Intent(this, LoginActivity.class);
        startActivity(intent);
    }
    else {
        //mWebView.loadUrl(mUrl,headers);
        mWebView.loadUrl(mUrl);
        Toast.makeText(PerformanceWebview.this,cookieText, Toast.LENGTH_SHORT).show();
        Log.d("URL", "URL: " + mUrl);
    }


}

我还尝试了将WebClient传递给WebView,并覆盖其shouldOverrideURl方法并将标头传递给它。另一种方法是使用WebSettings并传递ChromeClient。这里的任何答案都似乎不起作用。

找到了一个解决方案:打电话给客户,告诉他们的后端开发人员允许一种避免Cookies并接受我客户请求的方式。 - Makaveli
2个回答

2

首先确保Volley使用cookies:(请查看https://dev59.com/nmQn5IYBdhLWcg3woIYF#21271347 )

// Make volley remember cookies

// Do this only once on app startup, and keep the reference to the cookiemanager.
// I'm saving it on an App class, but you can do something different.

App.cookieManager = new CookieManager();
CookieHandler.setDefault(App.cookieManager);

// Note, we are using the java.net.CookieManager above.

然后,在使用Volley进行登录调用后,将cookie同步到WebView中:
// Sync cookies to webview
android.webkit.CookieManager webkitCookies = android.webkit.CookieManager.getInstance();

for (HttpCookie cookie : App.cookieManager.getCookieStore().getCookies()) {
    webkitCookies.setCookie(cookie.getDomain(), cookie.getName() + "=" + cookie.getValue());
}

if (Build.VERSION.SDK_INT >= 21) {
    webkitCookies.flush();
} else {
    CookieSyncManager.getInstance().sync();
}

0
class CookieStore_ implements CookieStore{
/** this map may have null keys! */
private final Map<URI, List<HttpCookie>> map = new HashMap<URI, List<HttpCookie>>();
private android.webkit.CookieManager manager;

public CookieStore_() {
    manager = android.webkit.CookieManager.getInstance();
}

public synchronized void add(URI uri, HttpCookie cookie) {
    if (cookie == null) {
        throw new NullPointerException("cookie == null");
    }

    uri = cookiesUri(uri);
    //add cookie to the CookieManager,be sure you have called
    //CookieSyncManager.createInstance(context) if your android version           
    //is lower than Lollipop
    manager.setCookie(uri.toString(),cookie.toString());

    List<HttpCookie> cookies = map.get(uri);
    if (cookies == null) {
        cookies = new ArrayList<HttpCookie>();
        map.put(uri, cookies);
    } else {
        cookies.remove(cookie);
    }
    cookies.add(cookie);
}

private URI cookiesUri(URI uri) {
    if (uri == null) {
        return null;
    }
    try {
        return new URI("http", uri.getHost(), null, null);
    } catch (URISyntaxException e) {
        return uri; // probably a URI with no host
    }
}

public synchronized List<HttpCookie> get(URI uri) {
    if (uri == null) {
        throw new NullPointerException("uri == null");
    }

    List<HttpCookie> result = new ArrayList<HttpCookie>();

    // get cookies associated with given URI. If none, returns an empty list
    List<HttpCookie> cookiesForUri = map.get(uri);
    if (cookiesForUri != null) {
        for (Iterator<HttpCookie> i = cookiesForUri.iterator(); i.hasNext(); ) {
            HttpCookie cookie = i.next();
            if (cookie.hasExpired()) {
                i.remove(); // remove expired cookies
            } else {
                result.add(cookie);
            }
        }
    }

    // get all cookies that domain matches the URI
    for (Map.Entry<URI, List<HttpCookie>> entry : map.entrySet()) {
        if (uri.equals(entry.getKey())) {
            continue; // skip the given URI; we've already handled it
        }

        List<HttpCookie> entryCookies = entry.getValue();
        for (Iterator<HttpCookie> i = entryCookies.iterator(); i.hasNext(); ) {
            HttpCookie cookie = i.next();
            if (!HttpCookie.domainMatches(cookie.getDomain(), uri.getHost())) {
                continue;
            }
            if (cookie.hasExpired()) {
                i.remove(); // remove expired cookies
            } else if (!result.contains(cookie)) {
                result.add(cookie);
            }
        }
    }

    return Collections.unmodifiableList(result);
}

public synchronized List<HttpCookie> getCookies() {
    List<HttpCookie> result = new ArrayList<HttpCookie>();
    for (List<HttpCookie> list : map.values()) {
        for (Iterator<HttpCookie> i = list.iterator(); i.hasNext(); ) {
            HttpCookie cookie = i.next();
            if (cookie.hasExpired()) {
                i.remove(); // remove expired cookies
            } else if (!result.contains(cookie)) {
                result.add(cookie);
            }
        }
    }
    return Collections.unmodifiableList(result);
}

public synchronized List<URI> getURIs() {
    List<URI> result = new ArrayList<URI>(map.keySet());
    result.remove(null); // sigh
    return Collections.unmodifiableList(result);
}

public synchronized boolean remove(URI uri, HttpCookie cookie) {
    if (cookie == null) {
        throw new NullPointerException("cookie == null");
    }

    List<HttpCookie> cookies = map.get(cookiesUri(uri));
    if (cookies != null) {
        return cookies.remove(cookie);
    } else {
        return false;
    }
}

public synchronized boolean removeAll() {
    boolean result = !map.isEmpty();
    map.clear();
    return result;
}

public void clearCookies(){
    map.clear();
}

}

这个类是从CookieStoreImpl复制而来,在其中添加了一个android.webkit.CookieManager。当调用add(URI uri, HttpCookie cookie)方法时,它会将cookie添加到cookieManager中。然后当webview加载具有与cookieManager匹配项的URL时,webview会将匹配的cookie添加到请求头中。

下面是一个辅助类

public class CookieUtil {
private CookieManager manager;
private CookieStore_ cookieStore_;
private CookieSyncManager syncManager;
private static CookieUtil cookieUtil;
private boolean isInitialed = false;

private CookieUtil(Context context) {
    manager = CookieManager.getInstance();

    if (!Util.hasLollipop()){
        syncManager = CookieSyncManager.createInstance(context);
    }
    cookieStore_ = new CookieStore_();
}

public void clearCookies(){
    if (manager.hasCookies()){
        if (Util.hasLollipop()){
            manager.removeAllCookies(null);
        }else {
            manager.removeAllCookie();
        }
    }
    cookieStore_.clearCookies();
}

public static CookieUtil getCookieUtil(Context context){
    if (cookieUtil == null)
        cookieUtil = new CookieUtil(context);
    return cookieUtil;
}

public void sync(){
    if (Util.hasLollipop()){
        manager.flush();
    }else {
        syncManager.sync();
    }
}

public void setThirdPartyCookieAcceptable(WebView webView){
    if (Util.hasLollipop()){
        manager.setAcceptThirdPartyCookies(webView,true);
    }
}

public void initCookieHandler(){
    if (isInitialed)
        return;
    isInitialed = true;
    CookieHandler.setDefault(new java.net.CookieManager(cookieStore_, CookiePolicy.ACCEPT_ORIGINAL_SERVER));
}

}

如果您不需要将Cookie保存到本地存储中,您可以通过一行代码将Volley的Cookie共享到WebView中。
CookieUtil.getCookieUtil(context).initCookieHandler();

在注销时不要忘记调用clearCookies()函数


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接