222 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			222 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Go
		
	
	
	
package state
 | 
						|
 | 
						|
import (
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"reflect"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/imdario/mergo"
 | 
						|
 | 
						|
	"github.com/helmfile/helmfile/pkg/tmpl"
 | 
						|
	"github.com/helmfile/helmfile/pkg/yaml"
 | 
						|
)
 | 
						|
 | 
						|
func (st *HelmState) Values() map[string]interface{} {
 | 
						|
	if st.RenderedValues == nil {
 | 
						|
		panic("[bug] RenderedValues is nil")
 | 
						|
	}
 | 
						|
 | 
						|
	return st.RenderedValues
 | 
						|
}
 | 
						|
 | 
						|
func (st *HelmState) createReleaseTemplateData(release *ReleaseSpec, vals map[string]interface{}) releaseTemplateData {
 | 
						|
	tmplData := releaseTemplateData{
 | 
						|
		Environment: st.Env,
 | 
						|
		KubeContext: st.OverrideKubeContext,
 | 
						|
		Namespace:   st.OverrideNamespace,
 | 
						|
		Chart:       st.OverrideChart,
 | 
						|
		Values:      vals,
 | 
						|
		Release: releaseTemplateDataRelease{
 | 
						|
			Name:        release.Name,
 | 
						|
			Chart:       release.Chart,
 | 
						|
			Namespace:   release.Namespace,
 | 
						|
			Labels:      release.Labels,
 | 
						|
			KubeContext: release.KubeContext,
 | 
						|
		},
 | 
						|
	}
 | 
						|
	tmplData.StateValues = &tmplData.Values
 | 
						|
	return tmplData
 | 
						|
}
 | 
						|
 | 
						|
func getBoolRefFromStringTemplate(templateRef string) (*bool, error) {
 | 
						|
	var result bool
 | 
						|
	if err := yaml.Unmarshal([]byte(templateRef), &result); err != nil {
 | 
						|
		return nil, fmt.Errorf("failed deserialising string %s: %v", templateRef, err)
 | 
						|
	}
 | 
						|
	return &result, nil
 | 
						|
}
 | 
						|
 | 
						|
