-
Notifications
You must be signed in to change notification settings - Fork 437
/
run_error.go
70 lines (59 loc) · 1.48 KB
/
run_error.go
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
package cmdutils
import (
"fmt"
"strconv"
"strings"
)
// RunError represents an error running a Cmd
type RunError struct {
command []string // [Name Args...]
output []byte // Captured Stdout / Stderr of the command
inner error // Underlying error if any
stackTrace error
}
var _ error = &RunError{}
func (e *RunError) Error() string {
if e == nil {
return ""
}
return fmt.Sprintf("command \"%s\" failed with error: %v", e.PrettyCommand(), e.inner)
}
// PrettyCommand pretty prints the command in a way that could be pasted
// into a shell
func (e *RunError) PrettyCommand() string {
if e == nil {
return "RunError is nil"
}
if len(e.command) == 0 {
return "no command args"
}
if len(e.command) == 1 {
return e.command[0]
}
// The above cases should not happen, but we defend against it
return PrettyCommand(e.command[0], e.command[1:]...)
}
func (e *RunError) OutputString() string {
if e == nil {
return ""
}
return string(e.output)
}
// Cause mimics github.com/pkg/errors's Cause pattern for errors
func (e *RunError) Cause() error {
if e == nil {
return nil
}
return e.stackTrace
}
// PrettyCommand takes arguments identical to Cmder.Command,
// it returns a pretty printed command that could be pasted into a shell
func PrettyCommand(name string, args ...string) string {
var out strings.Builder
out.WriteString(strconv.Quote(name))
for _, arg := range args {
out.WriteByte(' ')
out.WriteString(strconv.Quote(arg))
}
return out.String()
}