package helmexec import ( "bytes" "errors" "fmt" "io" "os" "os/exec" "path/filepath" "strings" "syscall" "go.uber.org/zap" ) const ( tmpPrefix = "helmfile-" tmpSuffix = "-exec" ) // Runner interface for shell commands type Runner interface { Execute(cmd string, args []string, env map[string]string) ([]byte, error) ExecuteStdIn(cmd string, args []string, env map[string]string, stdin io.Reader) ([]byte, error) } // ShellRunner implemention for shell commands type ShellRunner struct { Dir string Logger *zap.SugaredLogger } // Execute a shell command func (shell ShellRunner) Execute(cmd string, args []string, env map[string]string) ([]byte, error) { preparedCmd := exec.Command(cmd, args...) preparedCmd.Dir = shell.Dir preparedCmd.Env = mergeEnv(os.Environ(), env) return Output(preparedCmd, &logWriterGenerator{ log: shell.Logger, }) } // Execute a shell command func (shell ShellRunner) ExecuteStdIn(cmd string, args []string, env map[string]string, stdin io.Reader) ([]byte, error) { preparedCmd := exec.Command(cmd, args...) preparedCmd.Dir = shell.Dir preparedCmd.Env = mergeEnv(os.Environ(), env) preparedCmd.Stdin = stdin return Output(preparedCmd, &logWriterGenerator{ log: shell.Logger, }) } func Output(c *exec.Cmd, logWriterGenerators ...*logWriterGenerator) ([]byte, error) { if c.Stdout != nil { return nil, errors.New("exec: Stdout already set") } if c.Stderr != nil { return nil, errors.New("exec: Stderr already set") } var stdout bytes.Buffer var stderr bytes.Buffer var combined bytes.Buffer var logWriters []io.Writer id := newExecutionID() for _, g := range logWriterGenerators { logPrefix := fmt.Sprintf("%s:%s> ", filepath.Base(c.Path), id) logWriters = append(logWriters, g.Writer(logPrefix)) } c.Stdout = io.MultiWriter(append([]io.Writer{&stdout, &combined}, logWriters...)...) c.Stderr = io.MultiWriter(append([]io.Writer{&stderr, &combined}, logWriters...)...) err := c.Run() if err != nil { // TrimSpace is necessary, because otherwise helmfile prints the redundant new-lines after each error like: // // err: release "envoy2" in "helmfile.yaml" failed: exit status 1: Error: could not find a ready tiller pod // // err: release "envoy" in "helmfile.yaml" failed: exit status 1: Error: could not find a ready tiller pod switch ee := err.(type) { case *exec.ExitError: // Propagate any non-zero exit status from the external command, rather than throwing it away, // so that helmfile could return its own exit code accordingly waitStatus := ee.Sys().(syscall.WaitStatus) exitStatus := waitStatus.ExitStatus() err = newExitError(c.Path, c.Args, exitStatus, ee, stderr.String(), combined.String()) default: panic(fmt.Sprintf("unexpected error: %v", err)) } } return stdout.Bytes(), err } func mergeEnv(orig []string, new map[string]string) []string { wanted := env2map(orig) for k, v := range new { wanted[k] = v } return map2env(wanted) } func map2env(wanted map[string]string) []string { result := []string{} for k, v := range wanted { result = append(result, k+"="+v) } return result } func env2map(env []string) map[string]string { wanted := map[string]string{} for _, cur := range env { pair := strings.SplitN(cur, "=", 2) var v string // An environment can completely miss `=` and the right side. // If we didn't deal with that, this may fail due to an index-out-of-range error if len(pair) > 1 { v = pair[1] } wanted[pair[0]] = v } return wanted }