注:因为是第一次跟着教程做的。应该还有优化的空间,可能还会有比较蠢的代码片段。
Dockerfile分为两段:
- 第一段,构建一个临时的build用的镜像,用来编译生成go的运行文件。
- 第二段,构建一个纯净版空镜像scratch作为运行镜像,把编译好的运行文件和相关需要的配置文件什么的一起COPY到新的运行镜像里。这样运行镜像就可以非常小。如果为了调试方便,还是建议用alpine镜像。
PS,虽然我看到的教程里是可以用scratch来运行go编译后的运行文件。但是我没成功,最后还是用了alpine镜像。
Go工程结构
因为是练习,所以构建了一个相对简单的工程
打包思路
抛开docker,先梳理一下整个go工程的打包步骤:
- 编译工程需要有go 1.16的运行环境
- 由于使用了go mod,需要设置GOPROXY=https://goproxy.io,direct
- 工程启动文件是main.go,我习惯把工程编译到一个新的work文件夹里,运行文件起名server。所以编译命令是
go build -o work/server main.go
- 编译后的运行方式是
cd work
./server
第一段:编译镜像
- 使用go的官方基础镜像包,alpine版本。
FROM golang:1.16-alpine AS build
go mod我使用的是1.16的版本,对应的golang镜像也用对应版本。
注意这里用的是FROM ... AS ... (这个build要记住,后面要考)
- 定义工作目录是/project,并且把相关编译脚本拷贝到镜像的/project目录下
WORKDIR /project/
COPY ./project /project
- 运行编译命令,生成运行文件
RUN go env -w GOPROXY=https://goproxy.io,direct
RUN go build -o /project/work/server main.go
第二段:运行镜像
- 运行镜像使用alpine作为基础镜像
FROM alpine
- 配置一些基础设置,比如说时区
ENV TZ Asia/Shanghai
- 从编译镜像将运行需要的工程文件拷贝过来
COPY --from=build /project/work/ /work/
注:这里的build就是之前那个FROM ... AS build
- 配置工作目录和运行相关
# 定义工作目录为work
WORKDIR /work
# 开放http 80端口
EXPOSE 80
# 启动http服务
ENTRYPOINT ["./server"]
完整的Dockerfile
# 编译镜像
FROM golang:1.16-alpine AS build
WORKDIR /project/
COPY ./project /project
RUN go env -w GOPROXY=https://goproxy.io,direct
RUN go build -o /project/work/server main.go
# 运行镜像
FROM alpine
ENV TZ Asia/Shanghai
COPY --from=build /project/work/ /work/
# 定义工作目录为work
WORKDIR /work
# 开放http 80端口
EXPOSE 80
# 启动http服务
ENTRYPOINT ["./server"]
构建并推送镜像
# 在Dockerfile所在文件夹运行
docker build -t xxxx/xxxx:v1 .
# 构建成功后,运行测试一下
docker run -d xxxx/xxxx:v1
# docker ps查看一下运行情况
docker ps
# 查看一下容器对应的pid
docker inspect 554c4578242e|grep -i pid
# 查看对应的IP配置
nsenter -t 256418 -n ip a