将 go 项目打包成docker镜像的正确姿势

/ docker / 没有评论 / 4201浏览

第一种方式

第一步:设置 dockerfile:

FROM scratch
ADD /app //                        // app是 go build 生成的可执行文件
ADD /config.json //              // config.json 是根目录下的配置文件
ENTRYPOINT ["/app"]

第二步: 进入go 项目所在的根目录

第三步:设置环境变量

在控制台执行以下命令,否则无法生成二进制可执行文件

$ SET CGO_ENABLED=0
$ SET GOOS=linux
$ SET GOARCH=amd64
$ go env              // 查看环境变量是否设置成功,命令行在powershell模式下可能会失败,需要切换成cmd模式

第四步:编译

$ go build

第五步:打包成镜像

$ docker build . -t my-app
$ docker push my-app // 推送到远程镜像服务器

第六步:运行

$ docker run -p 8080:8080 my-app

第二种方式

golang编译的应用是不需要依赖其他运行环境的,那么为什么还需要打包成docker镜像呢?当需要附带配置和日志等文件时可以更方便的移植和运行,下面介绍从dockerfile编译成镜像。

在项目根目录新建dockerfile,内容如下:

FROM alpine:latest      #基于的镜像不是golang(733M),而是alpine(4.14M)
MAINTAINER fanyang "luckyfanyang@gmail.com"
WORKDIR $GOPATH/src/login
ADD config        $GOPATH/src/config     #此目录存放的是所有项目的数据库配置
ADD login/conf     $GOPATH/src/login/conf   #此目录存放的是beego项目的配置ADD login/login    $GOPATH/src/login/login  #编译后的可执行文件
RUN mkdir /lib64 && ln -s /lib/libc.musl-x86_64.so.1 /lib64/ld-linux-x86-64.so.2
EXPOSE 8080
ENTRYPOINT ["./login"]

ubuntu下打开终端定位到src目录(因为需要上传config目录,不能直接定位到login目录)执行命令:

docker build -t login -f ./login/dockerfile . 

打包成镜像以后运行命令:

docker run -d -p 8080:8080 login

有时候想共享数据目录使用下列命令:

docker run -itd --name voltest -v /opt/mytest alpine
docker run -it --name helloworld --volumes-from=voltest alpine

这样在voltest下mytest目录里新建的文件会出现在helloworld下的mytest目录,进入目录查询文件使用:

docker exec -it voltest sh

第三种

问题

我写好了一个基于gin的web程序,另外还有一些依赖。我费了千辛万苦才把依赖下载下来,我不想在docker镜像里面在下载一次依赖,如何能够快速的给我构建一个镜像。

有问题的解决方法

golang作为编译型语言,运行的是编译完成的二进制可执行文件,该文件是与运行系统有关的。我的开发系统是ubuntu1604。所以我的解决思路是如下

这个没问题,但是存在一个问题,该镜像太大了,300M。 我本身的程序打包完了才14M。能不能让我的镜像小一点?

于是我把FROM ubuntu:16.04 改成FROM amd64/alpine结果发现报了执行错误

`panic: standard_init_linux.go:175: exec user process caused "no such file or directory``

大概的原因就是我编译出问题,我可不想慢慢去找可是的apline的操作系统镜像。

完善的解决方法

利用golang镜像

原理是将你的环境通过挂载到golang镜像中去,例如我使用的golang版本为1.9

cd ~/go/src/your_app_path
docker run --rm -it -v "$GOPATH":/gopath -v "$(pwd)":/app -e "GOPATH=/gopath" -w /app golang:1.9 sh -c 'CGO_ENABLED=0 go build -a --installsuffix cgo --ldflags="-s" -o your_app_bin_name'

这个时候你的~/go/src/your_app_path路径就有一个your_app_bin_name

基于iron/base镜像

FROM iron/base
WORKDIR /app
COPY your_app_bin_name/app/
ENTRYPOINT ["./your_app_bin_name"]

这样我打包出来的镜像才5M加上我二进制文件的大小,只有19M

优缺点