今天小编给大家分享一下docker怎么打包golang应用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。
一、错误的打包方式
在本地环境编译,然后将可执行程序放入 alpine(docker.io/alpine:latest)
准备web程序
package main import ( "fmt" "net/http") func main() { server := &http.Server{ Addr: ":8888", } http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) { fmt.Fprintln(w, "hello world") }) fmt.Println("server startup...") if err := server.ListenAndServe(); err != nil { fmt.Printf("server startup failed, err:%v\n", err) }}
go build hello.go
dockerfile
FROM docker.io/alpine:latestMAINTAINER demo <juest a demo> #alpine内的包管理RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositories#安装bashRUN apk add --update bash && rm -rf /var/cache/apk/* RUN mkdir -p /data/goCOPY hello /data/go EXPOSE 8888 ENTRYPOINT ["/data/go/hello"]
构建镜像
docker build -t demo/go-hello:1.0 -f dockerfile .
将在本地生成将demo/go-hello:1.o镜像
创建并运行容器
原因:编译的hello二进制程序不是存静态程序,还依赖一些库,但这些库在alpine镜像中找不到。
二、正确的打包流程
需要放入alpine镜像里运行的go程序,可以直接使用golang:alpine来编译,但我们在golang:alpine基础上再构建一个镜像,这个镜像中包含bash、GO111MODULE、GOPROXY等环境变量。
dockerfile
FROM docker.io/golang:alpine RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositoriesRUN apk add --update bash && rm -rf /var/cache/apk/* # 为我们的镜像设置必要的环境变量ENV GO111MODULE=on \ CGO_ENABLED=0 \ GOOS=linux \ GOARCH=amd64\ GOPROXY=https://goproxy.cn,direct
构建自己的go编译镜像
docker build -t go-build:1.0 -f dockerfile .
运行go-build:1.0 镜像,编译go项目代码:
#xxx为本地go代码路径docker run -it --rm -v xxx:/data/go demo/go-build:1.0 /bin/bashcd /data/gogo build hello.go
生成了hello可执行文件,且为纯静态的。
将编译得到的hello二进制打入alpine:latest
dockerfile2
FROM docker.io/alpine:latestMAINTAINER demo <juest a demo> #alpine内的包管理RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositories#安装bashRUN apk add --update bash && rm -rf /var/cache/apk/* RUN mkdir -p /data/goCOPY hello /data/go EXPOSE 8888 ENTRYPOINT ["/data/go/hello"]
打包
docker build -t demo/go-hello:1.0 -f dockerfile2 .
运行demo/go-hello:1.0
三、使用scratch构建镜像
scratch为空镜像,适合那些没有任何外部依赖的程序,刚好前面的hello程序没有任何依赖!
dockerfile3
FROM scratchMAINTAINER demo <juest a demo> COPY hello / EXPOSE 8888 ENTRYPOINT ["/hello"]
构建
docker build -t demo/go-hello:2.0 -f dockerfile3 .
以scratch为基础构建出来的镜像是最小的
运行
四、参考以太坊的打包
目录结构
dockerfile
# Support setting various labels on the final imageARG COMMIT=""ARG VERSION=""ARG BUILDNUM="" # Build Geth in a stock Go builder containerFROM golang:alpine as builder RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositoriesRUN apk add --no-cache gcc musl-dev linux-headers git ENV GO111MODULE=on \ CGO_ENABLED=0 \ GOOS=linux \ GOARCH=amd64\ GOPROXY=https://goproxy.cn,direct # Get dependencies - will also be cached if we won't change go.mod/go.sumCOPY go.mod /web/COPY go.sum /web/RUN cd /web && go mod download ADD . /webRUN cd /web && go build -o ./cmd/app main.go # Pull Geth into a second stage deploy alpine containerFROM alpine:latest RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositoriesRUN apk add --no-cache ca-certificatesCOPY --from=builder /web/cmd/app /usr/local/bin/ EXPOSE 8080ENTRYPOINT ["app"] # Add some metadata labels to help programatic image consumptionARG COMMIT=""ARG VERSION=""ARG BUILDNUM="" LABEL commit="$COMMIT" version="$VERSION" buildnum="$BUILDNUM"
以上就是“docker怎么打包golang应用”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。