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 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157
|
/*
Copyright 2018 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package exec
import (
"bytes"
"context"
"io"
osexec "os/exec"
"sync"
"sigs.k8s.io/kind/pkg/errors"
)
// LocalCmd wraps os/exec.Cmd, implementing the kind/pkg/exec.Cmd interface
type LocalCmd struct {
*osexec.Cmd
}
var _ Cmd = &LocalCmd{}
// LocalCmder is a factory for LocalCmd, implementing Cmder
type LocalCmder struct{}
var _ Cmder = &LocalCmder{}
// Command returns a new exec.Cmd backed by Cmd
func (c *LocalCmder) Command(name string, arg ...string) Cmd {
return &LocalCmd{
Cmd: osexec.Command(name, arg...),
}
}
// CommandContext is like Command but includes a context
func (c *LocalCmder) CommandContext(ctx context.Context, name string, arg ...string) Cmd {
return &LocalCmd{
Cmd: osexec.CommandContext(ctx, name, arg...),
}
}
// SetEnv sets env
func (cmd *LocalCmd) SetEnv(env ...string) Cmd {
cmd.Env = env
return cmd
}
// SetStdin sets stdin
func (cmd *LocalCmd) SetStdin(r io.Reader) Cmd {
cmd.Stdin = r
return cmd
}
// SetStdout set stdout
func (cmd *LocalCmd) SetStdout(w io.Writer) Cmd {
cmd.Stdout = w
return cmd
}
// SetStderr sets stderr
func (cmd *LocalCmd) SetStderr(w io.Writer) Cmd {
cmd.Stderr = w
return cmd
}
// Run runs the command
// If the returned error is non-nil, it should be of type *RunError
func (cmd *LocalCmd) Run() error {
// Background:
// Go's stdlib will setup and use a shared fd when cmd.Stderr == cmd.Stdout
// In any other case, it will use different fds, which will involve
// two different io.Copy goroutines writing to cmd.Stderr and cmd.Stdout
//
// Given this, we must synchronize capturing the output to a buffer
// IFF ! interfaceEqual(cmd.Sterr, cmd.Stdout)
var combinedOutput bytes.Buffer
var combinedOutputWriter io.Writer = &combinedOutput
if cmd.Stdout == nil && cmd.Stderr == nil {
// Case 1: If stdout and stderr are nil, we can just use the buffer
// The buffer will be == and Go will use one fd / goroutine
cmd.Stdout = combinedOutputWriter
cmd.Stderr = combinedOutputWriter
} else if interfaceEqual(cmd.Stdout, cmd.Stderr) {
// Case 2: If cmd.Stdout == cmd.Stderr go will still share the fd,
// but we need to wrap with a MultiWriter to respect the other writer
// and our buffer.
// The MultiWriter will be == and Go will use one fd / goroutine
cmd.Stdout = io.MultiWriter(cmd.Stdout, combinedOutputWriter)
cmd.Stderr = cmd.Stdout
} else {
// Case 3: If cmd.Stdout != cmd.Stderr, we need to synchronize the
// combined output writer.
// Go will use different fds / write routines for stdout and stderr
combinedOutputWriter = &mutexWriter{
writer: &combinedOutput,
}
// wrap writers if non-nil
if cmd.Stdout != nil {
cmd.Stdout = io.MultiWriter(cmd.Stdout, combinedOutputWriter)
} else {
cmd.Stdout = combinedOutputWriter
}
if cmd.Stderr != nil {
cmd.Stderr = io.MultiWriter(cmd.Stderr, combinedOutputWriter)
} else {
cmd.Stderr = combinedOutputWriter
}
}
// TODO: should be in the caller or logger should be injected somehow ...
if err := cmd.Cmd.Run(); err != nil {
return errors.WithStack(&RunError{
Command: cmd.Args,
Output: combinedOutput.Bytes(),
Inner: err,
})
}
return nil
}
// interfaceEqual protects against panics from doing equality tests on
// two interfaces with non-comparable underlying types.
// This trivial is borrowed from the go stdlib in os/exec
// Note that the recover will only happen if a is not comparable to b,
// in which case we'll return false
// We've lightly modified this to pass errcheck (explicitly ignoring recover)
func interfaceEqual(a, b interface{}) bool {
defer func() {
_ = recover()
}()
return a == b
}
// mutexWriter is a simple synchronized wrapper around an io.Writer
type mutexWriter struct {
writer io.Writer
mu sync.Mutex
}
func (m *mutexWriter) Write(b []byte) (int, error) {
m.mu.Lock()
defer m.mu.Unlock()
n, err := m.writer.Write(b)
return n, err
}
|