162 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Go
		
	
	
	
| 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, "<no value>", "")
 | |
| 
 | |
| 	if len(yamlData) != len(sanitized) {
 | |
| 		msg := "replaced <no value>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
 | |
| }
 |