package util
import (
"bytes"
"io"
"os/exec"
)
type PipeCommand struct {
stack []*exec.Cmd
finalStdout, finalStderr io.Reader
pipestack []*io.PipeWriter
}
func NewPipeCommand(stack ...*exec.Cmd) (*PipeCommand, error) {
var errorbuffer bytes.Buffer
pipestack := make([]*io.PipeWriter, len(stack)-1)
i := 0
for ; i < len(stack)-1; i++ {
stdinpipe, stdoutpipe := io.Pipe()
stack[i].Stdout = stdoutpipe
stack[i].Stderr = &errorbuffer
stack[i+1].Stdin = stdinpipe
pipestack[i] = stdoutpipe
}
finalStdout, err := stack[i].StdoutPipe()
if err != nil {
return nil, err
}
finalStderr, err := stack[i].StderrPipe()
if err != nil {
return nil, err
}
pipeCommand := &PipeCommand{
stack: stack,
pipestack: pipestack,
finalStdout: finalStdout,
finalStderr: finalStderr,
}
return pipeCommand, nil
}
func (p *PipeCommand) Run() error {
return call(p.stack, p.pipestack)
}
func (p *PipeCommand) GetFinalStdout() io.Reader {
return p.finalStdout
}
func (p *PipeCommand) GetFinalStderr() io.Reader {
return p.finalStderr
}
func call(stack []*exec.Cmd, pipes []*io.PipeWriter) (err error) {
if stack[0].Process == nil {
if err = stack[0].Start(); err != nil {
return err
}
}
if len(stack) > 1 {
if err = stack[1].Start(); err != nil {
return err
}
defer func() {
if err == nil {
pipes[0].Close()
err = call(stack[1:], pipes[1:])
}
}()
}
return stack[0].Wait()
}