95

给定一个二进制文件,使用 Go 编译,GOOS=linuxGOARCH=amd64部署到docker基于 的容器alpine:3.3,如果 docker 引擎主机是 Ubuntu (15.10),则二进制文件将不会运行:

sh: /bin/artisan: not found

如果 docker 引擎主机(它是 的基础)部署在 Mac OS X 上的 VirtualBox VM 中,那么这个相同的二进制文件(为相同的 OS 和 arch 编译)将运行得很好。busyboxalpine

如果容器基于 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, one of server, provision or build 
4

6 回答 6

109

默认情况下,如果使用该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包。

于 2016-03-30T12:09:37.257 回答
92

我在使用 go 二进制文件时遇到了同样的问题,在将它添加到我的 docker 文件后我让它工作了:

RUN apk add --no-cache libc6-compat 
于 2018-06-14T15:57:18.443 回答
9

来自你的构建机器的 Go 编译器可能会将你的二进制文件与位于 Alpine 不同位置的库链接。就我而言,它是使用 /lib64 下的依赖项编译的,但 Alpine 不使用该文件夹。

FROM alpine:edge AS build
RUN apk update
RUN apk upgrade
RUN apk add --update go=1.8.3-r0 gcc=6.3.0-r4 g++=6.3.0-r4
WORKDIR /app
ENV GOPATH /app
ADD src /app/src
RUN go get server # server is name of our application
RUN CGO_ENABLED=1 GOOS=linux go install -a server

FROM alpine:edge
WORKDIR /app
RUN cd /app
COPY --from=build /app/bin/server /app/bin/server
CMD ["bin/server"]

我正在撰写有关此问题的文章。你可以在这里找到这个解决方案的草稿http://kefblog.com/2017-07-04/Golang-ang-docker

于 2017-07-04T19:41:13.243 回答
7

对我来说,诀窍是在链接器选项中启用静态链接:

$ go build -ldflags '-linkmode external -w -extldflags "-static"'

-linkmode选项告诉 Go 使用外部链接器,该-extldflags选项设置传递给链接器的选项,并且该-w标志禁用 DWARF 调试信息以提高二进制大小。

请参阅go tool link静态编译的 Go 程序,即使使用 cgo,也始终使用 musl 获取更多详细信息

于 2019-10-01T12:52:35.857 回答
3

我有一个需要CGO_ENABLED=1.

我在 debian-slim 容器中运行编译后的 go 二进制文件的修复方法是使用RUN GOOS=linux GOARCH=amd64 CGO_ENABLED=1 go build -o goapp

并在 debian slim 中运行以下命令

RUN apt-get update && apt-get install -y musl-dev
RUN ln -s /usr/lib/x86_64-linux-musl/libc.so /lib/libc.musl-x86_64.so.1

让我之后能够运行goapp

提示:ldd goapp表明容器中缺少 libc.musl-x86_64。

于 2021-02-12T08:36:52.547 回答
1

在 debian docker 容器中执行 go 二进制文件时,遇到了这个问题: /bin/bash: line 10: /my/go/binary: No such file or directory

二进制文件是通过使用 alpine 容器中的 docker-in-docker (dind) 使用命令构建的: GOOS=linux GOARCH=amd64 go build

通过在构建二进制文件时使用以下环境来修复它: CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build

于 2020-10-28T15:30:30.137 回答