gomock使用

作者: wangfeiq | 来源:发表于2022-10-31 22:58 被阅读0次

    安装

    // go 1.16之后版本
    go install github.com/golang/mock/mockgen@v1.6.0
    

    安装好后,可以使用mockgen -version查看版本。

    接口mock生成

    mockgen有两种使用模式,一种时source模式,一种是reflect模式。

    source模式

    mockgen -source=xxx.go [other options]

    • -source指定生成mock函数的文件。该文件需要包含相关的接口。

    reflect模式

    mockgen 包 接口,如mockgen net/http RoundTripper
    多个接口用,分隔,如mockgen database/sql/driver Conn,Driver
    .表示当前路径下的包mockgen . Conn,Driver

    通常情况下,还需要带上几个其他参数。
    mockgen -build_flags=--mod=mod -destination=mock/mockhttp.go -package=mock net/http RoundTripper

    • -build_flags表示编译参数,会直接传递给go buildgo build --mod=mod
    • -destination表示生成的mock函数存放的位置。
    • -package表示mock文件的包名,建议和-destination中的目录名字一致。

    使用反射模式的时候,如果出现以下报错,可以通过以下三种方法解决。

    cannot find package "."
    ... github.com/golang/mock/mockgen/model
    
    • 使用source模式。
    • 导入包import _ "github.com/golang/mock/mockgen/model"
    • mockgen命令加上--build_flags=--mod=mod选项

    接口mock使用

    func TestGetFromDB(t *testing.T) {
        // 创建gomock控制器,用来记录后续的操作信息
        ctrl := gomock.NewController(t)
        defer ctrl.Finish()
        // 调用mockgen生成代码中的NewMockTi方法,Ti是一个接口类型
        m := mock.NewMockTi(ctl)
        ...
    }
    

    接口mock的使用场景是我们在调用其他接口函数的时候,不真正调用该函数而直接拿到函数返回值。此时有多种情况:

    • 无论请求是什么都返回相同的内容。
    m.EXPECT().YourFunc(gomock.Any()).Return(yourReturn)
    
    • 根据接口函数的不同入参,返回不同的返回值。此时需要用到gomock的比较函数。
    m.EXPECT().YourFunc(gomock.Eq(yourInput)).Return(yourReturn)
    

    还有其他几个比较函数

    • gomock.Eq(value):表示一个等价于value值的参数
    • gomock.Not(value):表示一个非value值的参数
    • gomock.Any():表示任意值的参数
    • gomock.Nil():表示空值的参数
    • SetArg(n, value):设置第n(从0开始)个参数的值,通常用于指针参数或切片。
      这里单独说一下SetArg的适用场景,假设你有一个需要mock的接口如下:
    type YourInterface {
      SetValue(arg *int)
    }
    

    此时,打桩的时候就可以使用SetArg来修改参数的值。
    m.EXPECT().SetValue(gomock.Any()).SetArg(0, 7) // 将SetValue的第一个参数设置为7

    • 根据调用该接口函数的次数,按照事先填充的返回值顺序,每次调用返回对应顺序的返回值。如第一次调用返回a,第二次调用返回b。此处需要设置每个接口可调用的次数。
    first := m.EXPECT().YourFunc(gomock.Eq(yourInput)).Return(yourReturn1)
    second := m.EXPECT().YourFunc(gomock.Eq(yourInput)).Return(yourReturn2)
    gomock.InOrder(first, second)
    # 第一次调用返回结果1,第二次调用返回结果2
    
    • Call.Do():声明在匹配时要运行的操作
    m.EXPECT().Get(gomock.Any()).Do(func(key string) {
        t.Logf("input key is %v\n", key)
    })
    
    • Call.DoAndReturn():声明在匹配调用时要运行的操作,并且模拟返回该函数的返回值
    m.EXPECT().Get(gomock.Any()).DoAndReturn(func(key string)(int, error) {
        t.Logf("input key is %v\n", key)
        return 10, nil
    })
    
    • Call.MaxTimes():设置最大的调用次数为 n 次
    • Call.MinTimes():设置最小的调用次数为 n 次
    • Call.AnyTimes():允许调用次数为 0 次或更多次
    • Call.Times():设置调用次数为 n 次

    默认Mock接口只能调用一次,如果要多次调用,需要设置调用次数。m.EXPECT().YourFunc(gomock.Eq(yourInput)).Return(yourReturn).AnyTimes()

    完整的例子

    • 源代码(mocksdk.go)
    type Ti interface {
        Get(i int) int
    }
    
    type Num struct {
        num int
    }
    
    func (n Num) Get(i int) int {
        return n.num + i
    }
    
    • 生成mock代码:mockgen -destination mock/mockget.go -package mock -source ./mocksdk.go
    • 测试代码
    func TestGet(t *testing.T) {
        ctl := gomock.NewController(t)
        defer ctl.Finish()
        mockTi := mock.NewMockTi(ctl)
        mockTi .EXPECT().Get(gomock.Eq(2)).Return(1)
            
        // 初始化接口实例
        ti := Ti(mockTi)
        fmt.Println(ti.Get(2))
    }
    

    执行单测

    go test ./user

    • 查看测试覆盖率
      go test -cover ./user
    • 可视化覆盖内容
    1. 生成测试覆盖率的 profile 文件
      go test ./. -coverprofile=cover.out
    2. 利用 profile 文件生成可视化界面
      go tool cover -html=cover.out

    生成多个接口的mock

    我们可以利用 go:generate 来完成批量生成接口mock。
    在接口文件中,给接口添加对应的注释,然后运行go generate即可。

    package person
    //go:generate mockgen -destination=../mock/male_mock.go -package=mock github.com/EDDYCJY/mockd/person Male
    type Male interface {
      Get(id int64) error
    }
    

    go generate ./...

    go generate [-run regexp] [-n] [-v] [-x] [build flags] [file.go... | packages]

    相关文章

      网友评论

        本文标题:gomock使用

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