1

我试图在 Go 代码中使用fzf 。我参考了作者这里给出的例子。当我尝试为该功能编写测试时,它会因为fzf需要交互式输入而卡住。

代码:

func withFilter(command string, input func(in io.WriteCloser)) []string {
    shell := os.Getenv("SHELL")
    if len(shell) == 0 {
        shell = "sh"
    }
    cmd := exec.Command(shell, "-c", command)
    cmd.Stderr = os.Stderr
    in, _ := cmd.StdinPipe()
    go func() {
        input(in)
        in.Close()
    }()
    result, _ := cmd.Output()
    return strings.Split(string(result), "\n")
}

func filter() []string {
    filtered := withFilter("fzf -m", func(in io.WriteCloser) {
        for i := 0; i < 10; i++ {
            fmt.Fprintln(in, i)
            time.Sleep(5 * time.Millisecond)
        }
    })

    return filtered
}

测试:

func TestFilter(t *testing.T) {
    assert.Equal(t, []string{"1", "2", "3"}, filter())
}

我尝试调试并注意到它卡在cmd.Output(). 通过深入挖掘,该命令似乎无限期地等待输入,但是我不确定如何以编程方式提供它。我试着写信\nos.Stdin但没有用。

任何指针或解释将不胜感激。谢谢。

4

1 回答 1

1

Go 的美妙之处:它的io.Reader/io.Writer接口。

您必须提供您的withFilter(以及扩展filter)方法应该从哪里读取。所以,in必须提供。

// example

func readFrom(reader io.Reader) {
    r := bufio.NewScanner(reader)

    for r.Scan() {
        fmt.Println("got", r.Text())
    }

    fmt.Println("Done.")
}

func main() {
    const input = "Now is the winter of our discontent,\nMade glorious summer by this sun of York.\n"

    readFrom(strings.NewReader(input))

    // okay, with stdin
    readFrom(os.Stdin)

}

如您所见,第一个readFrom是完全可测试的,您只需将input变量更改为您想要测试的任何内容。第二个readFrom只会在stdin关闭时返回。

解决方案是写入os.Stdin. 考虑os.Stdin你的操作系统中的实现细节——而不是你必须在测试中包含的东西。

https://play.golang.org/p/D2wzHYrV2TM (os.Stdin 在操场上关闭)

于 2019-02-01T06:09:31.813 回答