阅读(124) (0)

Flutter实战 通过HttpClient发起HTTP请求

2021-03-09 09:25:13 更新

Dart IO 库中提供了用于发起 Http 请求的一些类,我们可以直接使用HttpClient来发起请求。使用HttpClient发起请求分为五步:

  1. 创建一个HttpClient

    HttpClient httpClient = new HttpClient();

  1. 打开 Http 连接,设置请求头:

   HttpClientRequest request = await httpClient.getUrl(uri);

这一步可以使用任意 Http Method,如httpClient.post(...)httpClient.delete(...)等。如果包含 Query 参数,可以在构建 uri 时添加,如:

   Uri uri=Uri(scheme: "https", host: "flutterchina.club", queryParameters: {
       "xx":"xx",
       "yy":"dd"
     });

通过HttpClientRequest可以设置请求 header,如:

   request.headers.add("user-agent", "test");

如果是 post 或 put 等可以携带请求体方法,可以通过 HttpClientRequest 对象发送 request body,如:

   String payload="...";
   request.add(utf8.encode(payload)); 
   //request.addStream(_inputStream); //可以直接添加输入流

  1. 等待连接服务器:

   HttpClientResponse response = await request.close();

这一步完成后,请求信息就已经发送给服务器了,返回一个HttpClientResponse对象,它包含响应头(header)和响应流(响应体的 Stream),接下来就可以通过读取响应流来获取响应内容。

  1. 读取响应内容:

   String responseBody = await response.transform(utf8.decoder).join();

我们通过读取响应流来获取服务器返回的数据,在读取时我们可以设置编码格式,这里是 utf8。

  1. 请求结束,关闭HttpClient

   httpClient.close();

关闭 client 后,通过该 client 发起的所有请求都会中止。

#示例

我们实现一个获取百度首页 html 的例子,示例效果如图11-1所示:

图11-1

点击“获取百度首页”按钮后,会请求百度首页,请求成功后,我们将返回内容显示出来并在控制台打印响应 header,代码如下:

import 'dart:convert';
import 'dart:io';


import 'package:flutter/material.dart';


class HttpTestRoute extends StatefulWidget {
  @override
  _HttpTestRouteState createState() => new _HttpTestRouteState();
}


class _HttpTestRouteState extends State<HttpTestRoute> {
  bool _loading = false;
  String _text = "";


  @override
  Widget build(BuildContext context) {
    return ConstrainedBox(
      constraints: BoxConstraints.expand(),
      child: SingleChildScrollView(
        child: Column(
          children: <Widget>[
            RaisedButton(
                child: Text("获取百度首页"),
                onPressed: _loading ? null : () async {
                  setState(() {
                    _loading = true;
                    _text = "正在请求...";
                  });
                  try {
                    //创建一个HttpClient
                    HttpClient httpClient = new HttpClient();
                    //打开Http连接
                    HttpClientRequest request = await httpClient.getUrl(
                        Uri.parse("https://www.baidu.com"));
                    //使用iPhone的UA
                    request.headers.add("user-agent", "Mozilla/5.0 (iPhone; CPU iPhone OS 10_3_1 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/10.0 Mobile/14E304 Safari/602.1");
                    //等待连接服务器(会将请求信息发送给服务器)
                    HttpClientResponse response = await request.close();
                    //读取响应内容
                    _text = await response.transform(utf8.decoder).join();
                    //输出响应头
                    print(response.headers);


                    //关闭client后,通过该client发起的所有请求都会中止。
                    httpClient.close();


                  } catch (e) {
                    _text = "请求失败:$e";
                  } finally {
                    setState(() {
                      _loading = false;
                    });
                  }
                }
            ),
            Container(
                width: MediaQuery.of(context).size.width-50.0,
                child: Text(_text.replaceAll(new RegExp(r"\s"), ""))
            )
          ],
        ),
      ),
    );
  }
}

控制台输出:

I/flutter (18545): connection: Keep-Alive
I/flutter (18545): cache-control: no-cache
I/flutter (18545): set-cookie: ....  //有多个,省略...
I/flutter (18545): transfer-encoding: chunked
I/flutter (18545): date: Tue, 30 Oct 2018 10:00:52 GMT
I/flutter (18545): content-encoding: gzip
I/flutter (18545): vary: Accept-Encoding
I/flutter (18545): strict-transport-security: max-age=172800
I/flutter (18545): content-type: text/html;charset=utf-8
I/flutter (18545): tracecode: 00525262401065761290103018, 00522983

#HttpClient配置

HttpClient有很多属性可以配置,常用的属性列表如下:

属性 含义
idleTimeout 对应请求头中的 keep-alive 字段值,为了避免频繁建立连接,httpClient 在请求结束后会保持连接一段时间,超过这个阈值后才会关闭连接。
connectionTimeout 和服务器建立连接的超时,如果超过这个值则会抛出 SocketException 异常。
maxConnectionsPerHost 同一个 host,同时允许建立连接的最大数量。
autoUncompress 对应请求头中的 Content-Encoding,如果设置为 true,则请求头中Content-Encoding 的值为当前 HttpClient 支持的压缩算法列表,目前只有"gzip"
userAgent 对应请求头中的 User-Agent 字段。

可以发现,有些属性只是为了更方便的设置请求头,对于这些属性,你完全可以通过HttpClientRequest直接设置 header,不同的是通过HttpClient设置的对整个httpClient都生效,而通过HttpClientRequest设置的只对当前请求生效。

#HTTP请求认证

Htt 协议的认证(Authentication)机制可以用于保护非公开资源。如果 Http 服务器开启了认证,那么用户在发起请求时就需要携带用户凭据,如果你在浏览器中访问了启用 Basic 认证的资源时,浏览就会弹出一个登录框,如:

image-20181031114207514

我们先看看 Basic 认证的基本过程:

  1. 客户端发送 http 请求给服务器,服务器验证该用户是否已经登录验证过了,如果没有的话, 服务器会返回一个 401 Unauthozied 给客户端,并且在响应 header 中添加一个 “WWW-Authenticate” 字段,例如:

   WWW-Authenticate: Basic realm="admin"

其中"Basic"为认证方式,realm 为用户角色的分组,可以在后台添加分组。

  1. 客户端得到响应码后,将用户名和密码进行 base64 编码(格式为用户名:密码),设置请求头 Authorization,继续访问 :

   Authorization: Basic YXXFISDJFISJFGIJIJG

服务器验证用户凭据,如果通过就返回资源内容。

注意,Http 的方式除了 Basic 认证之外还有:Digest 认证、Client 认证、Form Based 认证等,目前 Flutter 的 HttpClient 只支持 Basic 和 Digest 两种认证方式,这两种认证方式最大的区别是发送用户凭据时,对于用户凭据的内容,前者只是简单的通过 Base64 编码(可逆),而后者会进行哈希运算,相对来说安全一点点,但是为了安全起见,无论是采用 Basic 认证还是 Digest 认证,都应该在 Https 协议下,这样可以防止抓包和中间人攻击。

HttpClient关于 Http 认证的方法和属性:

  1. addCredentials(Uri url, String realm, HttpClientCredentials credentials)

该方法用于添加用户凭据,如:

   httpClient.addCredentials(_uri,
    "admin", 
     new HttpClientBasicCredentials("username","password"), //Basic认证凭据
   );

如果是 Digest 认证,可以创建 Digest 认证凭据:

   HttpClientDigestCredentials("username","password")

  1. authenticate(Future<bool> f(Uri url, String scheme, String realm))

这是一个 setter,类型是一个回调,当服务器需要用户凭据且该用户凭据未被添加时,httpClient会调用此回调,在这个回调当中,一般会调用addCredential()来动态添加用户凭证,例如:

   httpClient.authenticate=(Uri url, String scheme, String realm) async{
     if(url.host=="xx.com" && realm=="admin"){
       httpClient.addCredentials(url,
         "admin",
         new HttpClientBasicCredentials("username","pwd"), 
       );
       return true;
     }
     return false;
   };

