本篇文章主要讲解如何在 Docker 中进行 Go 单元测试,依赖 Docker 和 Go Modules。

为什么是 Docker

在 Docker 之前我们往往需要在 Jenkins 服务器上配置不同的 Go 版本以及针对每个项目配置 GOPATH,项目之间的隔离性差,经常出现基础库版本冲突的问题。

有了 Docker,我们可以在不同容器中运行单元测试,该测试不局限不同项目,甚至可以是同一项目不同分支。

所以在测试隔离性和项目测试并发度上都有很大提升,而且测试结束后,环境清理也简单许多。

为什么是 Go Modules

Go Modules 作为官方默认的包管理工具,基本解决了 Go 长期存在的包管理问题,它为我们的项目管理带来很多好处:

  • 自动解析和添加依赖
  • 签名验证
  • 依赖缓存
  • 支持相对路径依赖
  • 支持依赖一键打包,方便在离线环境下运行程序

实际例子
下面我们来看一个简单例子,来自 Gin Testing Example,项目目录结构为:

$ tree .
.
├── go.mod
├── go.sum
├── main.go
└── main_test.go

0 directories, 4 files

main.go 内容:

package main

func setupRouter() *gin.Engine {
    r := gin.Default()
    r.GET("/ping", func(c *gin.Context) {
        c.String(200, "pong")
    })
    return r
}

func main() {
    r := setupRouter()
    r.Run(":8080")
}

main_test.go 内容:

package main

import (
    "net/http"
    "net/http/httptest"
    "testing"

    "github.com/stretchr/testify/assert"
)

func TestPingRoute(t *testing.T) {
    router := setupRouter()

    w := httptest.NewRecorder()
    req, _ := http.NewRequest("GET", "/ping", nil)
    router.ServeHTTP(w, req)

    assert.Equal(t, 200, w.Code)
    assert.Equal(t, "pong", w.Body.String())
}

首先尝试在主机执行 go test ./... ,测试通过,下面我们想办法在 Docker 中进行测试。

Docker 运行测试

步骤1: 对项目依赖进行打包,方便在 Docker 无网环境下运行单元测试

go mod vendor

此时所有的依赖都打包放在项目根目录的 vendor 下面:

$ tree .
.
├── go.mod
├── go.sum
├── main.go
├── main_test.go
└── vendor
    ├── github.com
    │   ├── davecgh
    │   │   └── go-spew
    │   │       ├── LICENSE
    │   │       └── spew
    │   │           ├── bypass.go
    │   │           ├── bypasssafe.go
    │   │           ├── common.go

步骤2: 制作 Go Docker 镜像

因为我们只需要 Go 的标准环境,所以可以使用官方的标准镜像,例如 golang:1.12.1。

如果你有特殊需求,可以定制自己的 Go 镜像, 例如自定义启动命令,添加数据库依赖等:

  • Dockerfile:
FROM ubuntu
RUN apt-get update && apt-get install -y libssl1.0.0 libssl-dev gcc

RUN mkdir -p /data/db /opt/go/ /opt/gopath

# go1.xx.x.linux-amd64.tar.gz 解压缩后为 go
ADD go /opt/go 
RUN cp /opt/go/bin/* /usr/local/bin/
ENV GOROOT=/opt/go GOPATH=/opt/gopath

WORKDIR /ws
CMD GOPROXY=off go test -mod=vendor ./...
  • 执行 docker build:
    docker build . -t gotesting:v0.0.1

步骤3: 在 Docker 中运行 Golang 测试

  • 使用 golang:1.12.1:
$ docker run --volume=$(pwd):/ws \
     --workdir=/ws golang:1.12.1 \
    /bin/bash -c "GOPROXY=off go test -mod=vendor ./..."
ok      github.com/songjiayang/gin-test 0.011s
  • 使用自定义镜像 gotesting:v0.0.1:
    docker run --volume=$(pwd):/ws gotesting:v0.0.1
    ok      github.com/songjiayang/gin-test 0.014s

可以看到两种方式都能在 Docker 中运行 Go 单元测试,自定义镜像可以指定默认运行命令,使操作更简单,上面命令的含义是:

  • –volume=$(pwd):/ws :将项目根目录映射到 Docker 容器中的 /ws 目录。
  • –workdir=/ws : 指定容器运行工作目录为 /ws, 即映射外面项目根目录。
  • /bin/bash -c “GOPROXY=off go test -mod=vendor ./…” : 表示在 Docker 中以离线,vendor 模式运行 Go 单元测试。

做的更好

到目前为止,我们已经掌握在 Docker 中运行 Go 单元测试的方法。当然我们还可以做的更好,Docker 运行的标准输出可以和 Jenkins 天然集成,实现项目的自动测试。

作者:宋佳洋