Come si effettua una richiesta http utilizzando i cookie su Android?


121

Vorrei fare una richiesta http a un server remoto mentre gestisco correttamente i cookie (ad es. Memorizzando i cookie inviati dal server e inviando quei cookie quando effettuo richieste successive). Sarebbe bello conservare tutti i cookie, ma in realtà l'unico che mi interessa è il cookie di sessione.

Con java.net, sembra che il modo migliore per farlo sia usare java.net.CookieHandler (classe base astratta) e java.net.CookieManager (implementazione concreta). Android ha java.net.CookieHandler, ma non sembra avere java.net.CookieManager.

Potrei codificare tutto a mano ispezionando le intestazioni http, ma sembra che ci debba essere un modo più semplice.

Qual è il modo corretto per effettuare richieste http su Android preservando i cookie?



3
Proprio come una nota più di due anni dopo: java.net.CookieManagerè ora supportato in Android dalla versione 2.3 (livello API 9): developer.android.com/reference/java/net/CookieManager.html
Slauma

Risposte:


92

Si scopre che Google Android viene fornito con Apache HttpClient 4.0 e sono stato in grado di capire come farlo utilizzando l'esempio "Accesso basato su modulo" nella documentazione di 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
posso sapere come impostare i cookie sull'URL di richiesta per verificare se la sessione è valida o meno?
Praveen

GRAZIE per avermi presentato BasicNameValuePairs. Mi hanno aiutato.
bhekman

3
Capisco The method getCookieStore() is undefined for the type HttpClient, devo cambiare in List<Cookie> cookies = ((AbstractHttpClient) httpclient).getCookieStore().getCookies();? Perché se lo faccio, funziona.
Francisco Corrales Morales

@Praveen Dai un'occhiata qui per salvare i cookie: stackoverflow.com/a/5989115/2615737
Francisco Corrales Morales

@emmby: sfortunato modulo apache deprecato per Android. Quindi in questo momento la sua risposta non è utile. C'è un altro modo per farlo con HttpURLConnection?
Milad Faridnia

9

Un cookie è solo un'altra intestazione HTTP. Puoi sempre impostarlo mentre esegui una chiamata HTTP con la libreria apache o con HTTPUrlConnection. In ogni caso dovresti essere in grado di leggere e impostare i cookie HTTP in questo modo.

Puoi leggere questo articolo per maggiori informazioni.

Posso condividere la mia pace di codice per dimostrare quanto è facile farlo.

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;
    }

Questo mi aiuta molto, +1 get.setHeader ("Cookie", "PHPSESSID =" + token + ";"); fare il lavoro
Oussaki

@Hesam: sfortunatamente modulo apache deprecato per Android. Quindi in questo momento la sua risposta non è utile. C'è un altro modo per farlo con HttpURLConnection?
Milad Faridnia

7

Poiché la libreria Apache è deprecata , per coloro che vogliono utilizzare HttpURLConncetion, ho scritto questa classe per inviare Get e Post Request con l'aiuto di questa risposta :

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
Nice Class Milad! Mi ha aiutato molto! Ora, mi chiedevo, se volevo salvare il cookie affinché fosse disponibile se l'utente ha ucciso l'app e l'ha lanciata di nuovo, cosa dovrei esattamente memorizzare e dove (come)
Ki Jéy

1

Non lavoro con Google Android ma penso che scoprirai che non è così difficile farlo funzionare. Se leggi la parte rilevante del tutorial java , vedrai che un cookiehandler registrato riceve callback dal codice HTTP.

Quindi se non ci sono impostazioni predefinite (hai controllato se CookieHandler.getDefault() è davvero nullo?) Allora puoi semplicemente estendere CookieHandler, implementare put / get e farlo funzionare praticamente automaticamente. Assicurati di considerare l'accesso simultaneo e simili se segui quella strada.

modifica: Ovviamente dovresti impostare un'istanza della tua implementazione personalizzata come gestore predefinito attraverso CookieHandler.setDefault()per ricevere i callback. Ho dimenticato di dirlo.

Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.