我正在运行一个使用shelf包的简单Dart服务器。服务器向generativelanguage.googleapis.com
发起请求。在我自己的机器上一切正常。当Dart程序在位于德国的Ubuntu VPS上运行时,API返回{ "error": { "code": 400, "message": "User location is not supported for the API use.", "status": "FAILED_PRECONDITION" } }
。VPS实际上位于德国。Gemini API在德国是支持的。使用工具通过IP查找位置显示为德国。如果我在VPS上使用curl,响应是正常的,我可以获取数据。如果我在Dart中使用Process.run并调用curl,它也能工作。如果我使用dart:http、来自pub.dev的google_generative_ai包、dio、自定义HTTP客户端,它会返回不支持的用户位置。如果我在VPS上运行等效的Python Flask服务器,它也能工作。
我怀疑dart:http包的内部有问题?
我尝试创建自定义Http客户端并设置了所有可能的头信息,我尝试了多个User-Agent,我尝试了有ssl和无ssl的情况,我尝试了cors头信息,我尝试在本地机器上将Dart服务器docker化并在VPS上运行,我尝试分析网络流量,确实是从德国的正确IP发出的,我尝试在不同的端口上运行服务器,我尝试在nginx后面运行服务器,我尝试在WSL上运行服务器并且它能工作,我尝试从Dart应用程序请求我的IP并打印出来,我尝试从Dart应用程序向其他API发起HTTP请求,它们都能正常工作,我已经验证了Dart内部的API密钥是正确的(例如,如果我附加’asd’,我会得到关于API密钥无效的正确错误反馈)。
编辑:我还尝试在globe.dev上托管Dart服务器,它可以无问题地工作。
编辑2:使用curl -4会抛出相同的位置错误,使用curl -6可以正常工作。看起来hetzner的ipv4被阻止了?
编辑3:在Ubuntu上禁用ipv4和/或将ipv6优先级设置为更高值并不会改变dart:http的默认行为。手动强制dart:http使用ipv6并手动信任证书(因为直接使用https到ipv6时会失败)是一个丑陋的解决方法。
编辑4:使用squid代理并在dart的http客户端中添加代理也可以工作,而一切都使用ipv4。
编辑5:这是一个Dart VM问题。Dart首先查找ipv4,然后在延迟后查找ipv6。链接到github问题:https://github.com/dart-lang/sdk/issues/60192
回答:
这是一个Dart VM问题。Dart首先查找ipv4,然后在延迟后查找ipv6。链接到github问题:https://github.com/dart-lang/sdk/issues/60192
一些解决方法是使用自定义http客户端手动查找主机的ipv6地址,或者在http客户端中使用代理。