使用 expect 库测试 cli 交互流程的关键步骤是:1. 安装 expect 及其依赖;2. 编写或构建被测 cli 程序;3. 启动子进程并创建 expect 实例;4. 使用 expectstring 匹配提示信息;5. 使用 sendline 模拟用户输入;6. 再次使用 expectstring 验证输出结果;7. 延迟关闭 expect 实例以释放资源;同时需要注意缓冲、换行符、路径以及并发测试等问题以确保测试准确性和稳定性。
在写 CLI 工具时,测试交互流程是个让人头疼的问题。因为这类程序往往依赖用户的输入、命令行参数甚至多次交互才能完成任务。Golang 虽然没有内置的模拟终端机制,但借助
github.com/google/expect
首先需要引入 expect 包,它最初是为 Go 的测试用例设计的,支持与子进程通信并模拟输入输出。安装方式如下:
go get github.com/google/expect
这个包依赖
golang.org/x/crypto/ssh/terminal
立即学习“go语言免费学习笔记(深入)”;
为了演示方便,我们先写一个非常基础的 CLI 程序:它会提示用户输入名字,并输出欢迎语句。
// cli.go package main import ( "bufio" "fmt" "os" ) func main() { reader := bufio.NewReader(os.Stdin) fmt.Print("请输入你的名字: ") name, _ := reader.ReadString('\n') fmt.Printf("你好, %s\n", name) }
保存为
cli.go
mycli
接下来我们写一个测试文件来验证这个 CLI 是否正确处理了用户输入。
// cli_test.go package main import ( "os/exec" "runtime" "testing" "time" "github.com/google/expect" ) func TestCLIInteraction(t *testing.T) { // 构建要运行的命令 binPath := "./mycli" if runtime.GOOS == "windows" { binPath += ".exe" } // 启动子进程 cmd := exec.Command(binPath) exp, err := expect.NewExpect(cmd, time.Second) if err != nil { t.Fatal(err) } defer exp.Close() // 匹配提示信息 _, err = exp.ExpectString("请输入你的名字: ") if err != nil { t.Errorf("未匹配到提示语: %v", err) } // 模拟用户输入 err = exp.SendLine("Tom") if err != nil { t.Errorf("发送输入失败: %v", err) } // 验证输出结果 _, err = exp.ExpectString("你好, Tom") if err != nil { t.Errorf("未匹配到预期输出: %v", err) } }
exec.Command
注意:expect 默认有超时机制,如果你的程序执行较慢,可能需要适当调高超时时间。
在实际使用中可能会遇到以下几种情况:
fflush(stdout)
fmt.Fprint(os.Stdout, ...)
Println
SendLine()
.exe
基本上就这些。用 expect 来测试 CLI 交互流程虽然不是最直观的方式,但在 Golang 生态中已经算是比较成熟且实用的方案了。只要注意匹配时机和输出刷新逻辑,大多数场景都能覆盖。
以上就是Golang如何测试CLI交互流程 使用expect库模拟用户输入场景的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 //m.sbmmt.com/ All Rights Reserved | php.cn | 湘ICP备2023035733号