Skip to content
Snippets Groups Projects
config.go 13 KiB
Newer Older
steeve.chailloux's avatar
steeve.chailloux committed
	"crypto/sha256"
steeve.chailloux's avatar
steeve.chailloux committed
	"io"
	"io/ioutil"
	"os"
	"path/filepath"
	"github.com/rs/zerolog"
	"github.com/valyala/fasttemplate"
Florent Aide's avatar
Florent Aide committed
// Variable ...
steeve.chailloux's avatar
steeve.chailloux committed
type Sha struct {
steeve.chailloux's avatar
steeve.chailloux committed
type Chart struct {
type Arg struct {
}

type Create struct {
	Type string
	Name string
	Args []Arg  `yaml:",flow"`
func (k CmdCreateKey) BuildArgs(namespace string, args []Arg) []string {
	output := []string{
		"-n", namespace,
		"create",
		k.Type,
		k.Name,
		"--dry-run=client",
		"-o", "yaml"}

	for _, arg := range args {
		output = append(output, arg.Flag, arg.Value)
	}
	return output
}

// Config ...
type Config struct {
	Inherit   string
	NameSpace string
	Variables []Variable       `yaml:",flow"`
	Sha       []Sha            `yaml:",flow"`
	Charts    map[string]Chart `yaml:",flow"`
	Creates   []Create         `yaml:"create,flow"`
	Dir       string           // the directory in which we found the config file
}

// Absolutize makes all chart paths absolute
func (c *Config) Absolutize(dir string) error {
	for name, chart := range c.Charts {
		resolvedChartPath := filepath.Join(dir, chart.Path)
		absChartPath, err := filepath.Abs(resolvedChartPath)
		if err != nil {
			return fmt.Errorf("failed to find abs() for %s: %w", resolvedChartPath, err)
		}

		chart.Path = absChartPath
		c.Charts[name] = chart
Florent Aide's avatar
Florent Aide committed
// NewConfig returns a *Config
steeve.chailloux's avatar
steeve.chailloux committed
func NewConfig(configDir string) (*Config, error) {
	var configName = "beaver"
	config := Config{}

	for _, ext := range []string{"yaml", "yml"} {
		configPath := filepath.Join(configDir, fmt.Sprintf("%s.%s", configName, ext))
		configInfo, err := os.Stat(configPath)
		if err != nil || configInfo.IsDir() {
			continue
		}
		configFile, err := ioutil.ReadFile(configPath)
		if err != nil {
			return nil, fmt.Errorf("fail to read config file: %s - %w", configPath, err)
		}
		err = yaml.Unmarshal(configFile, &config)
steeve.chailloux's avatar
steeve.chailloux committed
		if err != nil {
			return nil, fmt.Errorf("fail unmarshal config file: %s - %w", configPath, err)
		}
Florent Aide's avatar
Florent Aide committed

	return nil, fmt.Errorf("no beaver file found in %s", configDir)
Florent Aide's avatar
Florent Aide committed
}
steeve.chailloux's avatar
steeve.chailloux committed
func NewCmdConfig(logger zerolog.Logger, rootDir, configDir string, dryRun bool, output string) *CmdConfig {
	cmdConfig := &CmdConfig{}
	cmdConfig.DryRun = dryRun
steeve.chailloux's avatar
steeve.chailloux committed
	cmdConfig.Output = output
Florent Aide's avatar
Florent Aide committed
	cmdConfig.RootDir = rootDir
	cmdConfig.Layers = append(cmdConfig.Layers, configDir)
	cmdConfig.Spec.Charts = make(map[string]CmdChart)
	cmdConfig.Spec.Creates = make(map[CmdCreateKey]CmdCreate)
steeve.chailloux's avatar
steeve.chailloux committed
	cmdConfig.Spec.Shas = []*CmdSha{}
Florent Aide's avatar
Florent Aide committed
	cmdConfig.Namespace = ""
	cmdConfig.Logger = logger
	return cmdConfig
}

func (c *CmdConfig) Initialize(tmpDir string) error {
Florent Aide's avatar
Florent Aide committed
	if len(c.Layers) != 1 {
		return fmt.Errorf("you must only have one layer when calling Initialize, found: %d", len(c.Layers))
	}
	var (
		weNeedToGoDeeper = true
		configLayers     []*Config
	)

	resolvedConfigDir := filepath.Join(c.RootDir, c.Layers[0])
	absConfigDir, err := filepath.Abs(resolvedConfigDir)
	if err != nil {
Florent Aide's avatar
Florent Aide committed
		return fmt.Errorf("failed to find abs() for %s: %w", resolvedConfigDir, err)
Florent Aide's avatar
Florent Aide committed
	dir := absConfigDir
	dirMap := make(map[string]interface{})
	// otherwise first layer will be present twice
	c.Layers = []string{}

Florent Aide's avatar
Florent Aide committed
	for weNeedToGoDeeper {
		// guard against recursive inherit loops
		_, present := dirMap[dir]
		if present {
			var dirList []string
			for k := range dirMap {
				dirList = append(dirList, k)
			}
			return fmt.Errorf("recursive inherit loop detected: dirs %s->%s", strings.Join(dirList, "->"), dir)
		}

Florent Aide's avatar
Florent Aide committed
		config, err := c.newConfigFromDir(dir)
		if err != nil {
			return fmt.Errorf("failed to create config from %s: %w", dir, err)
		}
		if config == nil {
			if len(c.Layers) == 1 {
				return fmt.Errorf("beaver file not found in directory: %s", dir)
		config.Dir = dir
		if err := config.Absolutize(dir); err != nil {
			return fmt.Errorf("failed to absolutize config from dir: %s, %w", dir, err)
		}
Florent Aide's avatar
Florent Aide committed
		// first config dir must return a real config...
		// others can be skipped
		if config == nil && len(configLayers) == 0 {
			return fmt.Errorf("failed to find config in dir: %s", dir)
		}

		absDir, err := filepath.Abs(dir)
		if err != nil {
			return fmt.Errorf("failed to find abs() for %s: %w", dir, err)
		}

		c.Layers = append(c.Layers, absDir)
		configLayers = append(configLayers, config)

		if config == nil || config.Inherit == "" {
Florent Aide's avatar
Florent Aide committed
			weNeedToGoDeeper = false
		} else {
			resolvedDir := filepath.Join(absDir, config.Inherit)
Florent Aide's avatar
Florent Aide committed
			newDir, err := filepath.Abs(resolvedDir)
			if err != nil {
				return fmt.Errorf("failed to find abs() for %s: %w", resolvedDir, err)
			}
			dir = newDir
Florent Aide's avatar
Florent Aide committed
	// reverse our layers list
	for i, j := 0, len(configLayers)-1; i < j; i, j = i+1, j-1 {
		configLayers[i], configLayers[j] = configLayers[j], configLayers[i]
Florent Aide's avatar
Florent Aide committed
	for _, config := range configLayers {
		c.Namespace = config.NameSpace
Florent Aide's avatar
Florent Aide committed
		c.MergeVariables(config)

		for k, chart := range config.Charts {
steeve.chailloux's avatar
steeve.chailloux committed
			c.Spec.Charts[k] = cmdChartFromChart(chart)
		}
		for _, k := range config.Creates {
			cmdCreate := CmdCreateKey{Type: k.Type, Name: k.Name}
			c.Spec.Creates[cmdCreate] = CmdCreate{
				Dir:  config.Dir,
				Args: k.Args,
			}
steeve.chailloux's avatar
steeve.chailloux committed
		for _, sha := range config.Sha {
			cmdSha := CmdSha{Key: sha.Key, Resource: sha.Resource}
			c.Spec.Shas = append(c.Spec.Shas, &cmdSha)
		}
Florent Aide's avatar
Florent Aide committed
	for i, j := 0, len(c.Layers)-1; i < j; i, j = i+1, j-1 {
		c.Layers[i], c.Layers[j] = c.Layers[j], c.Layers[i]
	}
steeve.chailloux's avatar
steeve.chailloux committed
	if err := c.hydrate(tmpDir, false); err != nil {
Florent Aide's avatar
Florent Aide committed
		return fmt.Errorf("failed to hydrate tmpDir (%s): %w", tmpDir, err)
Florent Aide's avatar
Florent Aide committed
	return nil
}
Florent Aide's avatar
Florent Aide committed
func (c *CmdConfig) newConfigFromDir(dir string) (*Config, error) {
	cfg, err := NewConfig(dir)
	if err != nil {
		if errors.As(err, &viper.ConfigFileNotFoundError{}) {
Florent Aide's avatar
Florent Aide committed
			return nil, err
		}
	}
	return cfg, nil
steeve.chailloux's avatar
steeve.chailloux committed
type CmdSha struct {
	Key      string
	Resource string
	Sha      string
}

type CmdConfig struct {
	Spec      CmdSpec
	RootDir   string
Florent Aide's avatar
Florent Aide committed
	Layers    []string
	Namespace string
	Logger    zerolog.Logger
	DryRun    bool
steeve.chailloux's avatar
steeve.chailloux committed
	Output    string
steeve.chailloux's avatar
steeve.chailloux committed
func (c CmdConfig) HasShas() bool {
	return len(c.Spec.Shas) > 0
}

func (c CmdConfig) SetShas(buildDir string) error {
	for _, sha := range c.Spec.Shas {
		if err := sha.SetSha(buildDir); err != nil {
			return err
		}
	}
	return nil
}

func (s *CmdSha) SetSha(buildDir string) error {
	fPath := filepath.Join(buildDir, s.Resource)
	f, err := os.Open(fPath)
	if err != nil {
		return fmt.Errorf("failed to open %s: %w", fPath, err)
	}
	defer f.Close()
	hash := sha256.New()
	if _, err := io.Copy(hash, f); err != nil {
		return fmt.Errorf("failed to read %s: %w", fPath, err)
	}
	s.Sha = fmt.Sprintf("%x", hash.Sum(nil))
	return nil
}

type CmdSpec struct {
	Variables []Variable
steeve.chailloux's avatar
steeve.chailloux committed
	Shas      []*CmdSha
	Charts    CmdCharts
	Creates   map[CmdCreateKey]CmdCreate
steeve.chailloux's avatar
steeve.chailloux committed
func (c CmdConfig) BuildYttArgs(paths, compiled []string) []string {
	// ytt -f $chartsTmpFile --file-mark "$(basename $chartsTmpFile):type=yaml-plain"\
	//   -f base/ytt/ -f base/ytt.yml -f ns1/ytt/ -f ns1/ytt.yml
	var args []string
	for _, c := range compiled {
steeve.chailloux's avatar
steeve.chailloux committed
		args = append(args, "-f", c, fmt.Sprintf("--file-mark=%s:type=yaml-plain", filepath.Base(c)))
steeve.chailloux's avatar
steeve.chailloux committed
	for _, path := range paths {
		args = append(args, "-f", path)
	return args
}

type CmdCharts map[string]CmdChart

type CmdChart struct {
	Type            string
	Path            string
	ValuesFileNames []string
}

const (
	HelmType = "helm"
	YttType  = "ytt"
)

// BuildArgs is in charge of producing the argument list to be provided
// to the cmd
func (c CmdChart) BuildArgs(name, namespace string) ([]string, error) {
	var args []string
	switch c.Type {
	case HelmType:
		// helm template name vendor/helm/mychart/ --namespace ns1 -f base.values.yaml -f ns.yaml -f ns.values.yaml
		args = append(args, "template", name, c.Path, "--namespace", namespace)
	case YttType:
		args = append(args, "-f", c.Path)
	default:
		return nil, fmt.Errorf("unsupported chart %s type: %q", c.Path, c.Type)
	}
	for _, vFile := range c.ValuesFileNames {
		args = append(args, "-f", vFile)
	}
	return args, nil
}

func cmdChartFromChart(c Chart) CmdChart {
	return CmdChart{
		Type:            c.Type,
		Path:            c.Path,
		ValuesFileNames: nil,
	}
}

steeve.chailloux's avatar
steeve.chailloux committed
func (c *CmdConfig) prepareVariables(doSha bool) (map[string]interface{}, error) {
	variables := make(map[string]interface{})
	for _, variable := range c.Spec.Variables {
		variables[variable.Name] = variable.Value
	}
	variables["namespace"] = c.Namespace
steeve.chailloux's avatar
steeve.chailloux committed
	for _, sha := range c.Spec.Shas {
		key := fmt.Sprintf("sha.%s", sha.Key)
steeve.chailloux's avatar
steeve.chailloux committed
		if doSha {
			if sha.Sha != "" {
steeve.chailloux's avatar
steeve.chailloux committed
			} else {
				return nil,  fmt.Errorf("SHA not found for %s", sha.Key)
			}
		} else {
			variables[key] = fmt.Sprintf("<[sha.%s]>", sha.Key)
steeve.chailloux's avatar
steeve.chailloux committed
		}
	}
	return variables, nil
}

func (c *CmdConfig) populate() {
Florent Aide's avatar
Florent Aide committed
	c.Spec.Charts = FindFiles(c.Layers, c.Spec.Charts)
	c.Spec.Ytt = findYtts(c.Layers)
func findYtts(layers []string) []string {
	var result []string
Florent Aide's avatar
Florent Aide committed

	// we cannot use findYaml here because the order matters
Florent Aide's avatar
Florent Aide committed
	for _, layer := range layers {
		yttDirPath := filepath.Join(layer, "ytt")
		yttDirInfo, err := os.Stat(yttDirPath)
		if err == nil && yttDirInfo.IsDir() {
			result = append(result, yttDirPath)
		}

		for _, ext := range []string{"yaml", "yml"} {
			yttFilePath := filepath.Join(layer, fmt.Sprintf("ytt.%s", ext))
			yttFileInfo, err := os.Stat(yttFilePath)
			if err == nil && !yttFileInfo.IsDir() {
				result = append(result, yttFilePath)
Florent Aide's avatar
Florent Aide committed
func FindFiles(layers []string, charts map[string]CmdChart) map[string]CmdChart {
	for name, chart := range charts {
Florent Aide's avatar
Florent Aide committed
		files := findYaml(layers, name)
		chart.ValuesFileNames = append(chart.ValuesFileNames, files...)
		charts[name] = chart
	}
	return charts
}

Florent Aide's avatar
Florent Aide committed
func findYaml(layers []string, name string) []string {
	var files []string
Florent Aide's avatar
Florent Aide committed
	for _, layer := range layers {
		for _, ext := range []string{"yaml", "yml"} {
Florent Aide's avatar
Florent Aide committed
			fpath := filepath.Join(layer, fmt.Sprintf("%s.%s", name, ext))
			if _, err := os.Stat(fpath); err == nil {
				files = append(files, fpath)
			}
		}
	}
	return files
}

func hydrate(input string, output *os.File, variables map[string]interface{}) error {
	byteTemplate, err := ioutil.ReadFile(input)
	if err != nil {
		return fmt.Errorf("failed to read %s: %w", input, err)
	}
	template := string(byteTemplate)

	t, err := fasttemplate.NewTemplate(template, "<[", "]>")
	if err != nil {
steeve.chailloux's avatar
steeve.chailloux committed
		return fmt.Errorf("unexpected error when parsing template: %w", err)
	}
	s := t.ExecuteString(variables)
	if _, err := output.Write([]byte(s)); err != nil {
		return fmt.Errorf("failed to template for %s: %w", output.Name(), err)
	}
	return nil
}

steeve.chailloux's avatar
steeve.chailloux committed
func hydrateFiles(tmpDir string, variables map[string]interface{}, paths []string) ([]string, error) {
	var result []string
	for _, path := range paths {
		fileInfo, err := os.Stat(path)
		if err != nil {
			return nil, fmt.Errorf("hydrateFiles could not stat file or dir %s: %w", path, err)
		}
		if fileInfo.IsDir() {
			result = append(result, path)
			continue
		}

		ext := filepath.Ext(path)
		tmpFile, err := ioutil.TempFile(tmpDir, fmt.Sprintf("%s-*%s", strings.TrimSuffix(filepath.Base(path), ext), ext))
		if err != nil {
			return nil, fmt.Errorf("hydrateFiles failed to create tempfile: %w", err)
		defer func() {
			_ = tmpFile.Close()
		}()
		if err := hydrate(path, tmpFile, variables); err != nil {
			return nil, fmt.Errorf("failed to hydrate: %w", err)
		}
		result = append(result, tmpFile.Name())
// hydrate expands templated variables in our config with concrete values
steeve.chailloux's avatar
steeve.chailloux committed
func (c *CmdConfig) hydrate(dirName string, doSha bool) error {
	variables , err := c.prepareVariables(doSha)
	if err != nil {
		return fmt.Errorf("Cannot prepare variables %w", err)
	}

	for key, chart := range c.Spec.Charts {
		paths, err := hydrateFiles(dirName, variables, chart.ValuesFileNames)
		if err != nil {
		chart.ValuesFileNames = paths
		c.Spec.Charts[key] = chart
	paths, err := hydrateFiles(dirName, variables, c.Spec.Ytt)
	if err != nil {
		return err
	}
	c.Spec.Ytt = paths
	return nil
}

// MergeVariables takes a config (from a file, not a cmd one) and import its
// variables into the current cmdconfig by replacing old ones
// and adding the new ones
func (c *CmdConfig) MergeVariables(other *Config) {
	for _, variable := range other.Variables {
		c.overlayVariable(variable)
	}
}

// overlayVariable takes a variable in and either replaces an existing variable
// of the same name or create a new variable in the config if no matching name
// is found
func (c *CmdConfig) overlayVariable(v Variable) {
	// find same variable by name and replace is value
	// if not found then create the variable
	for index, originalVariable := range c.Spec.Variables {
		if originalVariable.Name == v.Name {
			c.Spec.Variables[index].Value = v.Value
			return
		}
	}
	c.Spec.Variables = append(c.Spec.Variables, v)
}