一般情况下,在 golang
中执行一些命令如 git clone
,则可以使用 exec.Command
函数
func RunCommand(path, name string, arg ...string) (msg string, err error) {
cmd := exec.Command(name, arg...)
cmd.Dir = path
err = cmd.Run()
log.Println(cmd.Args)
if err != nil {
log.Println("err", err.Error(), "cmd", cmd.Args)
}
return
}
这种写法是没有问题,但是一旦执行出错返回值过于简洁了,比如
func main() {
msg, err := common.RunCommand("./", "/bin/bash", "-c", "git clone url")
if err != nil {
log.Fatal(err)
return
}
log.Println(msg)
}
执行后,返回 exit status 128
这种提示,太抽象了,还得专门去搜索引擎查看,若是想要看出更详细的原因还需如此
func RunCommand(path, name string, arg ...string) (msg string, err error) {
cmd := exec.Command(name, arg...)
var out bytes.Buffer
var stderr bytes.Buffer
cmd.Stdout = &out
cmd.Stderr = &stderr
cmd.Dir = path
err = cmd.Run()
log.Println(cmd.Args)
if err != nil {
msg = fmt.Sprint(err) + ": " + stderr.String()
err = errors.New(msg)
log.Println("err", err.Error(), "cmd", cmd.Args)
}
log.Println(out.String())
return
}
再次执行,返回
2022/04/03 20:33:49 [/bin/bash -c git clone url]
2022/04/03 20:33:49 err exit status 128: fatal: repository 'url' does not exist
cmd [/bin/bash -c git clone url]
2022/04/03 20:33:49
2022/04/03 20:33:49 exit status 128: fatal: repository 'url' does not exist
哦,原来是 repository 'url' does not exist
这个原因。
更进一步说下,为啥加了 Stdout
和 Stderr
就能接到值了呢,这是由于有些命令会把错误信息打到 Stdout
,而也有些命令会把错误信息打到 Stderr
,所以就得把两个都收着。
参考:How to debug “exit status 1” error when running exec.Command in Golang