Newer
Older
package runner
import (
"io/ioutil"
"os"
"path/filepath"
"github.com/go-yaml/yaml"
"github.com/spf13/viper"
"github.com/valyala/fasttemplate"
type Variable struct {
Name string
Value string
Key string
Resource string
Type string
Path string
Flag string
Value string
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
}
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 {
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
}
return nil
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)
if err != nil {
return nil, fmt.Errorf("fail unmarshal config file: %s - %w", configPath, err)
}
return &config, nil
return nil, fmt.Errorf("no beaver file found in %s", configDir)
func NewCmdConfig(logger zerolog.Logger, rootDir, configDir string, dryRun bool, output string) *CmdConfig {
cmdConfig.RootDir = rootDir
cmdConfig.Layers = append(cmdConfig.Layers, configDir)
cmdConfig.Spec.Charts = make(map[string]CmdChart)
cmdConfig.Spec.Creates = make(map[CmdCreateKey]CmdCreate)
cmdConfig.Logger = logger
return cmdConfig
}
func (c *CmdConfig) Initialize(tmpDir string) error {
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)
return fmt.Errorf("failed to find abs() for %s: %w", resolvedConfigDir, err)
// otherwise first layer will be present twice
c.Layers = []string{}
// 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)
}
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)
}
// 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)
resolvedDir := filepath.Join(absDir, config.Inherit)
newDir, err := filepath.Abs(resolvedDir)
if err != nil {
return fmt.Errorf("failed to find abs() for %s: %w", resolvedDir, err)
}
dir = newDir
// 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]
cmdCreate := CmdCreateKey{Type: k.Type, Name: k.Name}
c.Spec.Creates[cmdCreate] = CmdCreate{
Dir: config.Dir,
Args: k.Args,
}
for _, sha := range config.Sha {
cmdSha := CmdSha{Key: sha.Key, Resource: sha.Resource}
c.Spec.Shas = append(c.Spec.Shas, &cmdSha)
}
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]
}
if err := c.hydrate(tmpDir, false); err != nil {
return fmt.Errorf("failed to hydrate tmpDir (%s): %w", tmpDir, err)
func (c *CmdConfig) newConfigFromDir(dir string) (*Config, error) {
cfg, err := NewConfig(dir)
if err != nil {
if errors.As(err, &viper.ConfigFileNotFoundError{}) {
type CmdCreateKey struct {
Type string
Name string
type CmdCreate struct {
Dir string
Args []Arg
}
type CmdSha struct {
Key string
Resource string
Sha string
}
type CmdConfig struct {
Spec CmdSpec
RootDir string
Namespace string
Logger zerolog.Logger
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
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
Creates map[CmdCreateKey]CmdCreate
}
type Ytt []string
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 {
args = append(args, "-f", c, fmt.Sprintf("--file-mark=%s:type=yaml-plain", filepath.Base(c)))
for _, path := range paths {
args = append(args, "-f", path)
}
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,
}
}
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
key := fmt.Sprintf("sha.%s", sha.Key)
variables[key] = sha.Sha
} else {
return nil, fmt.Errorf("SHA not found for %s", sha.Key)
}
} else {
variables[key] = fmt.Sprintf("<[sha.%s]>", sha.Key)
}
func (c *CmdConfig) populate() {
// we cannot use findYaml here because the order matters
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)
}
}
}
return result
}
func FindFiles(layers []string, charts map[string]CmdChart) map[string]CmdChart {
for name, chart := range charts {
chart.ValuesFileNames = append(chart.ValuesFileNames, files...)
charts[name] = chart
}
return charts
}
for _, ext := range []string{"yaml", "yml"} {
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 {
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
}
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())
}
return result, nil
}
// hydrate expands templated variables in our config with concrete values
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) {
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)
}