为了账号安全,请及时绑定邮箱和手机立即绑定

Go 编译的二进制文件不会在 Ubuntu 主机上的 alpine docker 容器中运行

Go 编译的二进制文件不会在 Ubuntu 主机上的 alpine docker 容器中运行

Go
守着一只汪 2021-12-27 17:58:25
给定一个二进制文件,使用 GoGOOS=linux和编译GOARCH=amd64,部署到docker基于的容器alpine:3.3,如果 docker 引擎主机是 Ubuntu (15.10),二进制文件将不会运行:sh: /bin/artisan: not found如果 docker 引擎主机(它是 的基础)部署在 Mac OS X 上的 VirtualBox VM 中,则相同的二进制文件(为相同的操作系统和架构编译)将运行得很好。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.10COPY build/ /➜  docker git:(master) ✗ docker build -t artisan .Sending build context to Docker daemon 10.15 MBStep 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.10FROM phusion/baseimageCOPY build/ /➜  docker git:(master) ✗ docker build -t artisan .Sending build context to Docker daemon 10.15 MBStep 1 : FROM phusion/baseimage...➜  docker git:(master) ✗ docker run -it artisan sh# /bin/artisan08:16:39.424 [ERROR] artisan: need a command, one of server, provision or build
查看完整描述

3 回答

?
手掌心

TA贡献1942条经验 获得超3个赞

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


查看完整回答
反对 回复 2021-12-27
?
撒科打诨

TA贡献1934条经验 获得超2个赞

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

RUN apk add --no-cache libc6-compat


查看完整回答
反对 回复 2021-12-27
?
跃然一笑

TA贡献1826条经验 获得超6个赞

来自您的构建机器的 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找到此解决方案的草稿。


查看完整回答
反对 回复 2021-12-27
  • 3 回答
  • 0 关注
  • 390 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信