Project Icon

mockey

Go 语言的轻量级函数模拟库

Mockey 是一个轻量级的 Go 语言函数模拟库,主要用于简化单元测试编写。它支持模拟函数、方法和变量,通过运行时重写指令实现 monkey patch。Mockey 提供条件过滤和增量修改等高级功能,兼容多种操作系统和架构,适用于 Go 1.13 及以上版本。该库在字节跳动等公司的服务中得到广泛应用。

Mockey

English | 中文

Mockey 是一个简单易用的 Golang mock 库,可以快速方便地模拟函数和变量。目前广泛应用于字节跳动服务的单元测试编写中。底层通过在运行时重写函数指令实现 monkey patch。

  1. 编译时需要禁用内联和编译优化,否则可能会导致 mock 失败或报错。详见下方 FAQ 章节。
  2. 在实际编写单元测试的过程中,建议与 Convey 库一起使用。

安装

go get github.com/bytedance/mockey@latest

快速入门

import (
	"fmt"
	"testing"

	. "github.com/bytedance/mockey"
	. "github.com/smartystreets/goconvey/convey"
)

func Foo(in string) string {
	return in
}

type A struct{}

func (a A) Foo(in string) string { return in }

var Bar = 0

func TestMockXXX(t *testing.T) {
	PatchConvey("TestMockXXX", t, func() {
		Mock(Foo).Return("c").Build()   // 模拟函数
		Mock(A.Foo).Return("c").Build() // 模拟方法
		MockValue(&Bar).To(1)           // 模拟变量

		So(Foo("a"), ShouldEqual, "c")        // 断言 `Foo` 被模拟
		So(new(A).Foo("b"), ShouldEqual, "c") // 断言 `A.Foo` 被模拟
		So(Bar, ShouldEqual, 1)               // 断言 `Bar` 被模拟
	})
	// `PatchConvey` 外部自动释放模拟
	fmt.Println(Foo("a"))        // a
	fmt.Println(new(A).Foo("b")) // b
	fmt.Println(Bar)             // 0
}

特性

  • 模拟函数和方法
    • 基础
      • 普通/可变参数函数
      • 普通/可变参数方法
      • 嵌套结构体方法
      • 私有类型的导出方法(不同包下)
    • 高级
      • 模拟后执行原函数
      • 协程条件过滤
      • 增量式改变模拟行为
      • 获取目标函数执行次数
      • 获取模拟函数执行次数
  • 模拟变量
    • 普通变量
    • 函数变量

兼容性

操作系统支持

  • Mac OS(Darwin)
  • Linux
  • Windows

架构支持

  • AMD64
  • ARM64

版本支持

  • Go 1.13+

许可证

Mockey 采用 Apache License 2.0 版本进行分发。Mockey 第三方依赖的许可证说明可在此处查看。

FAQ

Go 1.23 编译出错 "link: github.com/bytedance/mockey/internal/monkey/common: invalid reference to runtime.sysAllocOS"

添加构建标志 -ldflags=-checklinkname=0

如何禁用内联和编译优化?

  1. 命令行:go test -gcflags="all=-l -N" -v ./...
  2. Goland:在 Run/Debug Configurations > Go tool arguments 对话框中填入 -gcflags="all=-l -N"

模拟后仍然进入原函数?

  1. 未禁用内联或编译优化:可以尝试使用 debug 模式,如果能跑通说明是这个问题,请前往 FAQ 的相关章节
  2. 未调用 Build() 方法:忘记调用 Build(),导致没有实际生效
  3. 目标函数不完全匹配:
func TestXXX(t *testing.T) {
   Mock((*A).Foo).Return("c").Build()
   fmt.Println(A{}.Foo("a")) // 进入原函数,因为模拟目标应该是 `A.Foo`

   a := A{}
   Mock(a.Foo).Return("c").Build()
   fmt.Println(a.Foo("a")) // 进入原函数,因为模拟目标应该是 `A.Foo` 或使用 `GetMethod` 从实例 `a` 中提取
}
  1. 目标函数在其他协程中执行:
