How do I make an http request using cookies on flutter?

后端 未结 4 928
花落未央
花落未央 2020-12-09 03:11

I\'d like to make an http request to a remote server while properly handling cookies (eg. storing cookies sent by the server, and sending those cookies when I make subsequen

相关标签:
4条回答
  • 2020-12-09 03:18

    I've published a small flutter library called requests to assist with cookie-aware http requests.

    dependencies:
      requests: ^3.0.1
    

    Usage:

    import 'package:requests/requests.dart';
    
    // ...
    
    // this will persist cookies
    var r1 = await Requests.post("https://example.com/api/v1/login", json: {"username":"...", "password":"..."} ); 
    r1.raiseForStatus();
    
    // this will re-use the persisted cookies
    var r2 = await Requests.get("https://example.com/api/v1/stuff"); 
    r2.raiseForStatus();
    print(r2.json()['id'])
    

    find out more about requests

    0 讨论(0)
  • 2020-12-09 03:21

    I found best solution to handle cookies for redirect also

    import 'dart:convert';
    import 'dart:io';
    
    class CustomHTTPClient{
      final HttpClient _client = new HttpClient();
      Map<String, String> _cookies = Map();
    
      CustomHTTPClient(){
        _client.connectionTimeout = Duration(seconds: 10);
      }
    
      Future<String> get(String url, {int maxRedirect = 3}) async {
        final parsedUrl = Uri.parse(url);
        return await _client.getUrl(parsedUrl)
            .then((HttpClientRequest request) {
          request.followRedirects = false;
          _beforeRequest(request);
          return request.close();
        }).then((HttpClientResponse response) async {
          _afterResponse(response);
          if(response.isRedirect && maxRedirect > 0){
            return await response.drain().then((value) => get(parsedUrl.resolve(response.headers.value('location')).toString(), maxRedirect: maxRedirect - 1));
          }
          return response.transform(utf8.decoder).join();
        }).catchError((error, stack){
          print(error);print(stack);
        });
      }
    
      void _beforeRequest(HttpClientRequest request){
        request.headers.set(HttpHeaders.acceptEncodingHeader, 'gzip, deflate, br');
    
        // Set cookie
        final String rawCookies = _cookies.keys.map((String name) => '$name=${_cookies[name]}').join('; ');
        if(rawCookies.isNotEmpty) request.headers.set(HttpHeaders.cookieHeader, rawCookies);
      }
    
      void _afterResponse(HttpClientResponse response){
        response.headers.forEach((String name, List<String> values){
          if(name == 'set-cookie'){ // Get cookies for next request
            values.forEach((String rawCookie){
              try{
                Cookie cookie = Cookie.fromSetCookieValue(rawCookie);
                _cookies[cookie.name] = cookie.value;
              } catch(e){
                final List<String> cookie = rawCookie.split(';')[0].split('=');
                _cookies[cookie[0]] = cookie[1];
              }
            });
            return false;
          }
        });
      }
    }
    
    0 讨论(0)
  • 2020-12-09 03:29

    I have improved the Richard Heap's solution to be capable to process multiple 'Set-cookies' and multiple cookies.

    In my case, the server returns multiples 'Set-cookies'. The http package concatenate all the set-cookies headers in one header and separate it by comma (',').

    class NetworkService {
    
      final JsonDecoder _decoder = new JsonDecoder();
      final JsonEncoder _encoder = new JsonEncoder();
    
      Map<String, String> headers = {"content-type": "text/json"};
      Map<String, String> cookies = {};
    
      void _updateCookie(http.Response response) {
        String allSetCookie = response.headers['set-cookie'];
    
        if (allSetCookie != null) {
    
          var setCookies = allSetCookie.split(',');
    
          for (var setCookie in setCookies) {
            var cookies = setCookie.split(';');
    
            for (var cookie in cookies) {
              _setCookie(cookie);
            }
          }
    
          headers['cookie'] = _generateCookieHeader();
        }
      }
    
      void _setCookie(String rawCookie) {
        if (rawCookie.length > 0) {
          var keyValue = rawCookie.split('=');
          if (keyValue.length == 2) {
            var key = keyValue[0].trim();
            var value = keyValue[1];
    
            // ignore keys that aren't cookies
            if (key == 'path' || key == 'expires')
              return;
    
            this.cookies[key] = value;
          }
        }
      }
    
      String _generateCookieHeader() {
        String cookie = "";
    
        for (var key in cookies.keys) {
          if (cookie.length > 0)
            cookie += ";";
          cookie += key + "=" + cookies[key];
        }
    
        return cookie;
      }
    
      Future<dynamic> get(String url) {
        return http.get(url, headers: headers).then((http.Response response) {
          final String res = response.body;
          final int statusCode = response.statusCode;
    
          _updateCookie(response);
    
          if (statusCode < 200 || statusCode > 400 || json == null) {
            throw new Exception("Error while fetching data");
          }
          return _decoder.convert(res);
        });
      }
    
      Future<dynamic> post(String url, {body, encoding}) {
        return http
            .post(url, body: _encoder.convert(body), headers: headers, encoding: encoding)
            .then((http.Response response) {
          final String res = response.body;
          final int statusCode = response.statusCode;
    
          _updateCookie(response);
    
          if (statusCode < 200 || statusCode > 400 || json == null) {
            throw new Exception("Error while fetching data");
          }
          return _decoder.convert(res);
        });
      }
    }
    
    0 讨论(0)
  • 2020-12-09 03:33

    Here's an example of how to grab a session cookie and return it on subsequent requests. You could easily adapt it to return multiple cookies. Make a Session class and route all your GETs and POSTs through it.

    class Session {
      Map<String, String> headers = {};
    
      Future<Map> get(String url) async {
        http.Response response = await http.get(url, headers: headers);
        updateCookie(response);
        return json.decode(response.body);
      }
    
      Future<Map> post(String url, dynamic data) async {
        http.Response response = await http.post(url, body: data, headers: headers);
        updateCookie(response);
        return json.decode(response.body);
      }
    
      void updateCookie(http.Response response) {
        String rawCookie = response.headers['set-cookie'];
        if (rawCookie != null) {
          int index = rawCookie.indexOf(';');
          headers['cookie'] =
              (index == -1) ? rawCookie : rawCookie.substring(0, index);
        }
      }
    }
    
    0 讨论(0)
提交回复
热议问题