问题描述
默认情况下,如果使用net
包,构建可能会生成一个带有一些动态链接的二进制文件,例如到 libc。您可以通过查看结果来动态检查与静态链接ldd output.bin
我遇到过两种解决方案:
- 禁用 CGO,通过
CGO_ENABLED=0
- 强制使用网络依赖的 Go 实现,netgo via
go build -tags netgo -a -v
,这是针对特定平台实现的
来自https://golang.org/doc/go1.2:
net 包默认需要 cgo,因为主机操作系统通常必须调解网络调用设置。但是,在某些系统上,可以在没有 cgo 的情况下使用网络,这样做很有用,例如避免动态链接。新的构建标记 netgo(默认关闭)允许在可能的情况下在纯 Go 中构建 net 包。
以上假设唯一的 CGO 依赖项是标准库的net
包。
我在使用 go 二进制文件时遇到了同样的问题,在将它添加到我的 docker 文件后我让它工作了:
RUN apk add --no-cache libc6-compat
解决方法
给定一个二进制文件,使用 GoGOOS=linux
和编译GOARCH=amd64
,部署到docker
基于的容器alpine:3.3
,如果 docker 引擎主机是 Ubuntu (15.10),二进制文件将不会运行:
sh: /bin/artisan: not found
如果 docker 引擎主机(它是 的基础)部署在 Mac OS X 上的 VirtualBox VM 中,则相同的二进制文件(为相同的操作系统和架构编译)将运行得很好。busybox``alpine
如果容器基于 Ubuntu 映像之一,则相同的二进制文件也将运行得非常好。
知道这个二进制文件缺少什么吗?
这是我为重现所做的工作(未显示在 OS X 上的 VirtualBox/busybox 中成功运行):
构建(即使拱门匹配,也使用标志显式构建):
➜ artisan git:(master) ✗ GOOS=linux GOARCH=amd64 go build
检查它可以在主机上运行:
➜ artisan git:(master) ✗ ./artisan
10:14:04.925 [ERROR] artisan: need a command,one of server,provision or build
复制到 docker 目录,构建,运行:
➜ artisan git:(master) ✗ cp artisan docker/build/bin/
➜ artisan git:(master) ✗ cd docker
➜ docker git:(master) ✗ cat Dockerfile
FROM docker:1.10
COPY build/ /
➜ docker git:(master) ✗ docker build -t artisan .
Sending build context to Docker daemon 10.15 MB
Step 1 : FROM docker:1.10
...
➜ docker git:(master) ✗ docker run -it artisan sh
/ # /bin/artisan
sh: /bin/artisan: not found
现在将图像库更改为phusion/baseimage
:
➜ docker git:(master) ✗ cat Dockerfile
#FROM docker:1.10
FROM phusion/baseimage
COPY build/ /
➜ docker git:(master) ✗ docker build -t artisan .
Sending build context to Docker daemon 10.15 MB
Step 1 : FROM phusion/baseimage
...
➜ docker git:(master) ✗ docker run -it artisan sh
# /bin/artisan
08:16:39.424 [ERROR] artisan: need a command,provision or build