打开APP
userphoto
未登录

开通VIP,畅享免费电子书等14项超值服

开通VIP
为 Go Web

你或许想在 Docker 中使用 Go,原因有:

  1. 如果你想在 Kubernetes 上运行,打包为镜像是必须的(就像我一样)
  2. 你不得不在同一台机器上运行不同的 Go 版本
  3. 开发和生产都需要精确的、可复制的、可共享的和确定的环境
  4. 你需要快速和简单的方式来构建和部署编译好的二进制文件
  5. 你想快速开始(任何安装了 Docker 的人都可以直接开始编写代码而不需要设置其他依赖或 GOPATH 环境变量)

恭喜你,你来对地方了。

我们将逐步构建一个基本的 Dockerfile,包括实时重载包管理,然后进行扩展,创建一个高度优化的生产版的镜像,其大小缩减了 100 倍。如果你使用 CI/CD 系统,镜像大小可能无关紧要,但是当 docker pushdocker pull 时,一个精简的镜像肯定会有帮助。

如果你只想要最终的代码,请看 GitHub

FROM golang:1.8.5-jessie as builder# install xzRUN apt-get update && apt-get install -y     xz-utils && rm -rf /var/lib/apt/lists/*# install UPXADD https://github.com/upx/upx/releases/download/v3.94/upx-3.94-amd64_linux.tar.xz /usr/localRUN xz -d -c /usr/local/upx-3.94-amd64_linux.tar.xz |     tar -xOf - upx-3.94-amd64_linux/upx > /bin/upx &&     chmod a+x /bin/upx# install glideRUN go get github.com/Masterminds/glide# setup the working directoryWORKDIR /go/src/appADD glide.yaml glide.yamlADD glide.lock glide.lock# install dependenciesRUN glide install# add source codeADD src src# build the sourceRUN go build src/main.goRUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o main src/main.go# strip and compress the binaryRUN strip --strip-unneeded mainRUN upx main# use a minimal alpine imageFROM alpine:3.7# add ca-certificates in case you need themRUN apk add --no-cache ca-certificates# set working directoryWORKDIR /root# copy the binary from builderCOPY --from=builder /go/src/app/main .# run the binaryCMD ["./main"]

假设我们的应用叫 go-docker,下面是项目的结构。所有的源码都在 src 目录下,Dockerfile 跟它在同一级目录。 main.go 定义了一个 web-app 并监听 8080 端口。

go-docker├── Dockerfile└── src    └── main.go

最简单的版本

FROM golang:1.8.5-jessie# create a working directoryWORKDIR /go/src/app# add source codeADD src src# run main.goCMD ["go", "run", "src/main.go"]

我们使用 debian jessie 版本的 golang 镜像,因为像 go get 这样的命令需要安装有 git 等工具。对于生产版本,我们会用更加轻量的版本,如 alpine

构建并运行该镜像:

$ cd go-docker$ docker build -t go-docker-dev .$ docker run --rm -it -p 8080:8080 go-docker-dev

成功后可以通过 http://localhost:8080 来访问。按下 Ctrl+C 可以中断服务。

但这并没有多大意义,因为每次修改代码时,我们都必须构建和运行docker 镜像。

一个更好的版本是将源代码挂载到 docker 容器中,并使用容器内的 shell 来停止和启动 go run

$ cd go-docker$ docker build -t go-docker-dev .$ docker run --rm -it -p 8080:8080 -v $(pwd):/go/src/app              go-docker-dev bashroot@id:/go/src/app# go run src/main.go

这个命令会提供一个 shell,我们可以在里面执行 go run src/main.go 以启动服务。我们可以在宿主机上编辑 main.go 并重新运行该命令来查看变化,因为现在源代码已经直接挂载到了容器中。

但是,如何管理包呢?

包管理和镜像分层

Go 的包管理 仍处在实验阶段。有很多工具可以选择,但是我最喜欢的是 Glide。我们将在容器中安装 Glide 并使用它。

go-docker 项目中新建两个文件 glide.yamlglide.lock

$ cd go-docker$ touch glide.yaml$ touch glide.lock

按照下面所示修改 Dockerfile 并构建一个新的镜像:

FROM golang:1.8.5-jessie# install glideRUN go get github.com/Masterminds/glide# create a working directoryWORKDIR /go/src/app# add glide.yaml and glide.lockADD glide.yaml glide.yamlADD glide.lock glide.lock# install packagesRUN glide install# add source codeADD src src# run main.goCMD ["go", "run", "src/main.go"]

如果你观察比较细致,你会发现 glide.yamlglide.lock 是分开添加的(并没有用 ADD . .),这样会导致有单独分离的层。将包管理分离为单独的层,可以充分利用 Docker 层的缓存,并且只有当对应的文件发生变化才会导致重新编译,比如:新增或删除了一个包。因此,glide install 不会在每次修改了代码之后都去执行。

让我们进入容器的 shell 安装一个包:

$ cd go-docker$ docker build -t go-docker-dev .$ docker run --rm -it -v $(pwd):/go/src/app go-docker-dev bashroot@id:/go/src/app# glide get github.com/golang/glog

Glide 会将所有包安装到 vendor 目录,该目录可以被 gitignoreddockerignored。使用 glide.lock 来锁定某个包的版本。要安装(或重新安装)glide.yaml 中提到的所有包,执行:

$ cd go-docker$ docker run --rm -it -p 8080:8080 -v $(pwd):/go/src/app              go-docker-dev bashroot@id:/go/src/app# glide install

现在 go-docker 目录有所增长:

.├── Dockerfile├── glide.lock├── glide.yaml├── src│   └── main.go└── vendor/

实时重载

