基于go的微服务搭建(四)- 用GoConvey做测试和mock

前端之家收集整理的这篇文章主要介绍了基于go的微服务搭建(四)- 用GoConvey做测试和mock前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

第四章:用GoConvey做测试和mock

我们应该怎样做微服务的测试?这有什么特别的挑战么.这节,我们将看下面几点:

  • 单元测试
  • 用Goconey写行为模式的单元测试
  • 介绍mocking技巧

因为这章不会改变核心服务代码,所以没有基测

微服务测试简介


首先,你必须记住测试金字塔:

单元测试必须作为你集成,e2e的基础,验收测试更不容易开发和维护
微服务有一些不同的测试难点,构建软件架构用到的准则和做测试一样.微服务的单元测试和传统的不太一样,我们会在这里讲一下.
总之,我强调几点:

  • 做正常地单元测试-你的商业逻辑,验证器等等不因为在微服务上运行而不同.
  • 集成的部分,例如和其他服务沟通,发送信息,使用数据库,这些必须用依赖注入的方法来设计,这样才能用上mock
  • 许多微服务的特性:配置文件,其他服务的沟通,弹力测试等.花很多的时间才能做一点测试.这些测试可以做集成测试,你把docker容器整体的做测试.这样性价比会比较高.

代码


完整代码

git checkout P4

介绍


go的单元测试又go的作者设计的遵从语言习惯的模式.测试文件由命名来识别.如果我们想测试handler.go中的东西,我们创建文件handlers_test.go,在同一个文件夹下.
我们从悲观测试开始,断言404,当我们请求不存在的地址

package service

import (
        . "github.com/smartystreets/goconvey/convey"
        "testing"
        "net/http/httptest"
)

func TestGetAccountWrongPath(t *testing.T) {

        Convey("Given a HTTP request for /invalid/123",t,func() {
                req := httptest.NewRequest("GET","/invalid/123",nil)
                resp := httptest.NewRecorder()

                Convey("When the request is handled by the Router",func() {
                        NewRouter().ServeHTTP(resp,req)

                        Convey("Then the response should be a 404",func() {
                                So(resp.Code,ShouldEqual,404)
                        })
                })
        })
}

这个测试显示"Given-when-then"(如果-当-推断)的模式.我们也用httptest包,我们用它来声明请求的object也用做回复的object用来作为断言的条件.
去accountservice下运行他:

> go test ./...
?       github.com/callistaenterprise/goblog/accountservice    [no test files]
?       github.com/callistaenterprise/goblog/accountservice/dbclient    [no test files]
?       github.com/callistaenterprise/goblog/accountservice/model    [no test files]
ok      github.com/callistaenterprise/goblog/accountservice/service    0.012s

./...会运行当前文件夹和所有子文件夹下的测试文件.我们也可以进入service文件夹下go test,这会运行这个文件夹下的测试.

Mocking


上面的测试不需要mock,因为我们不会用到GetAccount里面的DBClient.对于好的请求,我们需要返回结果,我们就需要mock客户端来连接BoltDb.有许多mocking的方法.我最喜欢的是stretchr/testify/mock这个包
在/dbclient文件夹下,创建mockclient.go来实现IBoltClient接口:

package dbclient

import (
        "github.com/stretchr/testify/mock"
        "github.com/callistaenterprise/goblog/accountservice/model"
)

// MockBoltClient is a mock implementation of a datastore client for testing purposes.
// Instead of the bolt.DB pointer,we're just putting a generic mock object from
// strechr/testify
type MockBoltClient struct {
        mock.Mock
}

// From here,we'll declare three functions that makes our MockBoltClient fulfill the interface IBoltClient that we declared in part 3.
func (m *MockBoltClient) QueryAccount(accountId string) (model.Account,error) {
        args := m.Mock.Called(accountId)
        return args.Get(0).(model.Account),args.Error(1)
}

func (m *MockBoltClient) OpenBoltDb() {
        // Does nothing
}

func (m *MockBoltClient) Seed() {
        // Does nothing
}

MockBoltClient现在可以作为我们可以编写的mock.向上边那样,我们隐式的定义了所有的函数,实现IBoltClient接口.
如果你不喜欢这样的mock方法,可以看一下mockery,他可以产生任何go接口的mock
QueryAccount函数里有点奇怪.但这就是testify的做法,这样能让我们有一个全面的内部控制的mock.

编写mock


我们创建下一个测试函数在handlers_test.go中:

func TestGetAccount(t *testing.T) {
        // Create a mock instance that implements the IBoltClient interface
        mockRepo := &dbclient.MockBoltClient{}

        // Declare two mock behavIoUrs. For "123" as input,return a proper Account struct and nil as error.
        // For "456" as input,return an empty Account object and a real error.
        mockRepo.On("QueryAccount","123").Return(model.Account{Id:"123",Name:"Person_123"},nil)
        mockRepo.On("QueryAccount","456").Return(model.Account{},fmt.Errorf("Some error"))
        
        // Finally,assign mockRepo to the DBClient field (it's in _handlers.go_,e.g. in the same package)
        DBClient = mockRepo
        Convey("Given a HTTP request for /accounts/123",func() {
        req := httptest.NewRequest("GET","/accounts/123",nil)
        resp := httptest.NewRecorder()

        Convey("When the request is handled by the Router",func() {
                NewRouter().ServeHTTP(resp,req)

                Convey("Then the response should be a 200",func() {
                        So(resp.Code,200)

                        account := model.Account{}
                        json.Unmarshal(resp.Body.Bytes(),&account)
                        So(account.Id,"123")
                        So(account.Name,"Person_123")
                })
        })
})
}

这段测试请求path/accounts/123,我们的mock实现了这个.在when中,我们断言http状态,反序列化Account结构,同时段验结果和我们的mock的结果相同.
我喜欢Goconvey因为这种"Given-when-then"的方式很容易读
我们也请求一个悲观地址/accounts/456,断言会得到http404:

Convey("Given a HTTP request for /accounts/456","/accounts/456",req)

                Convey("Then the response should be a 404",404)
                })
        })
})

跑一下.

> go test ./...
?       github.com/callistaenterprise/goblog/accountservice    [no test files]
?       github.com/callistaenterprise/goblog/accountservice/dbclient    [no test files]
?       github.com/callistaenterprise/goblog/accountservice/model    [no test files]
ok      github.com/callistaenterprise/goblog/accountservice/service    0.026s

全绿!goconvey有一个GUI能在我们每次保存文件自动执行所有测试.我不细讲了,这里看一下代码覆盖度报告:


goconvey这种用行为测试方式写的单元测试并不是每个人都喜欢.有许多其他的测试框架.你能搜索到很多.
如果我们看测试金字塔上面,我们会想写集成测试,或者最后的验收测试.我们之后会启动真正的boltDb,之后来讲一讲集成测试.也许会用go docker的远程api和写好的boltdb镜像

总结


这部分我们用goconvey写第一个单元测试.同事用mock包帮我们模拟.下一节,我们会启动docker swarm并且部署我们的服务进swarm中

猜你在找的Go相关文章