使用 <(..) 参数执行

标签 go exec

我正在尝试使用 exec 获得等价物

diff <(echo "foo") <(echo "bar")

产生:

1c1
< foo
---
> bar

但是当我尝试时:

cmd := exec.Command("diff", "<(echo foo)", "<(echo bar)")

输出是:

exit status 2
diff: <(echo foo): No such file or directory
diff: <(echo bar): No such file or directory

最佳答案

FUZxxl 的回答很简单,但需要使用 bash。 Bash 并不是真正的跨平台,作为一个解释器,它对代码注入(inject)是开放的。我将首先描述一种使用 Go 复制它的方法,然后描述 git 使用的答案。

<(echo foo) 在 bash 中做了两件事。首先,它创建一个文件描述符以传递给被调用的命令。它做的第二件事是将/dev/fd/n 作为参数传递。为了确认,请尝试:

echo <(echo foo) <(echo bar)

Echo 忽略 fds,只将文件描述符打印为字符串。

下面的 Go 代码会做类似的事情:

package main

import (
    "io"
    "log"
    "os"
    "os/exec"
    "strings"
)

func main() {
    cmd := exec.Command("diff", "/dev/fd/3", "/dev/fd/4")

    foo, err := readerFile(strings.NewReader("foo\n"))
    if err != nil {
        log.Fatal(err)
    }
    defer foo.Close()

    bar, err := readerFile(strings.NewReader("bar\n"))
    if err != nil {
        log.Fatal(err)
    }
    defer bar.Close()

    cmd.ExtraFiles = []*os.File{foo, bar}

    cmd.Stdout = os.Stdout
    cmd.Stderr = os.Stderr
    err = cmd.Run()
    if err != nil {
        log.Fatal(err)
    }
}

func readerFile(r io.Reader) (*os.File, error) {
    reader, writer, err := os.Pipe()

    if err != nil {
        return nil, err
    }

    go func() {
        io.Copy(writer, r)
        writer.Close()
    }()

    return reader, nil
}

显然,这只能在类 Unix 系统上运行,因此不是很跨平台。不过,它不受代码注入(inject)的影响。


GIT 的方式

Git 只是制作临时文件然后比较它们。这是非常跨平台的,并且不受注入(inject)影响。只需使用 ioutil.TempFile创建这两个文件,然后运行 ​​exec.Command("diff", file1, file2)

关于使用 <(..) 参数执行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29022974/

相关文章:

google-app-engine - Go 应用引擎 dev_appserver.py 不支持运行时 go112

php - exec() PHP 不能等待响应 (phpseclib)

java - 如何使管道与 Runtime.exec() 一起工作?

mysql - 使用golang在mysql中创建数据库

go - 如何使用RWMutex和升级锁

go - 在哪里可以找到gcloud调用的API?

fork 后子进程无法正常工作

c - 从主要参数执行程序

docker - docker attach 和 docker exec 的区别

go - 使用 "go test"列出所有测试用例