codegangsta/gin 是我最喜欢的实时重载工具。它简直就是为 Go web 服务而生的。我们使用 go get 来安装 gin:

FROM golang:1.8.5-jessie# install glideRUN go get github.com/Masterminds/glide# install ginRUN go get github.com/codegangsta/gin# create a working directoryWORKDIR /go/src/app# add glide.yaml and glide.lockADD glide.yaml glide.yamlADD glide.lock glide.lock# install packagesRUN glide install# add source codeADD src src# run main.goCMD ["go", "run", "src/main.go"]

构建镜像并运行 gin 以便当我们修改了 src 中的源代码时可以自动重新编译:

$ cd go-docker$ docker build -t go-docker-dev .$ docker run --rm -it -p 8080:8080 -v $(pwd):/go/src/app              go-docker-dev bashroot@id:/go/src/app# gin --path src --port 8080 run main.go

注意到 web-server 需要一个 PORT 的环境变量来监听,因为 gin 会随机设置 PORT 变量并代理到该端口的连接。

现在,修改 src 目录下的内容会触发重新编译,所有更新的内容可以实时在 http://localhost:8080 访问到。

一旦开发完毕,我们可以构建二进制文件并运行它,而不需要使用 go run 命令。可以使用相同的镜像来构建,或者也可以使用 Docker 的多阶段构建,即使用 golang 镜像来构建并使用迷你 linux 容器如 alpine 来运行服务。

单阶段生产构建

FROM golang:1.8.5-jessie# install glideRUN go get github.com/Masterminds/glide# create a working directoryWORKDIR /go/src/app# add glide.yaml and glide.lockADD glide.yaml glide.yamlADD glide.lock glide.lock# install packagesRUN glide install# add source codeADD src src# build main.goRUN go build src/main.go# run the binaryCMD ["./main"]

构建并运行该一体化的镜像:

$ cd go-docker$ docker build -t go-docker-prod .$ docker run --rm -it -p 8080:8080 go-docker-prod

因为底层使用了 Debian 镜像,该镜像会达到 750 MB 左右的大小(取决于你的源代码)。让我们看看如何缩减体积。

多阶段生产构建

多阶段构建允许你在一个完整的 OS 环境中进行构建,但构建后的二进制文件通过一个非常苗条的镜像来运行,该镜像仅比构建后的二进制文件略大一点而已。

FROM golang:1.8.5-jessie as builder# install glideRUN go get github.com/Masterminds/glide# setup the working directoryWORKDIR /go/src/appADD glide.yaml glide.yamlADD glide.lock glide.lock# install dependenciesRUN glide install# add source codeADD src src# build the sourceRUN go build src/main.goRUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o main src/main.go# use a minimal alpine imageFROM alpine:3.7# add ca-certificates in case you need themRUN apk update && apk add ca-certificates && rm -rf /var/cache/apk/*# set working directoryWORKDIR /root# copy the binary from builderCOPY --from=builder /go/src/app/main .# run the binaryCMD ["./main"]

现在二进制文件为 14 MB 左右,docker 镜像为 18 MB 左右。真是多亏了 alpine

想减小二进制文件体积吗?继续看吧。

福利:使用 UPX 来压缩二进制文件

Hasura,我们已经在到处使用 UPX 了,压缩后我们的 CLI 二进制文件从 50 MB 左右降到 8 MB左右,大大加快了下载速度。UPX 可以极快地进行原地解压,不需要额外的工具,因为它将解压器嵌入到了二进制文件内部。

FROM golang:1.8.5-jessie as builder# install xzRUN apt-get update && apt-get install -y     xz-utils && rm -rf /var/lib/apt/lists/*# install UPXADD https://github.com/upx/upx/releases/download/v3.94/upx-3.94-amd64_linux.tar.xz /usr/localRUN xz -d -c /usr/local/upx-3.94-amd64_linux.tar.xz |     tar -xOf - upx-3.94-amd64_linux/upx > /bin/upx &&     chmod a+x /bin/upx# install glideRUN go get github.com/Masterminds/glide# setup the working directoryWORKDIR /go/src/appADD glide.yaml glide.yamlADD glide.lock glide.lock# install dependenciesRUN glide install# add source codeADD src src# build the sourceRUN go build src/main.goRUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o main src/main.go# strip and compress the binaryRUN strip --strip-unneeded mainRUN upx main# use a minimal alpine imageFROM alpine:3.7# add ca-certificates in case you need themRUN apk update && apk add ca-certificates && rm -rf /var/cache/apk/*# set working directoryWORKDIR /root# copy the binary from builderCOPY --from=builder /go/src/app/main .# run the binaryCMD ["./main"]

UPX 压缩后的二进制文件为 3 MB 左右并且 docker 镜像为 6 MB 左右。

相比最开始的镜像,缩减了 100 倍

如果你有更好的建议或是你需要其他的使用场景,请在评论区留言或者去 HackerNewsReddit 进行讨论。

本站仅提供存储服务,所有内容均由用户发布,如发现有害或侵权内容,请点击举报
打开APP,阅读全文并永久保存 查看更多类似文章
猜你喜欢
类似文章
【热】打开小程序,算一算2024你的财运
容器编排技术
从Go程序第一行代码,到在 K8s 上运行,要经历多少步?
docker执行python文件
Docker run hello-world 失败的解决办法
关于docker的15个小tip
[N1盒子] armbian 安装docker +portainer面板以及汉化教程
更多类似文章 >>
生活服务
热点新闻
分享 收藏 导长图 关注 下载文章
绑定账号成功
后续可登录账号畅享VIP特权!
如果VIP功能使用有故障,
可点击这里联系客服!

联系客服