目錄
- 一、錯誤的打包方式
- 二、正確的打包流程
- 三、使用scratch構建鏡像
- 四、參考以太坊的打包
一、錯誤的打包方式
在本地環境編譯,然后將可執行程序放入 alpine(docker.io/alpine:latest)
1.準備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
2.dockerfile
FROM docker.io/alpine:latest MAINTAINER demo <juest a demo> #alpine內的包管理 RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositories #安裝bash RUN apk add --update bash && rm -rf /var/cache/apk/* RUN mkdir -p /data/go COPY hello /data/go EXPOSE 8888 ENTRYPOINT ["/data/go/hello"]
3.構建鏡像
docker build -t demo/go-hello:1.0 -f dockerfile .
4.將在本地生成將demo/go-hello:1.o鏡像
5.創建并運行容器
原因:編譯的hello二進制程序不是存靜態程序,還依賴一些庫,但這些庫在alpine鏡像中找不到。
二、正確的打包流程
需要放入alpine鏡像里運行的go程序,可以直接使用golang:alpine來編譯,但我們在golang:alpine基礎上再構建一個鏡像,這個鏡像中包含bash、GO111MODULE、GOPROXY等環境變量。
1.dockerfile
FROM docker.io/golang:alpine RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositories RUN apk add --update bash && rm -rf /var/cache/apk/* # 為我們的鏡像設置必要的環境變量 ENV GO111MODULE=on \ CGO_ENABLED=0 \ GOOS=linux \ GOARCH=amd64\ GOPROXY=https://goproxy.cn,direct
2.構建自己的go編譯鏡像
docker build -t go-build:1.0 -f dockerfile .
3.運行go-build:1.0 鏡像,編譯go項目代碼:
#xxx為本地go代碼路徑 docker run -it --rm -v xxx:/data/go demo/go-build:1.0 /bin/bash cd /data/go go build hello.go
生成了hello可執行文件,且為純靜態的。
將編譯得到的hello二進制打入alpine:latest
dockerfile2
FROM docker.io/alpine:latest MAINTAINER demo <juest a demo> #alpine內的包管理 RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositories #安裝bash RUN apk add --update bash && rm -rf /var/cache/apk/* RUN mkdir -p /data/go COPY hello /data/go EXPOSE 8888 ENTRYPOINT ["/data/go/hello"]
5.打包
docker build -t demo/go-hello:1.0 -f dockerfile2 .
6.運行demo/go-hello:1.0
三、使用scratch構建鏡像
scratch為空鏡像,適合那些沒有任何外部依賴的程序,剛好前面的hello程序沒有任何依賴!
1.dockerfile3
FROM scratch MAINTAINER demo <juest a demo> COPY hello / EXPOSE 8888 ENTRYPOINT ["/hello"]
2.構建
docker build -t demo/go-hello:2.0 -f dockerfile3 .
3.以scratch為基礎構建出來的鏡像是最小的
運行
四、參考以太坊的打包
目錄結構
dockerfile
# Support setting various labels on the final image ARG COMMIT="" ARG VERSION="" ARG BUILDNUM="" # Build Geth in a stock Go builder container FROM golang:alpine as builder RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositories RUN 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.sum COPY go.mod /web/ COPY go.sum /web/ RUN cd /web && go mod download ADD . /web RUN cd /web && go build -o ./cmd/app main.go # Pull Geth into a second stage deploy alpine container FROM alpine:latest RUN echo "https://mirror.tuna.tsinghua.edu.cn/alpine/v3.4/main" > /etc/apk/repositories RUN apk add --no-cache ca-certificates COPY --from=builder /web/cmd/app /usr/local/bin/ EXPOSE 8080 ENTRYPOINT ["app"] # Add some metadata labels to help programatic image consumption ARG COMMIT="" ARG VERSION="" ARG BUILDNUM="" LABEL commit="$COMMIT" version="$VERSION" buildnum="$BUILDNUM"