package com.ysvideo.zhibo.lib.common.okhttp.cookie.store;
|
|
import android.content.Context;
|
import android.content.SharedPreferences;
|
import android.text.TextUtils;
|
import android.util.Log;
|
|
import java.io.ByteArrayInputStream;
|
import java.io.ByteArrayOutputStream;
|
import java.io.IOException;
|
import java.io.ObjectInputStream;
|
import java.io.ObjectOutputStream;
|
import java.util.ArrayList;
|
import java.util.Collection;
|
import java.util.HashMap;
|
import java.util.List;
|
import java.util.Locale;
|
import java.util.Map;
|
import java.util.concurrent.ConcurrentHashMap;
|
|
import okhttp3.Cookie;
|
import okhttp3.HttpUrl;
|
|
/**
|
* <pre>
|
* OkHttpClient client = new OkHttpClient.Builder()
|
* .cookieJar(new JavaNetCookieJar(new CookieManager(
|
* new PersistentCookieStore(getApplicationContext()),
|
* CookiePolicy.ACCEPT_ALL))
|
* .build();
|
*
|
* </pre>
|
* <p/>
|
* from http://stackoverflow.com/questions/25461792/persistent-cookie-store-using-okhttp-2-on-android
|
* <p/>
|
* <br/>
|
* A persistent cookie store which implements the Apache HttpClient CookieStore interface.
|
* Cookies are stored and will persist on the user's device between application sessions since they
|
* are serialized and stored in SharedPreferences. Instances of this class are
|
* designed to be used with AsyncHttpClient#setCookieStore, but can also be used with a
|
* regular old apache HttpClient/HttpContext if you prefer.
|
*/
|
public class PersistentCookieStore implements CookieStore
|
{
|
|
private static final String LOG_TAG = "PersistentCookieStore";
|
private static final String COOKIE_PREFS = "CookiePrefsFile";
|
private static final String COOKIE_NAME_PREFIX = "cookie_";
|
|
private final HashMap<String, ConcurrentHashMap<String, Cookie>> cookies;
|
private final SharedPreferences cookiePrefs;
|
|
/**
|
* Construct a persistent cookie store.
|
*
|
* @param context Context to attach cookie store to
|
*/
|
public PersistentCookieStore(Context context)
|
{
|
cookiePrefs = context.getSharedPreferences(COOKIE_PREFS, 0);
|
cookies = new HashMap<String, ConcurrentHashMap<String, Cookie>>();
|
|
// Load any previously stored cookies into the store
|
Map<String, ?> prefsMap = cookiePrefs.getAll();
|
for (Map.Entry<String, ?> entry : prefsMap.entrySet())
|
{
|
if (entry.getValue() != null && !((String) entry.getValue()).startsWith(COOKIE_NAME_PREFIX))
|
{
|
String[] cookieNames = TextUtils.split((String) entry.getValue(), ",");
|
for (String name : cookieNames)
|
{
|
String encodedCookie = cookiePrefs.getString(COOKIE_NAME_PREFIX + name, null);
|
if (encodedCookie != null)
|
{
|
Cookie decodedCookie = decodeCookie(encodedCookie);
|
if (decodedCookie != null)
|
{
|
if (!cookies.containsKey(entry.getKey()))
|
cookies.put(entry.getKey(), new ConcurrentHashMap<String, Cookie>());
|
cookies.get(entry.getKey()).put(name, decodedCookie);
|
}
|
}
|
}
|
|
}
|
}
|
}
|
|
protected void add(HttpUrl uri, Cookie cookie)
|
{
|
String name = getCookieToken(cookie);
|
|
if (cookie.persistent())
|
{
|
if (!cookies.containsKey(uri.host()))
|
{
|
cookies.put(uri.host(), new ConcurrentHashMap<String, Cookie>());
|
}
|
cookies.get(uri.host()).put(name, cookie);
|
} else
|
{
|
if (cookies.containsKey(uri.host()))
|
{
|
cookies.get(uri.host()).remove(name);
|
}else
|
{
|
return ;
|
}
|
}
|
|
// Save cookie into persistent store
|
SharedPreferences.Editor prefsWriter = cookiePrefs.edit();
|
prefsWriter.putString(uri.host(), TextUtils.join(",", cookies.get(uri.host()).keySet()));
|
prefsWriter.putString(COOKIE_NAME_PREFIX + name, encodeCookie(new SerializableHttpCookie(cookie)));
|
prefsWriter.apply();
|
}
|
|
protected String getCookieToken(Cookie cookie)
|
{
|
return cookie.name() + cookie.domain();
|
}
|
|
@Override
|
public void add(HttpUrl uri, List<Cookie> cookies)
|
{
|
for (Cookie cookie : cookies)
|
{
|
add(uri, cookie);
|
}
|
}
|
|
@Override
|
public List<Cookie> get(HttpUrl uri)
|
{
|
ArrayList<Cookie> ret = new ArrayList<Cookie>();
|
if (cookies.containsKey(uri.host()))
|
{
|
Collection<Cookie> cookies = this.cookies.get(uri.host()).values();
|
for (Cookie cookie : cookies)
|
{
|
if (isCookieExpired(cookie))
|
{
|
remove(uri, cookie);
|
} else
|
{
|
ret.add(cookie);
|
}
|
}
|
}
|
|
return ret;
|
}
|
|
private static boolean isCookieExpired(Cookie cookie)
|
{
|
return cookie.expiresAt() < System.currentTimeMillis();
|
}
|
|
@Override
|
public boolean removeAll()
|
{
|
SharedPreferences.Editor prefsWriter = cookiePrefs.edit();
|
prefsWriter.clear();
|
prefsWriter.apply();
|
cookies.clear();
|
return true;
|
}
|
|
|
@Override
|
public boolean remove(HttpUrl uri, Cookie cookie)
|
{
|
String name = getCookieToken(cookie);
|
|
if (cookies.containsKey(uri.host()) && cookies.get(uri.host()).containsKey(name))
|
{
|
cookies.get(uri.host()).remove(name);
|
|
SharedPreferences.Editor prefsWriter = cookiePrefs.edit();
|
if (cookiePrefs.contains(COOKIE_NAME_PREFIX + name))
|
{
|
prefsWriter.remove(COOKIE_NAME_PREFIX + name);
|
}
|
prefsWriter.putString(uri.host(), TextUtils.join(",", cookies.get(uri.host()).keySet()));
|
prefsWriter.apply();
|
|
return true;
|
} else
|
{
|
return false;
|
}
|
}
|
|
@Override
|
public List<Cookie> getCookies()
|
{
|
ArrayList<Cookie> ret = new ArrayList<Cookie>();
|
for (String key : cookies.keySet())
|
ret.addAll(cookies.get(key).values());
|
|
return ret;
|
}
|
|
|
protected String encodeCookie(SerializableHttpCookie cookie)
|
{
|
if (cookie == null)
|
return null;
|
ByteArrayOutputStream os = new ByteArrayOutputStream();
|
try
|
{
|
ObjectOutputStream outputStream = new ObjectOutputStream(os);
|
outputStream.writeObject(cookie);
|
} catch (IOException e)
|
{
|
Log.d(LOG_TAG, "IOException in encodeCookie", e);
|
return null;
|
}
|
|
return byteArrayToHexString(os.toByteArray());
|
}
|
|
protected Cookie decodeCookie(String cookieString)
|
{
|
byte[] bytes = hexStringToByteArray(cookieString);
|
ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(bytes);
|
Cookie cookie = null;
|
try
|
{
|
ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
|
cookie = ((SerializableHttpCookie) objectInputStream.readObject()).getCookie();
|
} catch (IOException e)
|
{
|
Log.d(LOG_TAG, "IOException in decodeCookie", e);
|
} catch (ClassNotFoundException e)
|
{
|
Log.d(LOG_TAG, "ClassNotFoundException in decodeCookie", e);
|
}
|
|
return cookie;
|
}
|
|
/**
|
* Using some super basic byte array <-> hex conversions so we don't have to rely on any
|
* large Base64 libraries. Can be overridden if you like!
|
*
|
* @param bytes byte array to be converted
|
* @return string containing hex values
|
*/
|
protected String byteArrayToHexString(byte[] bytes)
|
{
|
StringBuilder sb = new StringBuilder(bytes.length * 2);
|
for (byte element : bytes)
|
{
|
int v = element & 0xff;
|
if (v < 16)
|
{
|
sb.append('0');
|
}
|
sb.append(Integer.toHexString(v));
|
}
|
return sb.toString().toUpperCase(Locale.US);
|
}
|
|
/**
|
* Converts hex values from strings to byte arra
|
*
|
* @param hexString string of hex-encoded values
|
* @return decoded byte array
|
*/
|
protected byte[] hexStringToByteArray(String hexString)
|
{
|
int len = hexString.length();
|
byte[] data = new byte[len / 2];
|
for (int i = 0; i < len; i += 2)
|
{
|
data[i / 2] = (byte) ((Character.digit(hexString.charAt(i), 16) << 4) + Character.digit(hexString.charAt(i + 1), 16));
|
}
|
return data;
|
}
|
}
|