开发者

docker打包golang应用的过程分析

目录
  • 一、错误的打包方式
  • 二、正确的打包流程
  • 三、使用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镜像

docker打包golang应用的过程分析

5.创建并运行容器

docker打包golang应用的过程分析

原因:编译的hello二进制程序不是存静态程序,还依赖一些库,但这些库在alpine镜像中找不到。

二、正确的打包流程

需要放入alpine镜像里运行的go程序,可以直接使用golang:alpine来编译,但我们在golang:alpine基础上再构建一个镜像,这个镜像中包含bash、GO11js1MODULE、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 .

docker打包golang应用的过程分析

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可执行文件,且为纯静态的。

docker打包golang应用的过程分析

将编译得到的hello二进制打入alpine:latest

dockerfile2

FROM       docker.io/alpine:latest
MAINTAINER demo <juest a demo>
 
#alpine内的包管理
RUN echo "https开发者_Go学习://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

docker打包golang应用的过程分析

三、使用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为基础构建出来的镜像是最小的

docker打包golang应用的过程分析

运行

docker打包golang应用的过程分析

四、参考以太坊的打包

目录结构

docker打包golang应用的过程分析

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 --fjsrom=builder /web/cmd/app /usr/local/bin/
 
EXPOSE 8080
ENTRYPOINT ["app"]
 
# Add some metadata djuxOIlabels to help programatic image consumption
ARG COMMIT=""
ARG VERSION=""
ARG BUILDNUM=""
 
LABEL commit="$COMMIT" version="$VERSION" buildnum="$BUILDNUM"

到此这篇关于docker打包golang应用的文章就介绍到这了,更多相关docker打包gola编程客栈ng应用内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多http://www.devze.com多支持我们!

0

上一篇:

下一篇:

精彩评论

暂无评论...
验证码 换一张
取 消

最新开发

开发排行榜