This also provides a SetPrepareCmd hook for tests to be able to define stubs for commands that are supposed to be run
59 lines
1.3 KiB
Go
59 lines
1.3 KiB
Go
package utils
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"os/exec"
|
|
"strings"
|
|
)
|
|
|
|
// Runnable is typically an exec.Cmd or its stub in tests
|
|
type Runnable interface {
|
|
Output() ([]byte, error)
|
|
Run() error
|
|
}
|
|
|
|
// PrepareCmd extends exec.Cmd with extra error reporting features and provides a
|
|
// hook to stub command execution in tests
|
|
var PrepareCmd = func(cmd *exec.Cmd) Runnable {
|
|
return &cmdWithStderr{cmd}
|
|
}
|
|
|
|
// SetPrepareCmd overrides PrepareCmd and returns a func to revert it back
|
|
func SetPrepareCmd(fn func(*exec.Cmd) Runnable) func() {
|
|
origPrepare := PrepareCmd
|
|
PrepareCmd = fn
|
|
return func() {
|
|
PrepareCmd = origPrepare
|
|
}
|
|
}
|
|
|
|
// cmdWithStderr augments Output() by adding stderr to the error message
|
|
type cmdWithStderr struct {
|
|
*exec.Cmd
|
|
}
|
|
|
|
func (c cmdWithStderr) Output() ([]byte, error) {
|
|
errStream := &bytes.Buffer{}
|
|
c.Cmd.Stderr = errStream
|
|
out, err := c.Cmd.Output()
|
|
if err != nil {
|
|
err = &CmdError{errStream, c.Cmd.Args, err}
|
|
}
|
|
return out, err
|
|
}
|
|
|
|
// CmdError provides more visibility into why an exec.Cmd had failed
|
|
type CmdError struct {
|
|
Stderr *bytes.Buffer
|
|
Args []string
|
|
Err error
|
|
}
|
|
|
|
func (e CmdError) Error() string {
|
|
msg := e.Stderr.String()
|
|
if msg != "" && !strings.HasSuffix(msg, "\n") {
|
|
msg += "\n"
|
|
}
|
|
return fmt.Sprintf("%s%s: %s", msg, e.Args[0], e.Err)
|
|
}
|