func updateBoolTemplatedValues(r *ReleaseSpec) error {
 | 
						|
	if r.InstalledTemplate != nil {
 | 
						|
		if installed, err := getBoolRefFromStringTemplate(*r.InstalledTemplate); err != nil {
 | 
						|
			return fmt.Errorf("installedTemplate: %v", err)
 | 
						|
		} else {
 | 
						|
			r.InstalledTemplate = nil
 | 
						|
			r.Installed = installed
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if r.WaitTemplate != nil {
 | 
						|
		if wait, err := getBoolRefFromStringTemplate(*r.WaitTemplate); err != nil {
 | 
						|
			return fmt.Errorf("waitTemplate: %v", err)
 | 
						|
		} else {
 | 
						|
			r.WaitTemplate = nil
 | 
						|
			r.Wait = wait
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if r.VerifyTemplate != nil {
 | 
						|
		if verify, err := getBoolRefFromStringTemplate(*r.VerifyTemplate); err != nil {
 | 
						|
			return fmt.Errorf("verifyTemplate: %v", err)
 | 
						|
		} else {
 | 
						|
			r.VerifyTemplate = nil
 | 
						|
			r.Verify = verify
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (st *HelmState) ExecuteTemplates() (*HelmState, error) {
 | 
						|
	r := *st
 | 
						|
 | 
						|
	vals := st.Values()
 | 
						|
 | 
						|
	for i, rt := range st.Releases {
 | 
						|
		release, err := st.releaseWithInheritedTemplate(&rt, nil)
 | 
						|
		if err != nil {
 | 
						|
			var cyclicInheritanceErr CyclicReleaseTemplateInheritanceError
 | 
						|
			if errors.As(err, &cyclicInheritanceErr) {
 | 
						|
				return nil, fmt.Errorf("unable to load release %q with template: %w", rt.Name, cyclicInheritanceErr)
 | 
						|
			}
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		if release.KubeContext == "" {
 | 
						|
			release.KubeContext = r.HelmDefaults.KubeContext
 | 
						|
		}
 | 
						|
		if release.Labels == nil {
 | 
						|
			release.Labels = map[string]string{}
 | 
						|
		}
 | 
						|
		for k, v := range st.CommonLabels {
 | 
						|
			release.Labels[k] = v
 | 
						|
		}
 | 
						|
		if len(release.ApiVersions) == 0 {
 | 
						|
			release.ApiVersions = st.ApiVersions
 | 
						|
		}
 | 
						|
		if release.KubeVersion == "" {
 | 
						|
			release.KubeVersion = st.KubeVersion
 | 
						|
		}
 | 
						|
 | 
						|
		successFlag := false
 | 
						|
		for it, prev := 0, release; it < 6; it++ {
 | 
						|
			tmplData := st.createReleaseTemplateData(prev, vals)
 | 
						|
			renderer := tmpl.NewFileRenderer(st.fs, st.basePath, tmplData)
 | 
						|
			r, err := release.ExecuteTemplateExpressions(renderer)
 | 
						|
			if err != nil {
 | 
						|
				return nil, fmt.Errorf("failed executing templates in release \"%s\".\"%s\": %v", st.FilePath, release.Name, err)
 | 
						|
			}
 | 
						|
			if reflect.DeepEqual(prev, r) {
 | 
						|
				successFlag = true
 | 
						|
				if err := updateBoolTemplatedValues(r); err != nil {
 | 
						|
					return nil, fmt.Errorf("failed executing templates in release \"%s\".\"%s\": %v", st.FilePath, release.Name, err)
 | 
						|
				}
 | 
						|
				st.Releases[i] = *r
 | 
						|
				break
 | 
						|
			}
 | 
						|
			prev = r
 | 
						|
		}
 | 
						|
		if !successFlag {
 | 
						|
			return nil, fmt.Errorf("failed executing templates in release \"%s\".\"%s\": %s", st.FilePath, release.Name,
 | 
						|
				"recursive references can't be resolved")
 | 
						|
		}
 | 
						|
 | 
						|
		if st.Releases[i].Chart == "" {
 | 
						|
			return nil, fmt.Errorf("encountered empty chart while reading release %q", st.Releases[i].Name)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return &r, nil
 | 
						|
}
 | 
						|
 | 
						|
type CyclicReleaseTemplateInheritanceError struct {
 | 
						|
	Message string
 | 
						|
}
 | 
						|
 | 
						|
func (e CyclicReleaseTemplateInheritanceError) Error() string {
 | 
						|
	return e.Message
 | 
						|
}
 | 
						|
 | 
						|
// releaseWithInheritedTemplate generates a new ReleaseSpec from a ReleaseSpec, by recursively inheriting
 | 
						|
// release templates referenced by the spec's `inherit` field.
 | 
						|
// The third parameter retains the current state of the recursive call, to detect a cyclic dependency a.k.a
 | 
						|
// a cyclic relese template inheritance.
 | 
						|
// This functions fails with a CyclicReleaseTemplateInheritanceError if it finds a cyclic inheritance.
 | 
						|
func (st *HelmState) releaseWithInheritedTemplate(r *ReleaseSpec, inheritancePath []string) (*ReleaseSpec, error) {
 | 
						|
	var merged ReleaseSpec
 | 
						|
 | 
						|
	for _, inherit := range r.Inherit {
 | 
						|
		templateName := inherit.Template
 | 
						|
		if templateName == "" {
 | 
						|
			return r, nil
 | 
						|
		}
 | 
						|
 | 
						|
		path := append([]string{}, inheritancePath...)
 | 
						|
		path = append(path, templateName)
 | 
						|
 | 
						|
		var cycleFound bool
 | 
						|
		for _, t := range inheritancePath {
 | 
						|
			if t == templateName {
 | 
						|
				cycleFound = true
 | 
						|
				break
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if cycleFound {
 | 
						|
			return nil, CyclicReleaseTemplateInheritanceError{Message: fmt.Sprintf("cyclic inheritance detected: %s", strings.Join(path, "->"))}
 | 
						|
		}
 | 
						|
 | 
						|
		template, defined := st.Templates[templateName]
 | 
						|
		if !defined {
 | 
						|
			return nil, fmt.Errorf("release %q tried to inherit inexistent release template %q", r.Name, templateName)
 | 
						|
		}
 | 
						|
 | 
						|
		src, err := st.releaseWithInheritedTemplate(&template.ReleaseSpec, path)
 | 
						|
		if err != nil {
 | 
						|
			return nil, fmt.Errorf("unable to load release template %q: %w", templateName, err)
 | 
						|
		}
 | 
						|
 | 
						|
		for _, k := range inherit.Except {
 | 
						|
			switch k {
 | 
						|
			case "labels":
 | 
						|
				src.Labels = map[string]string{}
 | 
						|
			case "values":
 | 
						|
				src.Values = nil
 | 
						|
			case "valuesTemplate":
 | 
						|
				src.ValuesTemplate = nil
 | 
						|
			case "setTemplate":
 | 
						|
				src.SetValuesTemplate = nil
 | 
						|
			case "set":
 | 
						|
				src.SetValues = nil
 | 
						|
			default:
 | 
						|
				return nil, fmt.Errorf("%q is not allowed under `inherit`. Allowed values are \"set\", \"setTemplate\", \"values\", \"valuesTemplate\", and \"labels\"", k)
 | 
						|
			}
 | 
						|
 | 
						|
			st.logger.Debugf("excluded field %q when inheriting template %q to release %q", k, templateName, r.Name)
 | 
						|
		}
 | 
						|
 | 
						|
		if err := mergo.Merge(&merged, src, mergo.WithAppendSlice, mergo.WithSliceDeepCopy); err != nil {
 | 
						|
			return nil, fmt.Errorf("unable to inherit release template %q: %w", templateName, err)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if err := mergo.Merge(&merged, r, mergo.WithAppendSlice, mergo.WithSliceDeepCopy); err != nil {
 | 
						|
		return nil, fmt.Errorf("unable to load release %q: %w", r.Name, err)
 | 
						|
	}
 | 
						|
 | 
						|
	merged.Inherit = nil
 | 
						|
 | 
						|
	return &merged, nil
 | 
						|
}
 |