(轉!)Flutter中http請求抓包解決方案

普遍的做法是,選擇一種方法,試試看;如果失敗了,沒關係,再試試別的方法。不管怎麼樣,重要的是先去嘗試。

                      ---富蘭克林·羅斯福

前言

前陣子有同學反饋Flutter中的http請求無法通過fiddler抓包,作者喜歡使用Charles抓包工具,於是抽時間寫了個小demo測試了一下,結論是在手機上設置代理,Charles確實抓不到請求數據包。於是對該問題進行了分析:

  1. 確定使用的是http發起的get請求,理論上http協議應該可以被Charles抓到包的,如果沒有抓到包,那可能是沒有走代理,於是乎通過將筆記本連接的wifi斷開測試了一下手機上APP發起http請求,發現請求成功,證實確實沒有走代理;

  2. 爲什麼http請求沒有通過wifi走代理呢,因爲之前安卓原生使用的一些http框架都是正常走代理的啊,那是不是有可能代碼中有api方法可以設置請求不走代理,於是乎就研讀了一下Flutter中http相關的源碼,最終找到了答案。

http請求源碼跟蹤

http.dart中的HttpClient是一個抽象類,成員方法的具體實現在http_impl.dart中,http的get請求實現如下:

Future<HttpClientRequest> getUrl(Uri url) => _openUrl("get", url);

Future<_HttpClientRequest> _openUrl(String method, Uri uri) {
  .
  .
  .
  // Check to see if a proxy server should be used for this connection.
  var proxyConf = const _ProxyConfiguration.direct();
  if (_findProxy != null) {
    // TODO(sgjesse): Keep a map of these as normally only a few
    // configuration strings will be used.
    try {
      proxyConf = new _ProxyConfiguration(_findProxy(uri));
    } catch (error, stackTrace) {
      return new Future.error(error, stackTrace);
    }
  }
  return _getConnection(uri.host, port, proxyConf, isSecure)
      .then((_ConnectionInfo info) {
    .
    .
    .
  });
}

首先,我們可以發現方法中有一行註釋// Check to see if a proxy server should be used for this connection.,意思是“檢查是否應該使用代理服務器進行此連接”;

然後,有一個proxyConf對象初始化和根據_findProxy來創建新的proxyConf對象的語句,然後通過_getConnection(uri.host, port, proxyConf, isSecure)來創建連接,_getConnection的源碼如下:

Future<_ConnectionInfo> _getConnection(String uriHost, int uriPort,
    _ProxyConfiguration proxyConf, bool isSecure) {
  Iterator<_Proxy> proxies = proxyConf.proxies.iterator;

  Future<_ConnectionInfo> connect(error) {
    if (!proxies.moveNext()) return new Future.error(error);
    _Proxy proxy = proxies.current;
    String host = proxy.isDirect ? uriHost : proxy.host;
    int port = proxy.isDirect ? uriPort : proxy.port;
    return _getConnectionTarget(host, port, isSecure)
        .connect(uriHost, uriPort, proxy, this)
        // On error, continue with next proxy.
        .catchError(connect);
  }

  return connect(new HttpException("No proxies given"));
}

 

從代碼中我們可以看到根據代理配置信息來將請求的host和port進行重置,然後創建真實的連接。

跟蹤以上源碼我們發現dart中http請求是否走代理是需要配置的,而_findProxy變量和配置的代理信息有關。

http__impl.dart文件中的_HttpClient類中定義了_findProxy的默認值

Function _findProxy = HttpClient.findProxyFromEnvironment;

HttpClient類中findProxyFromEnvironment方法的實現

static String findProxyFromEnvironment(Uri url,
    {Map<String, String> environment}) {
  HttpOverrides overrides = HttpOverrides.current;
  if (overrides == null) {
    return _HttpClient._findProxyFromEnvironment(url, environment);
  }
  return overrides.findProxyFromEnvironment(url, environment);
}

_HttpClient類中_findProxyFromEnvironment方法的實現

