翻阅HttpClient源码解决Flutter Archive后无法请求代理接口,我的GPT客户端终于自由啦~

前言

时间一转眼我的上一篇博文更新已经接近3个月有余。我这段时间都在干嘛?除了4月因为裁员的事情心有余悸,这两三个月我都在捣鼓Mac的GPT客户端。转眼我的Mac客户端Ai Chat - 你问我答也上线了2个多月了。这几天也在准备把之前一次性返回接口,改回流式返回。其实1.0上线的时候我也就遇到了这个问题,当我的Flutter Mac桌面包Archive后,使用官方HttpClient的接口直接请求OpenAi的接口,因为我们有墙,我必须开启VPN才能请求,但是就是每次Archive后我都无法再请求成功。当时为了快速上线,我选择了直接用原生Api也就是NSURlSession的Api然后通过PlatforChannel返回。但是最近想升级优化,不再通过此方式,所以也就捣鼓了下为什么HttpClient Archive后无法请求代理接口,所以就有了以下的研究。

1685591133.gif

研究过程

  • 查阅HttpClient源码

我发现HttpClient有一个特别的方法引起了我的关注

/// Sets the function used to resolve the proxy server to be used for
  /// opening a HTTP connection to the specified [url]. If this
  /// function is not set, direct connections will always be used.
  ///
  /// The string returned by [f] must be in the format used by browser
  /// PAC (proxy auto-config) scripts. That is either
  ///
  ///     "DIRECT"
  ///
  /// for using a direct connection or
  ///
  ///     "PROXY host:port"
  ///
  /// for using the proxy server `host` on port `port`.
  ///
  /// A configuration can contain several configuration elements
  /// separated by semicolons, e.g.
  ///
  ///     "PROXY host:port; PROXY host2:port2; DIRECT"
  ///
  /// The static function [findProxyFromEnvironment] on this class can
  /// be used to implement proxy server resolving based on environment
  /// variables.
  void set findProxy(String Function(Uri url)? f);

这个方法的注释写得很明白了,只要实现这个方法,返回特别的代理,它就能强制某个Uri用代理接口,当时我一下就又了80%的信心能解决次问题了。

  • 尝试使用默认代理配置

紧接着我又发现了有一个默认的配置


 HttpClient client = HttpClient();
  ///     client.findProxy = HttpClient.findProxyFromEnvironment;

后面我尝试在debug以及Archive下分别打印了HttpClient.findProxyFromEnvironment。这个闭包函数。惊奇的发现,debug下,这个函数是能反正我当前的代理信息的(也就是全局翻墙配置)类似127.0.0.1:8888 这样的信息。但是Archive后的包是无法获取到的。

  • 结论

这里就破案了,当Flutter Mac的包Archive后(而且开启了Sandbox,只有开启了App才能上架到AppStore),应用会有个安全措施,阻止你获取系统环境变量等敏感信息,导致自然也无法获取到代理信息,所以如果你的接口需要代理(也就是俗称的翻墙或提子)才能请求(如openai的接口),那么就会请求失败。

修改方案

既然由于无法获取到代理信息呆滞请求失败,那么我们只需要请求接口前把代理信息给获取到那么理论上就能成功请求了。那么我们如何能获取到代理信息呢?答案就是通过原生获取,由于原生苹果官方Api的强大,我们苹果官方原生是有接口能获取到代理信息的,这也间接解释了,为什么我1.0版本通过NSURlSession苹果官方的Http接口就没有这个问题。然后我们通过插件的方式,返回到Flutter端即可。

原生Swift接口如下


func getProxySetting() -> NSDictionary? {
        guard let proxySettings = CFNetworkCopySystemProxySettings()?.takeUnretainedValue(),
            let url = URL(string: "https://api.openai.com/") else {
                return nil
        }
        let proxies = CFNetworkCopyProxiesForURL((url as CFURL), proxySettings).takeUnretainedValue() as NSArray
        guard let settings = proxies.firstObject as? NSDictionary,
            let _ = settings.object(forKey: (kCFProxyTypeKey as String)) as? String else {
                return nil
        }

        if let hostName = settings.object(forKey: (kCFProxyHostNameKey as String)), let port = settings.object(forKey: (kCFProxyPortNumberKey as String)) {
            return ["host":hostName, "port":port]
        }
        return nil;
    }

由于时间的关系,我并没有直接自己编写插件而是自己使用了native_flutter_proxy这个插件。但是这个插件是不支持macos的,但是问题不大我们只需要把源码下载下来,新增一个macos下的插件,他们的原生代码和iOS是一样的。

Flutter端代码如下,我们只需要在请求之前,获取到代理信息,返回到findProxy方法即可。

 bool enabled = false;
    String? host;
    int? port;
    try {
      ProxySetting settings = await NativeProxyReader.proxySetting;
      enabled = settings.enabled;
      host = settings.host;
      port = settings.port;
    } catch (e) {
      print(e);
    }

    String steamData = "";
    HttpClient httpClient = HttpClient();

    if (enabled && host != null) {
      httpClient.findProxy = (uri) {
        return "PROXY $host:$port";
      };
      print("proxy enabled $host:$port");
    } else {
      print("not have proxy");
    }

AppStore体验地址

附上目前效果,由于我的apiKey是支持GPT-4的,所以也可选择GPT-4返回

1685591133.gif

iOS版或m1系列mac 可以兼容安装。iOS版是直接使用私有云的,不订阅的话每天只能聊500token以下,望理解。另流式返回的还在审核中,iOS目前还是一次性返回。

Ai Chat Plus - 你问我答

Mac版

Ai Chat - 你问我答

目前我的Mac应用只要使用自己的apiKey是能直接使用的(前提就是要你自己能翻墙以及有api key),或者也能订阅我的云服务器(那就不需要翻墙),但是要一定费用,不需要就不订阅就好了哈,[Sick]勿喷,毕竟辛苦肝了几个月了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容