Получить код выхода - Go
Я использую пакет: os/exec http://golang.org/pkg/os/exec/ для выполнения команды в операционной системе, но, похоже, я не нашел способ получить код выхода. Я могу прочитать результат, хотя
т.
package main
import(
"os/exec"
"bytes"
"fmt"
"log"
)
func main() {
cmd := exec.Command("somecommand", "parameter")
var out bytes.Buffer
cmd.Stdout = &out
if err := cmd.Run() ; err != nil {
//log.Fatal( cmd.ProcessState.Success() )
log.Fatal( err )
}
fmt.Printf("%q\n", out.String() )
}
Ответы
Ответ 1
Легко определить, был ли код выхода 0 или что-то еще. В первом случае cmd.Wait()
вернет ноль (если не возникнет другая ошибка при настройке каналов).
К сожалению, нет независимого от платформы способа получить код выхода в случае ошибки. Это также причина, почему он не является частью API. Следующий фрагмент будет работать с Linux, но я не тестировал его на других платформах:
package main
import "os/exec"
import "log"
import "syscall"
func main() {
cmd := exec.Command("git", "blub")
if err := cmd.Start(); err != nil {
log.Fatalf("cmd.Start: %v", err)
}
if err := cmd.Wait(); err != nil {
if exiterr, ok := err.(*exec.ExitError); ok {
// The program has exited with an exit code != 0
// This works on both Unix and Windows. Although package
// syscall is generally platform dependent, WaitStatus is
// defined for both Unix and Windows and in both cases has
// an ExitStatus() method with the same signature.
if status, ok := exiterr.Sys().(syscall.WaitStatus); ok {
log.Printf("Exit Status: %d", status.ExitStatus())
}
} else {
log.Fatalf("cmd.Wait: %v", err)
}
}
}
Просто следуйте документации API docs, чтобы узнать больше :)
Ответ 2
Начиная с версии 1.12 golang, код выхода доступен изначально и в кросс-платформенном режиме. См. ExitError и ExitCode().
ExitCode возвращает код завершения завершенного процесса или -1, если процесс не завершился или был прерван по сигналу.
if err := cmd.Run() ; err != nil {
if exitError, ok := err.(*exec.ExitError); ok {
return exitError.ExitCode()
}
}
Ответ 3
Здесь моя расширенная версия, основанная на @tux21b answer
utils/cmd.go
package utils
import (
"bytes"
"log"
"os/exec"
"syscall"
)
const defaultFailedCode = 1
func RunCommand(name string, args ...string) (stdout string, stderr string, exitCode int) {
log.Println("run command:", name, args)
var outbuf, errbuf bytes.Buffer
cmd := exec.Command(name, args...)
cmd.Stdout = &outbuf
cmd.Stderr = &errbuf
err := cmd.Run()
stdout = outbuf.String()
stderr = errbuf.String()
if err != nil {
// try to get the exit code
if exitError, ok := err.(*exec.ExitError); ok {
ws := exitError.Sys().(syscall.WaitStatus)
exitCode = ws.ExitStatus()
} else {
// This will happen (in OSX) if `name` is not available in $PATH,
// in this situation, exit code could not be get, and stderr will be
// empty string very likely, so we use the default fail code, and format err
// to string and set to stderr
log.Printf("Could not get exit code for failed program: %v, %v", name, args)
exitCode = defaultFailedCode
if stderr == "" {
stderr = err.Error()
}
}
} else {
// success, exitCode should be 0 if go is ok
ws := cmd.ProcessState.Sys().(syscall.WaitStatus)
exitCode = ws.ExitStatus()
}
log.Printf("command result, stdout: %v, stderr: %v, exitCode: %v", stdout, stderr, exitCode)
return
}
Я тестировал его на OSX, если он не работает так, как ожидалось, на других платформах, пожалуйста, скажите мне, чтобы мы могли сделать это лучше.