linkerd2/pkg/charts/charts.go

178 lines
4.8 KiB
Go

package charts
import (
"bytes"
"net/http"
"path"
"strings"
"github.com/ghodss/yaml"
"github.com/linkerd/linkerd2/pkg/charts/static"
"github.com/linkerd/linkerd2/pkg/version"
"helm.sh/helm/v3/pkg/chart/loader"
"helm.sh/helm/v3/pkg/chartutil"
"helm.sh/helm/v3/pkg/engine"
)
const versionPlaceholder = "linkerdVersionValue"
var (
// L5dPartials is the list of templates in partials chart
// Keep this slice synced with the contents of /charts/partials
L5dPartials = []string{
"charts/partials/" + chartutil.ChartfileName,
"charts/partials/templates/_proxy.tpl",
"charts/partials/templates/_proxy-init.tpl",
"charts/partials/templates/_volumes.tpl",
"charts/partials/templates/_resources.tpl",
"charts/partials/templates/_metadata.tpl",
"charts/partials/templates/_helpers.tpl",
"charts/partials/templates/_debug.tpl",
"charts/partials/templates/_capabilities.tpl",
"charts/partials/templates/_trace.tpl",
"charts/partials/templates/_nodeselector.tpl",
"charts/partials/templates/_tolerations.tpl",
"charts/partials/templates/_affinity.tpl",
"charts/partials/templates/_addons.tpl",
"charts/partials/templates/_validate.tpl",
"charts/partials/templates/_pull-secrets.tpl",
}
)
// Chart holds the necessary info to render a Helm chart
type Chart struct {
Name string
Dir string
Namespace string
RawValues []byte
Files []*loader.BufferedFile
Fs http.FileSystem
}
func (c *Chart) render(partialsFiles []*loader.BufferedFile) (bytes.Buffer, error) {
if err := FilesReader(c.Fs, c.Dir+"/", c.Files); err != nil {
return bytes.Buffer{}, err
}
// static.Templates is used as partials are always available there
if err := FilesReader(static.Templates, "", partialsFiles); err != nil {
return bytes.Buffer{}, err
}
// Create chart and render templates
chart, err := loader.LoadFiles(append(c.Files, partialsFiles...))
if err != nil {
return bytes.Buffer{}, err
}
releaseOptions := chartutil.ReleaseOptions{
Name: c.Name,
IsInstall: true,
IsUpgrade: false,
Namespace: c.Namespace,
}
var rawMapValues map[string]interface{}
err = yaml.Unmarshal(c.RawValues, &rawMapValues)
if err != nil {
return bytes.Buffer{}, err
}
valuesToRender, err := chartutil.ToRenderValues(chart, rawMapValues, releaseOptions, nil)
if err != nil {
return bytes.Buffer{}, err
}
renderedTemplates, err := engine.Render(chart, valuesToRender)
if err != nil {
return bytes.Buffer{}, err
}
// Merge templates and inject
var buf bytes.Buffer
for _, tmpl := range c.Files {
t := path.Join(releaseOptions.Name, tmpl.Name)
if _, err := buf.WriteString(renderedTemplates[t]); err != nil {
return bytes.Buffer{}, err
}
}
return buf, nil
}
// Render returns a bytes buffer with the result of rendering a Helm chart
func (c *Chart) Render() (bytes.Buffer, error) {
l5dPartials := []*loader.BufferedFile{}
for _, template := range L5dPartials {
l5dPartials = append(l5dPartials, &loader.BufferedFile{
Name: template,
})
}
return c.render(l5dPartials)
}
// RenderCNI returns a bytes buffer with the result of rendering a Helm chart
func (c *Chart) RenderCNI() (bytes.Buffer, error) {
cniPartials := []*loader.BufferedFile{
{Name: "charts/partials/" + chartutil.ChartfileName},
{Name: "charts/partials/templates/_helpers.tpl"},
{Name: "charts/partials/templates/_pull-secrets.tpl"},
}
return c.render(cniPartials)
}
// RenderNoPartials returns a bytes buffer with the result of rendering a Helm chart with no partials
func (c *Chart) RenderNoPartials() (bytes.Buffer, error) {
return c.render([]*loader.BufferedFile{})
}
// ReadFile updates the buffered file with the data read from disk
func ReadFile(fs http.FileSystem, dir string, f *loader.BufferedFile) error {
filename := dir + f.Name
if dir == "" {
filename = filename[7:]
}
file, err := fs.Open(filename)
if err != nil {
return err
}
defer file.Close()
buf := new(bytes.Buffer)
if _, err := buf.ReadFrom(file); err != nil {
return err
}
f.Data = buf.Bytes()
return nil
}
// FilesReader reads all the files from a directory
func FilesReader(fs http.FileSystem, dir string, files []*loader.BufferedFile) error {
for _, f := range files {
if err := ReadFile(fs, dir, f); err != nil {
return err
}
}
return nil
}
// InsertVersion returns the chart values file contents passed in
// with the version placeholder replaced with the current version
func InsertVersion(data []byte) []byte {
dataWithVersion := strings.Replace(string(data), versionPlaceholder, version.Version, -1)
return []byte(dataWithVersion)
}
// InsertVersionValues returns the chart values with the version placeholder
// replaced with the current version.
func InsertVersionValues(values chartutil.Values) (chartutil.Values, error) {
raw, err := values.YAML()
if err != nil {
return nil, err
}
return chartutil.ReadValues(InsertVersion([]byte(raw)))
}