一个建议是,如果所有请求都需要认证,那么应该在 HttpClient 初始化时就调用addCredentials()来添加全局凭证,而不是去动态添加。

#代理

可以通过findProxy来设置代理策略,例如,我们要将所有请求通过代理服务器(192.168.1.2:8888)发送出去:

  client.findProxy = (uri) {
    // 如果需要过滤uri,可以手动判断
    return "PROXY 192.168.1.2:8888";
 };

findProxy 回调返回值是一个遵循浏览器 PAC 脚本格式的字符串,详情可以查看 API 文档,如果不需要代理,返回"DIRECT"即可。

在 APP 开发中,很多时候我们需要抓包来调试,而抓包软件(如 charles)就是一个代理,这时我们就可以将请求发送到我们的抓包软件,我们就可以在抓包软件中看到请求的数据了。

有时代理服务器也启用了身份验证,这和 http 协议的认证是相似的,HttpClient 提供了对应的 Proxy 认证方法和属性:

set authenticateProxy(
    Future<bool> f(String host, int port, String scheme, String realm));
void addProxyCredentials(
    String host, int port, String realm, HttpClientCredentials credentials);

他们的使用方法和上面“HTTP请求认证”一节中介绍的addCredentialsauthenticate 相同,故不再赘述。

#证书校验

Https 中为了防止通过伪造证书而发起的中间人攻击,客户端应该对自签名或非 CA 颁发的证书进行校验。HttpClient对证书校验的逻辑如下:

  1. 如果请求的 Https 证书是可信 CA 颁发的,并且访问 host 包含在证书的 domain 列表中(或者符合通配规则)并且证书未过期,则验证通过。
  2. 如果第一步验证失败,但在创建 HttpClient 时,已经通过 SecurityContext 将证书添加到证书信任链中,那么当服务器返回的证书在信任链中的话,则验证通过。
  3. 如果1、2验证都失败了,如果用户提供了badCertificateCallback回调,则会调用它,如果回调返回true,则允许继续链接,如果返回false,则终止链接。

综上所述,我们的证书校验其实就是提供一个badCertificateCallback回调,下面通过一个示例来说明。

#示例

假设我们的后台服务使用的是自签名证书,证书格式是 PEM 格式,我们将证书的内容保存在本地字符串中,那么我们的校验逻辑如下:

String PEM="XXXXX";//可以从文件读取
...
httpClient.badCertificateCallback=(X509Certificate cert, String host, int port){
  if(cert.pem==PEM){
    return true; //证书一致,则允许发送数据
  }
  return false;
};

X509Certificate是证书的标准格式,包含了证书除私钥外所有信息,读者可以自行查阅文档。另外,上面的示例没有校验 host,是因为只要服务器返回的证书内容和本地的保存一致就已经能证明是我们的服务器了(而不是中间人),host 验证通常是为了防止证书和域名不匹配。

对于自签名的证书,我们也可以将其添加到本地证书信任链中,这样证书验证时就会自动通过,而不会再走到badCertificateCallback回调中:

SecurityContext sc=new SecurityContext();
//file为证书路径
sc.setTrustedCertificates(file);
//创建一个HttpClient
HttpClient httpClient = new HttpClient(context: sc);

注意,通过setTrustedCertificates()设置的证书格式必须为 PEM 或 PKCS12,如果证书格式为 PKCS12,则需将证书密码传入,这样则会在代码中暴露证书密码,所以客户端证书校验不建议使用 PKCS12 格式的证书。

#总结

值得注意的是,HttpClient提供的这些属性和方法最终都会作用在请求 header 里,我们完全可以通过手动去设置 header 来实现,之所以提供这些方法,只是为了方便开发者而已。另外,Http 协议是一个非常重要的、使用最多的网络协议,每一个开发者都应该对 http 协议非常熟悉。