227 lines
6.4 KiB
Go
227 lines
6.4 KiB
Go
package state
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"reflect"
|
|
"strings"
|
|
|
|
"dario.cat/mergo"
|
|
|
|
"github.com/helmfile/helmfile/pkg/tmpl"
|
|
"github.com/helmfile/helmfile/pkg/yaml"
|
|
)
|
|
|
|
func (st *HelmState) Values() map[string]any {
|
|
if st.RenderedValues == nil {
|
|
panic("[bug] RenderedValues is nil")
|
|
}
|
|
|
|
return st.RenderedValues
|
|
}
|
|
|
|
func (st *HelmState) createReleaseTemplateData(release *ReleaseSpec, vals map[string]any) 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,
|
|
ChartVersion: release.Version,
|
|
},
|
|
}
|
|
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
|
|
case "setString":
|
|
src.SetStringValues = nil
|
|
case "secrets":
|
|
src.Secrets = nil
|
|
default:
|
|
return nil, fmt.Errorf("%q is not allowed under `inherit`. Allowed values are \"set\", \"setTemplate\", \"values\", \"valuesTemplate\", \"secrets\", 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
|
|
}
|