gomock使用

安装

// 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]

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容

  • 安装 安装gomock软件包和mockgen代码生成工具 基本用法 1.用mockgen为要模拟的接口生成模拟。2...
    一枚平庸的软件工程师阅读 7,154评论 0 2
  • 在Golang的官方Repo(https://github.com/golang/)中有一个单独的工程叫"mock...
    CZ_Golang阅读 64,326评论 1 43
  • 序言 要写出好的测试代码,必须精通相关的测试框架。对于Golang的程序员来说,至少需要掌握下面四个测试框架: G...
    Aedan阅读 2,293评论 0 0
  • 原文地址:使用 Gomock 进行单元测试 前言 在实际项目中,需要进行单元测试的时候。却往往发现有一大堆依赖项。...
    EDDYCJY阅读 8,966评论 0 5
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,709评论 2 59