对于dockerfile而言,何为完美? 我认为应该知足如下三点:linux
话说不说,直接上拿走可用的Dockerfile。golang
FROM golang:1.13.5-alpine3.10 AS builder
WORKDIR /build RUN adduser -u 10001 -D app-runner
ENV GOPROXY https://goproxy.cn
COPY go.mod . COPY go.sum . RUN go mod download
COPY . . RUN CGO_ENABLED=0 GOARCH=amd64 GOOS=linux go build -a -o your-application .
FROM alpine:3.10 AS final
WORKDIR /app COPY --from=builder /build/your-application /app/ #COPY --from=builder /build/config /app/config
COPY --from=builder /etc/passwd /etc/passwd COPY --from=builder /etc/ssl/certs/ca-certificates.crt /etc/ssl/certs/
USER app-runner
ENTRYPOINT ["/app/your-application"] 复制代码
首先,这个dockerfile分为builder和final两部分。docker
builder选择了golang:1.13.5-alpine3.10
做为编译的基础镜像,相比于golang:1.13
, 一方面是由于它体积更小,另外一方面是我发现golang:1.13
的编译结果,在alpine:3.10
中会报not found
的错误,虽然说有人提供了其它的解决方案,可是能直接避免,为啥不避免呢。shell
RUN adduser -u 10001 -D app-runner
复制代码
接着是建立了一个app-runner
的用户, -D
表示无密码。缓存
此用户的信息是是须要拷到final中,做为应用程序的启动用户。这是为了不使用container中的默认用户root
,那但是有安全漏洞的,详细解释,能够参考这篇medium上的文章Processes In Containers Should Not Run As Root安全
再下面的四行,bash
ENV GOPROXY https://goproxy.cn
COPY go.mod .
COPY go.sum .
RUN go mod download
复制代码
是配置了国内的代理,安装依赖包了。这里用go mod download
的好处是下次构建镜像文件时,当go.mod
和go.sum
没有改变时,它是有缓存的,能够避免重复下载依赖包,加快构建。app
builder的最后,就是把当前目录的文件拷过去,编译代码了。ui
COPY . .
RUN CGO_ENABLED=0 GOARCH=amd64 GOOS=linux go build -a -o your-application .
复制代码
final选择了alpine:3.10
,一方面是体积小,只有5m;另外一方面也是和构建镜像的alpine版本保持一致。spa
接下来几行没啥说的,就是把构建结果、配置文件(有的话)和用户的相关文件拷过去。
下面的这步必定不要忘记了,
USER app-runner
复制代码
没有它,container启动时就是用root用户启动了!!! 若是被攻击了,那黑客但是就有root权限了(不要问我为啥会被攻击)。
最后,设置一个ENTRYPOINT
,完事!
若是你程序的启动过程比较复杂,或者是要在启动时根据环境变量的值作不一样的操做,那仍是写个shell文件吧。