美文网首页
Unit Test In Go With Docker

Unit Test In Go With Docker

作者: 51reboot | 来源:发表于2019-12-19 17:17 被阅读0次

    本篇文章主要讲解如何在 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 天然集成,实现项目的自动测试。
    作者:宋佳洋
    51reboot K8s课程1.18日开课,详情WeChat;17812796384 去了解

    相关文章

      网友评论

          本文标题:Unit Test In Go With Docker

          本文链接:https://www.haomeiwen.com/subject/agpdnctx.html