Go by Example: Spawning Processes

ときには、私たちの Go プログラムは他の Go でないプロセスを生成する必要があります。

package main
import (
    "fmt"
    "io"
    "os/exec"
)
func main() {

まずは引数なし、つまり入力を受け取らずに、 標準出力へ何か表示するだけの単純なコマンドから始めましょう。 exec.Command ヘルパーは、 この外部プロセスを表すオブジェクトを作成します。

    dateCmd := exec.Command("date")

Output メソッドは、コマンドを実行して終了を待ち、 その標準出力を取得します。エラーがなければ、 dateOut は日付情報のバイト列を保持します。

    dateOut, err := dateCmd.Output()
    if err != nil {
        panic(err)
    }
    fmt.Println("> date")
    fmt.Println(string(dateOut))

Output メソッドや Command のその他のメソッドは、 コマンドを実行するのに問題(例. パス誤り)が見つかると *exec.Error を返します。また、コマンドは実行できても 終了コードが非ゼロだった場合は、*exec.ExitError を返します。

    _, err = exec.Command("date", "-x").Output()
    if err != nil {
        switch e := err.(type) {
        case *exec.Error:
            fmt.Println("failed executing:", err)
        case *exec.ExitError:
            fmt.Println("command exit rc =", e.ExitCode())
        default:
            panic(err)
        }
    }

次に、もう少し複雑なケースとして、 パイプを使って外部プロセスの stdin にデータを与え、 結果を stdout から取得する例を見てみましょう。

    grepCmd := exec.Command("grep", "hello")

次の例では、標準入力/出力のパイプを明示的につかみ、 プロセスを開始します。入力データを書き込み、出力を読み込み、 そして最後にプロセスが終了するのを待ちます。

    grepIn, _ := grepCmd.StdinPipe()
    grepOut, _ := grepCmd.StdoutPipe()
    grepCmd.Start()
    grepIn.Write([]byte("hello grep\ngoodbye grep"))
    grepIn.Close()
    grepBytes, _ := io.ReadAll(grepOut)
    grepCmd.Wait()

上の例ではエラーチェックを省略しましたが、いつもの if err != nil パターンを使うことができます。 また、上の例では StdoutPipe の結果だけを取得しましたが、 全く同じ方法で StderrPipe からも取得できます。

    fmt.Println("> grep hello")
    fmt.Println(string(grepBytes))

コマンドを起動するときは、1 つのコマンドライン文字列を 渡せるのではなく、コマンドと引数の配列を明示的に 渡す必要がある点に注意してください。 文字列でコマンドを起動したいなら、 bash-c オプションを使うことができます。

    lsCmd := exec.Command("bash", "-c", "ls -a -l -h")
    lsOut, err := lsCmd.Output()
    if err != nil {
        panic(err)
    }
    fmt.Println("> ls -a -l -h")
    fmt.Println(string(lsOut))
}

起動されたプログラムは、それらをコマンドラインから 直接実行したときと同じ出力を返します。

$ go run spawning-processes.go
> date
Thu 05 May 2022 10:10:12 PM PDT

date コマンドは -x フラグをもたないため、 エラーメッセージと非ゼロのエラーコードで終了します。

command exited with rc = 1
> grep hello
hello grep
> ls -a -l -h
drwxr-xr-x  4 mark 136B Oct 3 16:29 .
drwxr-xr-x 91 mark 3.0K Oct 3 12:50 ..
-rw-r--r--  1 mark 1.3K Oct 3 16:28 spawning-processes.go

Next example: Exec'ing Processes.