Як зробити запит http за допомогою файлів cookie на Android?


121

Я хотів би зробити запит http на віддалений сервер під час належної обробки файлів cookie (наприклад, зберігання файлів cookie, надісланих сервером, та надсилання цих файлів cookie, коли я роблю наступні запити). Було б непогано зберегти будь-які файли cookie, але насправді єдине, що мені важливо - це сесійне cookie.

З java.net виявляється, що кращим способом зробити це є використання java.net.CookieHandler (абстрактний базовий клас) та java.net.CookieManager (конкретна реалізація). Android має java.net.CookieHandler, але, схоже, він не має java.net.CookieManager.

Я міг би це зашифрувати вручну, оглянувши заголовки http, але, схоже, має бути простіший спосіб.

Який правильний спосіб робити http запити на Android, зберігаючи файли cookie?


Ви спробували org.apache.http.cookie ?
Джек Л.

3
Як і замітка, більш ніж через два роки: java.net.CookieManagerтепер підтримується в Android з версії 2.3 (рівень API 9): developer.android.com/reference/java/net/CookieManager.html
Slauma

Відповіді:


92

Виявляється, Google Android поставляється з Apache HttpClient 4.0, і я зміг зрозуміти, як це зробити за допомогою прикладу " Вхід на основі форми" в документах HttpClient :

https://github.com/apache/httpcomponents-client/blob/master/httpclient5/src/test/java/org/apache/hc/client5/http/examples/ClientFormLogin.java


import java.util.ArrayList;
import java.util.List;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.cookie.Cookie;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.protocol.HTTP;

/**
 * A example that demonstrates how HttpClient APIs can be used to perform
 * form-based logon.
 */
public class ClientFormLogin {

    public static void main(String[] args) throws Exception {

        DefaultHttpClient httpclient = new DefaultHttpClient();

        HttpGet httpget = new HttpGet("https://portal.sun.com/portal/dt");

        HttpResponse response = httpclient.execute(httpget);
        HttpEntity entity = response.getEntity();

        System.out.println("Login form get: " + response.getStatusLine());
        if (entity != null) {
            entity.consumeContent();
        }
        System.out.println("Initial set of cookies:");
        List<Cookie> cookies = httpclient.getCookieStore().getCookies();
        if (cookies.isEmpty()) {
            System.out.println("None");
        } else {
            for (int i = 0; i < cookies.size(); i++) {
                System.out.println("- " + cookies.get(i).toString());
            }
        }

        HttpPost httpost = new HttpPost("https://portal.sun.com/amserver/UI/Login?" +
                "org=self_registered_users&" +
                "goto=/portal/dt&" +
                "gotoOnFail=/portal/dt?error=true");

        List <NameValuePair> nvps = new ArrayList <NameValuePair>();
        nvps.add(new BasicNameValuePair("IDToken1", "username"));
        nvps.add(new BasicNameValuePair("IDToken2", "password"));

        httpost.setEntity(new UrlEncodedFormEntity(nvps, HTTP.UTF_8));

        response = httpclient.execute(httpost);
        entity = response.getEntity();

        System.out.println("Login form get: " + response.getStatusLine());
        if (entity != null) {
            entity.consumeContent();
        }

        System.out.println("Post logon cookies:");
        cookies = httpclient.getCookieStore().getCookies();
        if (cookies.isEmpty()) {
            System.out.println("None");
        } else {
            for (int i = 0; i < cookies.size(); i++) {
                System.out.println("- " + cookies.get(i).toString());
            }
        }

        // When HttpClient instance is no longer needed, 
        // shut down the connection manager to ensure
        // immediate deallocation of all system resources
        httpclient.getConnectionManager().shutdown();        
    }
}

11
чи можу я знати, як встановити файли cookie на URL-адресу запиту, щоб перевірити, чи дійсний він чи ні?
Praveen

ДЯКУЮ, що ви познайомили мене з BasicNameValuePairs. Вони мені допомогли.
bhekman

3
Я розумію The method getCookieStore() is undefined for the type HttpClient, чи повинен я змінитись List<Cookie> cookies = ((AbstractHttpClient) httpclient).getCookieStore().getCookies();? Тому що якщо я це зробити, це працює.
Франциско Корралес Моралес

@Praveen Погляньте тут, щоб зберегти файли cookie: stackoverflow.com/a/5989115/2615737
Франциско Корралес Моралес

@emmby: невдалий андроїд, застарілий модуль apache. Тож зараз його відповідь не корисна. Чи є інший спосіб зробити це за допомогою HttpURLConnection?
Мілад Фарідня

9

Файл cookie - це ще один заголовок HTTP. Ви завжди можете встановити його під час здійснення дзвінка HTTP за допомогою бібліотеки apache або за допомогою HTTPUrlConnection. У будь-якому випадку ви зможете читати та встановлювати HTTP-файли таким чином.

Ви можете прочитати цю статтю для отримання додаткової інформації.

Я можу поділитися своїм мирним кодом, щоб продемонструвати, як легко ви можете це зробити.

public static String getServerResponseByHttpGet(String url, String token) {

        try {
            HttpClient client = new DefaultHttpClient();
            HttpGet get = new HttpGet(url);
            get.setHeader("Cookie", "PHPSESSID=" + token + ";");
            Log.d(TAG, "Try to open => " + url);

            HttpResponse httpResponse = client.execute(get);
            int connectionStatusCode = httpResponse.getStatusLine().getStatusCode();
            Log.d(TAG, "Connection code: " + connectionStatusCode + " for request: " + url);

            HttpEntity entity = httpResponse.getEntity();
            String serverResponse = EntityUtils.toString(entity);
            Log.d(TAG, "Server response for request " + url + " => " + serverResponse);

            if(!isStatusOk(connectionStatusCode))
                return null;

            return serverResponse;

        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        } catch (ClientProtocolException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }

        return null;
    }

