使用 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 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号