func TestXXX(t *testing.T) {
   PatchConvey("TestXXX", t, func() {
      Mock(Foo).Return("c").Build()
      go Foo("a") // 执行 'foo' 的时机不确定
   })
   // 主协程走到这里时,相关模拟已被 'PatchConvey' 释放。如果 'foo' 在此之前执行,则模拟成功,否则失败
   fmt.Println("over")
   time.Sleep(time.Second)
}

报错 "function is too short to patch"?

  1. 未禁用内联或编译优化:可以尝试使用 debug 模式,如果能跑通说明是这个问题,请前往 FAQ 的相关章节
  2. 函数确实太短:说明目标函数少于一行,导致编译后的机器码太短。一般两行以上就不会出现这个问题
  3. 重复模拟同一个函数:在最小单元的 PatchConvey 中重复模拟同一个函数。如果有这种需求,请获取 Mocker 实例后重新模拟
  4. 其他工具模拟了这个函数:例如 monkey 或其他工具已经模拟了这个函数
项目侧边栏1项目侧边栏2
推荐项目
Project Cover

豆包MarsCode

豆包 MarsCode 是一款革命性的编程助手,通过AI技术提供代码补全、单测生成、代码解释和智能问答等功能,支持100+编程语言,与主流编辑器无缝集成,显著提升开发效率和代码质量。

Project Cover

AI写歌

Suno AI是一个革命性的AI音乐创作平台,能在短短30秒内帮助用户创作出一首完整的歌曲。无论是寻找创作灵感还是需要快速制作音乐,Suno AI都是音乐爱好者和专业人士的理想选择。

Project Cover

有言AI

有言平台提供一站式AIGC视频创作解决方案,通过智能技术简化视频制作流程。无论是企业宣传还是个人分享,有言都能帮助用户快速、轻松地制作出专业级别的视频内容。

Project Cover

Kimi

Kimi AI助手提供多语言对话支持,能够阅读和理解用户上传的文件内容,解析网页信息,并结合搜索结果为用户提供详尽的答案。无论是日常咨询还是专业问题,Kimi都能以友好、专业的方式提供帮助。

Project Cover

阿里绘蛙

绘蛙是阿里巴巴集团推出的革命性AI电商营销平台。利用尖端人工智能技术,为商家提供一键生成商品图和营销文案的服务,显著提升内容创作效率和营销效果。适用于淘宝、天猫等电商平台,让商品第一时间被种草。

Project Cover

吐司

探索Tensor.Art平台的独特AI模型,免费访问各种图像生成与AI训练工具,从Stable Diffusion等基础模型开始,轻松实现创新图像生成。体验前沿的AI技术,推动个人和企业的创新发展。

Project Cover

SubCat字幕猫

SubCat字幕猫APP是一款创新的视频播放器,它将改变您观看视频的方式!SubCat结合了先进的人工智能技术,为您提供即时视频字幕翻译,无论是本地视频还是网络流媒体,让您轻松享受各种语言的内容。

Project Cover

美间AI

美间AI创意设计平台,利用前沿AI技术,为设计师和营销人员提供一站式设计解决方案。从智能海报到3D效果图,再到文案生成,美间让创意设计更简单、更高效。

Project Cover

AIWritePaper论文写作

AIWritePaper论文写作是一站式AI论文写作辅助工具,简化了选题、文献检索至论文撰写的整个过程。通过简单设定,平台可快速生成高质量论文大纲和全文,配合图表、参考文献等一应俱全,同时提供开题报告和答辩PPT等增值服务,保障数据安全,有效提升写作效率和论文质量。

投诉举报邮箱: service@vectorlightyear.com
@2024 懂AI·鲁ICP备2024100362号-6·鲁公网安备37021002001498号