205 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			205 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Go
		
	
	
	
| package helmexec
 | |
| 
 | |
| import (
 | |
| 	"io"
 | |
| 	"io/ioutil"
 | |
| 	"os"
 | |
| 	"strings"
 | |
| 
 | |
| 	"sync"
 | |
| 
 | |
| 	"go.uber.org/zap"
 | |
| 	"go.uber.org/zap/zapcore"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	command = "helm"
 | |
| )
 | |
| 
 | |
| type execer struct {
 | |
| 	helmBinary      string
 | |
| 	runner          Runner
 | |
| 	logger          *zap.SugaredLogger
 | |
| 	kubeContext     string
 | |
| 	extra           []string
 | |
| 	decryptionMutex sync.Mutex
 | |
| }
 | |
| 
 | |
| func NewLogger(writer io.Writer, logLevel string) *zap.SugaredLogger {
 | |
| 	var cfg zapcore.EncoderConfig
 | |
| 	cfg.MessageKey = "message"
 | |
| 	out := zapcore.AddSync(writer)
 | |
| 	var level zapcore.Level
 | |
| 	err := level.Set(logLevel)
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 	core := zapcore.NewCore(
 | |
| 		zapcore.NewConsoleEncoder(cfg),
 | |
| 		out,
 | |
| 		level,
 | |
| 	)
 | |
| 	return zap.New(core).Sugar()
 | |
| }
 | |
| 
 | |
| // New for running helm commands
 | |
| func New(logger *zap.SugaredLogger, kubeContext string) *execer {
 | |
| 	return &execer{
 | |
| 		helmBinary:  command,
 | |
| 		logger:      logger,
 | |
| 		kubeContext: kubeContext,
 | |
| 		runner:      &ShellRunner{},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (helm *execer) SetExtraArgs(args ...string) {
 | |
| 	helm.extra = args
 | |
| }
 | |
| 
 | |
| func (helm *execer) SetHelmBinary(bin string) {
 | |
| 	helm.helmBinary = bin
 | |
| }
 | |
| 
 | |
| func (helm *execer) AddRepo(name, repository, certfile, keyfile, username, password string) error {
 | |
| 	var args []string
 | |
| 	args = append(args, "repo", "add", name, repository)
 | |
| 	if certfile != "" && keyfile != "" {
 | |
| 		args = append(args, "--cert-file", certfile, "--key-file", keyfile)
 | |
| 	}
 | |
| 	if username != "" && password != "" {
 | |
| 		args = append(args, "--username", username, "--password", password)
 | |
| 	}
 | |
| 	helm.logger.Infof("Adding repo %v %v", name, repository)
 | |
| 	out, err := helm.exec(args...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) UpdateRepo() error {
 | |
| 	helm.logger.Info("Updating repo")
 | |
| 	out, err := helm.exec("repo", "update")
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) UpdateDeps(chart string) error {
 | |
| 	helm.logger.Infof("Updating dependency %v", chart)
 | |
| 	out, err := helm.exec("dependency", "update", chart)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) SyncRelease(name, chart string, flags ...string) error {
 | |
| 	helm.logger.Infof("Upgrading %v", chart)
 | |
| 	out, err := helm.exec(append([]string{"upgrade", "--install", "--reset-values", name, chart}, flags...)...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) ReleaseStatus(name string) error {
 | |
| 	helm.logger.Infof("Getting status %v", name)
 | |
| 	out, err := helm.exec(append([]string{"status", name})...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) DecryptSecret(name string) (string, error) {
 | |
| 	// Prevents https://github.com/roboll/helmfile/issues/258
 | |
| 	helm.decryptionMutex.Lock()
 | |
| 	defer helm.decryptionMutex.Unlock()
 | |
| 
 | |
| 	helm.logger.Infof("Decrypting secret %v", name)
 | |
| 	out, err := helm.exec(append([]string{"secrets", "dec", name})...)
 | |
| 	helm.write(out)
 | |
| 
 | |
| 	tmpFile, err := ioutil.TempFile("", "secret")
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	defer tmpFile.Close()
 | |
| 
 | |
| 	// os.Rename seems to results in "cross-device link` errors in some cases
 | |
| 	// Instead of moving, copy it to the destination temp file as a work-around
 | |
| 	// See https://github.com/roboll/helmfile/issues/251#issuecomment-417166296f
 | |
| 	decFilename := name + ".dec"
 | |
| 	decFile, err := os.Open(decFilename)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	defer decFile.Close()
 | |
| 
 | |
| 	_, err = io.Copy(tmpFile, decFile)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	if err := decFile.Close(); err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	if err := os.Remove(decFilename); err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	return tmpFile.Name(), err
 | |
| }
 | |
| 
 | |
| func (helm *execer) TemplateRelease(chart string, flags ...string) error {
 | |
| 	out, err := helm.exec(append([]string{"template", chart}, flags...)...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) DiffRelease(name, chart string, flags ...string) error {
 | |
| 	helm.logger.Infof("Comparing %v %v", name, chart)
 | |
| 	out, err := helm.exec(append([]string{"diff", "upgrade", "--allow-unreleased", name, chart}, flags...)...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) Lint(chart string, flags ...string) error {
 | |
| 	helm.logger.Infof("Linting %v", chart)
 | |
| 	out, err := helm.exec(append([]string{"lint", chart}, flags...)...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) Fetch(chart string, flags ...string) error {
 | |
| 	helm.logger.Infof("Fetching %v", chart)
 | |
| 	out, err := helm.exec(append([]string{"fetch", chart}, flags...)...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) DeleteRelease(name string, flags ...string) error {
 | |
| 	helm.logger.Infof("Deleting %v", name)
 | |
| 	out, err := helm.exec(append([]string{"delete", name}, flags...)...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) TestRelease(name string, flags ...string) error {
 | |
| 	helm.logger.Infof("Testing %v", name)
 | |
| 	out, err := helm.exec(append([]string{"test", name}, flags...)...)
 | |
| 	helm.write(out)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (helm *execer) exec(args ...string) ([]byte, error) {
 | |
| 	cmdargs := args
 | |
| 	if len(helm.extra) > 0 {
 | |
| 		cmdargs = append(cmdargs, helm.extra...)
 | |
| 	}
 | |
| 	if helm.kubeContext != "" {
 | |
| 		cmdargs = append(cmdargs, "--kube-context", helm.kubeContext)
 | |
| 	}
 | |
| 	helm.logger.Debugf("exec: %s %s", helm.helmBinary, strings.Join(cmdargs, " "))
 | |
| 	return helm.runner.Execute(helm.helmBinary, cmdargs)
 | |
| }
 | |
| 
 | |
| func (helm *execer) write(out []byte) {
 | |
| 	if len(out) > 0 {
 | |
| 		helm.logger.Infof("%s", out)
 | |
| 	}
 | |
| }
 |