static String _findProxyFromEnvironment(
    Uri url, Map<String, String> environment) {
  checkNoProxy(String option) {
    if (option == null) return null;
    Iterator<String> names = option.split(",").map((s) => s.trim()).iterator;
    while (names.moveNext()) {
      var name = names.current;
      if ((name.startsWith("[") &&
              name.endsWith("]") &&
              "[${url.host}]" == name) ||
          (name.isNotEmpty && url.host.endsWith(name))) {
        return "DIRECT";
      }
    }
    return null;
  }

  checkProxy(String option) {
    if (option == null) return null;
    option = option.trim();
    if (option.isEmpty) return null;
    int pos = option.indexOf("://");
    if (pos >= 0) {
      option = option.substring(pos + 3);
    }
    pos = option.indexOf("/");
    if (pos >= 0) {
      option = option.substring(0, pos);
    }
    // Add default port if no port configured.
    if (option.indexOf("[") == 0) {
      var pos = option.lastIndexOf(":");
      if (option.indexOf("]") > pos) option = "$option:1080";
    } else {
      if (option.indexOf(":") == -1) option = "$option:1080";
    }
    return "PROXY $option";
  }

  // Default to using the process current environment.
  if (environment == null) environment = _platformEnvironmentCache;

  String proxyCfg;

  String noProxy = environment["no_proxy"];
  if (noProxy == null) noProxy = environment["NO_PROXY"];
  if ((proxyCfg = checkNoProxy(noProxy)) != null) {
    return proxyCfg;
  }

  if (url.scheme == "http") {
    String proxy = environment["http_proxy"];
    if (proxy == null) proxy = environment["HTTP_PROXY"];
    if ((proxyCfg = checkProxy(proxy)) != null) {
      return proxyCfg;
    }
  } else if (url.scheme == "https") {
    String proxy = environment["https_proxy"];
    if (proxy == null) proxy = environment["HTTPS_PROXY"];
    if ((proxyCfg = checkProxy(proxy)) != null) {
      return proxyCfg;
    }
  }
  return "DIRECT";
}

從以上代碼中可以發現代理配置從environment中讀取,設置代理時必須指定http_proxyhttps_proxy等。而從_openUrl方法實現中proxyConf = new _ProxyConfiguration(_findProxy(uri));得出默認情況下environment是爲空的,所以要想在Flutter的http請求中使用代理,則要指定相應的代理配置,即設置httpClient.findProxy的值。

示例代碼:

_getHttpData() async {
  var httpClient = new HttpClient();
  httpClient.findProxy = (url) {
    return HttpClient.findProxyFromEnvironment(url, environment: {"http_proxy": 'http://192.168.124.7:8888',});
  };
  var uri =
      new Uri.http('t.weather.sojson.com', '/api/weather/city/101210101');
  var request = await httpClient.getUrl(uri);
  var response = await request.close();
  if (response.statusCode == 200) {
    print('請求成功');
    var responseBody = await response.transform(Utf8Decoder()).join();
    print('responseBody = $responseBody');
  } else {
    print('請求失敗');
  }
}

以上代碼設置後即可使用Fiddler或Charles抓包了。

注:

  • 代碼中已設置代理,手機wifi不再需要進行代理設置;

  • 192.168.124.7該IP爲我們需要抓包的Charles所在電腦IP;

第二種抓包解決方案

如果使用Flutter寫的APP不手動設置代理,則可以使用另一種方案來抓包。

通過電腦設置熱點 -> 使用手機連接電腦熱點上網 -> 在電腦上使用Wireshark抓數據包。

具體步驟如下(macOS系統下):

1. 打開系統偏好設置,找到“共享”

 

 

2. 打開“共享”,顯示以下窗口,並選擇共享以下來源的連接爲指定的有線網絡,用以下端口共享給電腦選擇爲Wi-Fi

 

 

3. 點擊右下角Wi-Fi選項按鈕,顯示如下,填寫對應信息後點擊“好”保存

 

 

4. 回到剛纔的“共享”窗口,打開左側窗口中的服務“互聯網共享”

 

 

5. 然後打開Wireshark軟件界面,首頁選擇對應開熱點的網絡雙擊

 

 

6. 請求接口域名t.weather.sojson.com對應的IP爲 58.222.18.24,則在上面輸入框中輸入請求過濾條件 "ip.dst == 58.222.18.24",然後通過手機APP發起網絡請求

查看接口的IP地址

$ ping t.weather.sojson.com
PING nm.ctn.aicdn.com (58.222.18.24): 56 data bytes
64 bytes from 58.222.18.24: icmp_seq=0 ttl=54 time=16.792 ms
64 bytes from 58.222.18.24: icmp_seq=1 ttl=54 time=16.926 ms
64 bytes from 58.222.18.24: icmp_seq=2 ttl=54 time=15.804 ms
複製代碼

 

 

7. 選擇對應的http請求,箭頭指定行,右鍵點擊,選擇Follow->HTTP Stream選項

 

 

8. 彈出具體網絡請求信息窗口如下

 

 

寫在最後

本篇分享了兩種Flutter中http數據包的抓包解決方案,大家可以根據實際情況來選擇使用。如果有其他Flutter相關的問題,歡迎大家通過公衆號發消息留言。

轉自:https://juejin.im/post/5c85f504e51d4510a06d3f0a

嗯。。可以測試其解決方案,我個人對flutter抓包還是使用as直接去抓包,方便。

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章