当前位置: > > > > 如何模拟 *exec.Cmd / exec.Command()?
如何模拟 *exec.Cmd / exec.Command()?
来源:stackoverflow
2024-04-25 10:24:35
0浏览
收藏
积累知识,胜过积蓄金银!毕竟在Golang开发的过程中,会遇到各种各样的问题,往往都是一些细节知识点还没有掌握好而导致的,因此基础知识点的积累是很重要的。下面本文《如何模拟 *exec.Cmd / exec.Command()?》,就带大家讲解一下知识点,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~
问题内容
我需要模拟 exec.command()。
我可以使用以下方式模拟它:
var rname string
var rargs []string
mockexeccommand := func(name string, arg ...string) *exec.cmd {
rname = name
rargs = arg
return nil
}
但是,这在实际代码中不起作用,因为它会抱怨 nil 指针,因为返回的 exec.cmd 调用 run()。
我试图像这样嘲笑它:
type mock exec.cmd
func (m *mock) run() error {
return nil
}
var rname string
var rargs []string
mockexeccommand := func(name string, arg ...string) *exec.cmd {
rname = name
rargs = arg
m := mock{}
return &m
}
但它抱怨:不能使用 &m (*mock 类型的值)作为 return 语句中的 *exec.cmd 值compilerincompleteassign。
有什么办法可以解决这个问题吗?有没有更好的方法来模拟 exec.command()?
如果我返回“模拟”命令,模拟函数就会起作用,尽管我也更喜欢控制 run() 函数:
var rName string
var rArgs []string
mockExecCommand := func(name string, arg ...string) *exec.Cmd {
rName = name
rArgs = arg
return exec.Command("echo")
}
正确答案
实际上有一种方法可以做到这一点。所有功劳均归于 文章。请查看以下内容的解释:
func fakeExecCommand(command string, args...string) *exec.Cmd {
cs := []string{"-test.run=TestHelperProcess", "--", command}
cs = append(cs, args...)
cmd := exec.Command(os.Args[0], cs...)
cmd.Env = []string{"GO_WANT_HELPER_PROCESS=1"}
return cmd
}
func TestHelperProcess(t *testing.T){
if os.Getenv("GO_WANT_HELPER_PROCESS") != "1" {
return
}
os.Exit(0)
}
虽然劫持测试可执行文件以运行特定函数是有效的,但仅使用常规依赖项注入会更直接。不需要魔法。
设计一个可以运行命令的接口(例如 commandexecutor),然后将其中之一作为运行命令所需的任何函数的输入。然后,您可以在测试期间提供满足接口的模拟实现(手工制作或使用您选择的工具(如 gomock)生成)。为您的生产代码提供真正的实现(调用 exec 包)。您的模拟实现甚至可以对参数进行断言,以便您知道命令正在正确“执行”。
到这里,我们也就讲完了《如何模拟 *exec.Cmd / exec.Command()?》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注米云公众号,带你了解更多关于的知识点!
