Golang 交叉编译 OSX 二进制文件挂起,而 VPN 在 net/http.Get() 中

Golang cross-compiled OSX binary hangs while VPN'ed in net/http.Get()

我创建了一个简单的 go 程序,它执行 HTTP GET 请求并打印执行该请求所花费的时间:

package main

import (
    "log"
    "net/http"
    "time"
)

func main() {
    start := time.Now()
    http.Get("https://google.com")
    log.Printf("Elapsed: %v", time.Since(start))
}

在 OSX (Sierra 10.12.1) 上本地构建时,执行请求所用的时间是合理的(<500 毫秒)。

$ for i in `seq 1 10`; do ./httpgettest ; done
2017/08/15 14:20:44 Elapsed: 525.989928ms
2017/08/15 14:20:45 Elapsed: 479.785494ms
2017/08/15 14:20:45 Elapsed: 477.800294ms
2017/08/15 14:20:46 Elapsed: 494.060461ms
2017/08/15 14:20:46 Elapsed: 477.368627ms
2017/08/15 14:20:47 Elapsed: 454.152783ms
2017/08/15 14:20:47 Elapsed: 463.760254ms
2017/08/15 14:20:48 Elapsed: 470.52473ms
2017/08/15 14:20:48 Elapsed: 461.632058ms
2017/08/15 14:20:49 Elapsed: 465.769262ms

然后我从 golang Linux docker 容器交叉编译相同的程序:

GOOS=darwin GOARCH=amd64 go build -v -o httpgettest-xcompiled

然而,当在 VPN 中执行生成的二进制文件时,经过的时间总是超过 10 秒:

$ for i in `seq 1 10`; do ./httpgettest-xcompiled ; done
2017/08/15 14:19:43 Elapsed: 10.532037349s
2017/08/15 14:19:54 Elapsed: 10.525551121s
2017/08/15 14:20:04 Elapsed: 10.572715005s
2017/08/15 14:20:15 Elapsed: 10.532407157s
2017/08/15 14:20:25 Elapsed: 10.54083169s
2017/08/15 14:20:36 Elapsed: 10.625399239s
2017/08/15 14:20:46 Elapsed: 10.539333467s
2017/08/15 14:20:57 Elapsed: 10.533211943s
2017/08/15 14:21:07 Elapsed: 10.539430574s
2017/08/15 14:21:18 Elapsed: 10.527510134s

一旦我断开与 VPN 的连接,数字就会下降。

有人可以解释一下交叉编译的二进制文件在使用 VPN 时速度很慢,但 mac 本机编译的版本始终性能良好吗?这可以使用 golang 1.7.6、1.8.3 和 1.9 docker 容器重现。

我使用 Tunnelblick 连接到 OpenVPN 服务器。

看起来Go 在本地编译时使用系统的本机DNS 解析代码(通过C 库),但在交叉编译时使用Go 自己用Go 编写的解析器。作为确认,在 CGO_ENABLED=0 时听起来很慢。虽然它不是专门针对您在这里遇到的问题,但 https://github.com/golang/go/issues/16345 在快速搜索中出现并提到了 CGO_ENABLED 如何确定哪个解析器运行。

不过,我不确定如何编译要跨平台使用 cgo 本机解析器的二进制文件(可能至少需要安装 C 交叉编译器)或让解析器在 Go 中工作在您的达尔文 VPN 设置下。