summaryrefslogtreecommitdiff
path: root/util/process.go
blob: 6a6d17d4ad0309597a9f5aa34b15a27e55978b73 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
// Copyright (C) 2019-2023 Algorand, Inc.
// This file is part of go-algorand
//
// go-algorand is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// go-algorand is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with go-algorand.  If not, see <https://www.gnu.org/licenses/>.

package util

import (
	"io"
	"os"
	"os/exec"
	"sync"
	"time"
)

// ExecAndCaptureOutput runs the specified command and args and captures
// stdout into a string, returning the string or an error upon completion.
func ExecAndCaptureOutput(command string, args ...string) (string, string, error) {
	rStdout, wStdout, err := os.Pipe()
	if err != nil {
		return "", "", err
	}
	rStderr, wStderr, err := os.Pipe()
	if err != nil {
		return "", "", err
	}

	subcmd := exec.Command(command, args...)
	subcmd.Stdout = wStdout
	subcmd.Stderr = wStderr

	outputStdout := make([]byte, 0, 10240)
	outputStderr := make([]byte, 0, 1024)

	var wg sync.WaitGroup
	wg.Add(2)

	reader := func(input *os.File, output *[]byte) {
		defer wg.Done()

		for {
			buf := make([]byte, 1024)
			read, e := input.Read(buf)
			if e == io.EOF {
				break
			}
			if read > 0 {
				*output = append(*output, buf[0:read]...)
			} else {
				time.Sleep(time.Microsecond)
			}
		}
	}
	go reader(rStdout, &outputStdout)
	go reader(rStderr, &outputStderr)

	err = subcmd.Run()
	wStdout.Close()
	wStderr.Close()

	wg.Wait()

	return string(outputStdout), string(outputStderr), err
}