Получить код выхода - Go

Я использую пакет: os / exechttp://golang.org/pkg/os/exec/ выполнить команду в операционной системе, но я, похоже, не нашел способ получить код выхода. Я могу прочитать вывод хотя

то есть.

<code>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() )
}
</code>

Ответы на вопрос(3)

код выхода доступен изначально и в кросс-платформенном режиме.Смотрите ExitError а такжеExitCode ().

ExitCode returns the exit code of the exited process, or -1 if the process hasn't exited or was terminated by a signal.

if err := cmd.Run() ; err != nil {
    if exitError, ok := err.(*exec.ExitError); ok {
        return exitError.ExitCode()
    }
}

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, если он не работает должным образом на других платформах, пожалуйста, сообщите мне, чтобы мы могли сделать его лучше.

 08 сент. 2017 г., 00:54
Люблю эту функцию
 08 февр. 2017 г., 00:38
Работает на Ubuntu 16.04 Linux и возвращает стандартный вывод вместе с exitCode - именно то, что я искал ...
Решение Вопроса

был ли код выхода 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")
    }

    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)
        }
    }
}

Простоследовать апи документы узнать больше :)

 OscarRyz30 апр. 2012 г., 17:18
Список рассылки кое-что говорит о конкретной Windows, но я только что запустил этот код и работает отлично, это на ходу1.
 30 апр. 2012 г., 17:16
Определенно, получить код выхода не так-то просто, но в документах есть, по крайней мере, некоторые подсказки, на которые вам следует обратить внимание. Но они вообще не упоминают Windows, поэтому я с нетерпением жду вашего ответа.
 30 апр. 2012 г., 17:23
@OscarRyz ОтProcessState вWaitStatus: вы можете использоватьfmt.Printf("%T\n", exiterr.Sys()) распечатать тип значения, или вы можете проверить источники вgolang.org/src/pkg.
 OscarRyz30 апр. 2012 г., 17:56
Atom и tux21b спасибо вам обоим :) Я прочитал документы, но застрял в таких местах, какConvert it to the appropriate underlying type : P Итак, моя стратегия былаAsk on StackOverflow и работал \ о /
 OscarRyz30 апр. 2012 г., 17:12
Я заглянул в документы, но так и не нашел путиProcessState вWaitStatus Как ты это делаешь?. Кроме того, я только что нашел в списке рассылки решение (точно так же, как вы описали)groups.google.com/forum/?fromgroups#!searchin/golang-nuts/…  Я пытаюсь сделать это сейчас в Windows и сообщу вам :)

Ваш ответ на вопрос