package app import ( "bytes" "fmt" "strings" "github.com/google/go-cmp/cmp" "github.com/helmfile/helmfile/pkg/environment" "github.com/helmfile/helmfile/pkg/runtime" "github.com/helmfile/helmfile/pkg/state" "github.com/helmfile/helmfile/pkg/tmpl" ) func prependLineNumbers(text string) string { buf := bytes.NewBufferString("") lines := strings.Split(text, "\n") for i, line := range lines { buf.WriteString(fmt.Sprintf("%2d: %s\n", i, line)) } return buf.String() } func (r *desiredStateLoader) renderPrestate(firstPassEnv, overrode *environment.Environment, baseDir, filename string, content []byte) (*environment.Environment, *state.HelmState) { initEnv, err := firstPassEnv.Merge(overrode) if err != nil { return firstPassEnv, nil } tmplData := state.NewEnvironmentTemplateData(*initEnv, r.namespace, map[string]any{}) firstPassRenderer := tmpl.NewFirstPassRenderer(baseDir, tmplData) // parse as much as we can, tolerate errors, this is a preparse yamlBuf, err := firstPassRenderer.RenderTemplateContentToBuffer(content) if err != nil { r.logger.Debugf("first-pass rendering input of \"%s\":\n%s", filename, prependLineNumbers(string(content))) r.logger.Debugf("template syntax error: %v", err) if yamlBuf == nil { // we have a template syntax error, let the second parse report return firstPassEnv, nil } } yamlData := yamlBuf.String() r.logger.Debugf("first-pass rendering output of \"%s\":\n%s", filename, prependLineNumbers(yamlData)) // Work-around for https://github.com/golang/go/issues/24963 sanitized := strings.ReplaceAll(yamlData, "", "") if len(yamlData) != len(sanitized) { msg := "replaced s to workaround https://github.com/golang/go/issues/24963 to address https://github.com/roboll/helmfile/issues/553:\n%s" r.logger.Debugf(msg, cmp.Diff(yamlData, sanitized)) } c := r.underlying() c.Strict = false // create preliminary state, as we may have an environment. Tolerate errors. prestate, err := c.ParseAndLoad([]byte(sanitized), baseDir, filename, r.env, true, false, firstPassEnv, overrode) if err != nil { if _, ok := err.(*state.StateLoadError); ok { r.logger.Debugf("could not deduce `environment:` block, configuring only .Environment.Name. error: %v", err) } r.logger.Debugf("error in first-pass rendering: result of \"%s\":\n%s", filename, prependLineNumbers(yamlBuf.String())) } if prestate != nil { firstPassEnv = &prestate.Env } return firstPassEnv, prestate } type RenderOpts struct { } func (r *desiredStateLoader) renderTemplatesToYaml(baseDir, filename string, content []byte) (*bytes.Buffer, error) { env := &environment.Environment{Name: r.env, Values: map[string]any(nil)} return r.renderTemplatesToYamlWithEnv(baseDir, filename, content, env, nil) } func (r *desiredStateLoader) renderTemplatesToYamlWithEnv(baseDir, filename string, content []byte, inherited, overrode *environment.Environment) (*bytes.Buffer, error) { return r.twoPassRenderTemplateToYaml(inherited, overrode, baseDir, filename, content) } func (r *desiredStateLoader) twoPassRenderTemplateToYaml(inherited, overrode *environment.Environment, baseDir, filename string, content []byte) (*bytes.Buffer, error) { // try a first pass render. This will always succeed, but can produce a limited env var phase string if !runtime.V1Mode { phase = "first-pass " } r.logger.Debugf("%srendering starting for \"%s\": inherited=%v, overrode=%v", phase, filename, inherited, overrode) initEnv, err := inherited.Merge(nil) if err != nil { return nil, err } var ( renderingPhase string finalEnv *environment.Environment vals map[string]any ) if runtime.V1Mode { var err error finalEnv, err = initEnv.Merge(overrode) if err != nil { return nil, err } vals, err = finalEnv.GetMergedValues() if err != nil { return nil, err } } else { r.logger.Debugf("first-pass uses: %v", initEnv) firstPassEnv, err := initEnv.Merge(nil) if err != nil { return nil, err } renderedEnv, prestate := r.renderPrestate(firstPassEnv, overrode, baseDir, filename, content) r.logger.Debugf("first-pass produced: %v", renderedEnv) mergedEnv, err := inherited.Merge(renderedEnv) if err != nil { return nil, err } mergedEnv, err = mergedEnv.Merge(overrode) if err != nil { return nil, err } r.logger.Debugf("first-pass rendering result of \"%s\": %v", filename, *mergedEnv) renderingPhase = "second-pass " finalEnv = mergedEnv vals, err = finalEnv.GetMergedValues() if err != nil { return nil, err } if prestate != nil { prestate.Env = *mergedEnv r.logger.Debugf("vals:\n%v\ndefaultVals:%v", vals, prestate.DefaultValues) } } tmplData := state.NewEnvironmentTemplateData(*finalEnv, r.namespace, vals) renderer := tmpl.NewFileRenderer(r.fs, baseDir, tmplData) yamlBuf, err := renderer.RenderTemplateContentToBuffer(content) if err != nil { r.logger.Debugf("%srendering failed, input of \"%s\":\n%s", renderingPhase, filename, prependLineNumbers(string(content))) return nil, err } r.logger.Debugf("%srendering result of \"%s\":\n%s", renderingPhase, filename, prependLineNumbers(yamlBuf.String())) return yamlBuf, nil }