Це мені дуже допомагає, +1 get.setHeader ("Cookie", "PHPSESSID =" + token + ";"); зробіть роботу
Oussaki

@Hesam: невдалий андроїд, застарілий модуль apache. Тож зараз його відповідь не корисна. Чи існує інший спосіб зробити це за допомогою HttpURLConnection?
Мілад Фарідня

7

Оскільки бібліотека Apache застаріла , для тих, хто хоче користуватися HttpURLConncetion, я написав цей клас, щоб надіслати Get and Post Request за допомогою цієї відповіді :

public class WebService {

static final String COOKIES_HEADER = "Set-Cookie";
static final String COOKIE = "Cookie";

static CookieManager msCookieManager = new CookieManager();

private static int responseCode;

public static String sendPost(String requestURL, String urlParameters) {

    URL url;
    String response = "";
    try {
        url = new URL(requestURL);

        HttpURLConnection conn = (HttpURLConnection) url.openConnection();
        conn.setReadTimeout(15000);
        conn.setConnectTimeout(15000);
        conn.setRequestMethod("POST");

        conn.setRequestProperty("Content-Type", "application/json; charset=utf-8");

        if (msCookieManager.getCookieStore().getCookies().size() > 0) {
            //While joining the Cookies, use ',' or ';' as needed. Most of the server are using ';'
            conn.setRequestProperty(COOKIE ,
                    TextUtils.join(";", msCookieManager.getCookieStore().getCookies()));
        }

        conn.setDoInput(true);
        conn.setDoOutput(true);

        OutputStream os = conn.getOutputStream();
        BufferedWriter writer = new BufferedWriter(
                new OutputStreamWriter(os, "UTF-8"));

        if (urlParameters != null) {
            writer.write(urlParameters);
        }
        writer.flush();
        writer.close();
        os.close();

        Map<String, List<String>> headerFields = conn.getHeaderFields();
        List<String> cookiesHeader = headerFields.get(COOKIES_HEADER);

        if (cookiesHeader != null) {
            for (String cookie : cookiesHeader) {
                msCookieManager.getCookieStore().add(null, HttpCookie.parse(cookie).get(0));
            }
        }

        setResponseCode(conn.getResponseCode());

        if (getResponseCode() == HttpsURLConnection.HTTP_OK) {

            String line;
            BufferedReader br = new BufferedReader(new InputStreamReader(conn.getInputStream()));
            while ((line = br.readLine()) != null) {
                response += line;
            }
        } else {
            response = "";
        }
    } catch (Exception e) {
        e.printStackTrace();
    }

    return response;
}


// HTTP GET request
public static String sendGet(String url) throws Exception {

    URL obj = new URL(url);
    HttpURLConnection con = (HttpURLConnection) obj.openConnection();

    // optional default is GET
    con.setRequestMethod("GET");

    //add request header 
    con.setRequestProperty("User-Agent", "Mozilla");
    /*
    * /programming/16150089/how-to-handle-cookies-in-httpurlconnection-using-cookiemanager
    * Get Cookies form cookieManager and load them to connection:
     */
    if (msCookieManager.getCookieStore().getCookies().size() > 0) {
        //While joining the Cookies, use ',' or ';' as needed. Most of the server are using ';'
        con.setRequestProperty(COOKIE ,
                TextUtils.join(";", msCookieManager.getCookieStore().getCookies()));
    }

    /*
    * /programming/16150089/how-to-handle-cookies-in-httpurlconnection-using-cookiemanager
    * Get Cookies form response header and load them to cookieManager:
     */
    Map<String, List<String>> headerFields = con.getHeaderFields();
    List<String> cookiesHeader = headerFields.get(COOKIES_HEADER);
    if (cookiesHeader != null) {
        for (String cookie : cookiesHeader) {
            msCookieManager.getCookieStore().add(null, HttpCookie.parse(cookie).get(0));
        }
    }


    int responseCode = con.getResponseCode();

    BufferedReader in = new BufferedReader(
            new InputStreamReader(con.getInputStream()));
    String inputLine;
    StringBuffer response = new StringBuffer();

    while ((inputLine = in.readLine()) != null) {
        response.append(inputLine);
    }
    in.close();

    return response.toString();
}

public static void setResponseCode(int responseCode) {
    WebService.responseCode = responseCode;
    Log.i("Milad", "responseCode" + responseCode);
}


public static int getResponseCode() {
    return responseCode;
}
}

2
Гарний клас Мілад! Це мені дуже допомогло! Тепер мені було цікаво, чи хотів би я зберегти файл cookie, щоб він став доступним, якщо користувач убив додаток і запустив його знову, що саме потрібно зберігати і де (як)
Ki Jéy

1

Я не працюю з google android, але я думаю, ви виявите, що це не так важко налагодити це. Якщо ви прочитаєте відповідний біт підручника java, ви побачите, що зареєстрований cookie-обробник отримує зворотні дзвінки з HTTP-коду.

Отже, якщо за замовчуванням немає (ви перевірили, чи CookieHandler.getDefault()дійсно це недійсне?), Ви можете просто розширити CookieHandler, реалізувати put / get і змусити його працювати майже автоматично. Не забудьте врахувати паралельний доступ тощо, якщо ви йдете цим маршрутом.

редагувати: Очевидно, вам доведеться встановити екземпляр власної реалізації як обробник за замовчуванням, CookieHandler.setDefault()щоб отримувати зворотні дзвінки. Забув це згадати.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.