linkerd2/jaeger/cmd/install.go

153 lines
4.0 KiB
Go

package cmd
import (
"bytes"
"io"
"os"
"path"
"time"
"github.com/linkerd/linkerd2/jaeger/static"
"github.com/linkerd/linkerd2/pkg/charts"
partials "github.com/linkerd/linkerd2/pkg/charts/static"
"github.com/linkerd/linkerd2/pkg/flags"
"github.com/linkerd/linkerd2/pkg/healthcheck"
api "github.com/linkerd/linkerd2/pkg/public"
"github.com/spf13/cobra"
"helm.sh/helm/v3/pkg/chart/loader"
"helm.sh/helm/v3/pkg/chartutil"
"helm.sh/helm/v3/pkg/cli/values"
"helm.sh/helm/v3/pkg/engine"
)
var (
templatesJaeger = []string{
"templates/namespace.yaml",
"templates/jaeger-injector.yaml",
"templates/rbac.yaml",
"templates/tracing.yaml",
}
)
func newCmdInstall() *cobra.Command {
var skipChecks bool
var wait time.Duration
var options values.Options
cmd := &cobra.Command{
Use: "install [flags]",
Args: cobra.NoArgs,
Short: "Output Kubernetes resources to install jaeger extension",
Long: `Output Kubernetes resources to install jaeger extension.`,
Example: ` # Default install.
linkerd jaeger install | kubectl apply -f -
# Install Jaeger extension into a non-default namespace.
linkerd jaeger install --namespace custom | kubectl apply -f -
The installation can be configured by using the --set, --values, --set-string and --set-file flags.
A full list of configurable values can be found at https://www.github.com/linkerd/linkerd2/tree/main/jaeger/charts/linkerd-jaeger/README.md
`,
RunE: func(cmd *cobra.Command, args []string) error {
if !skipChecks {
// Wait for the core control-plane to be up and running
api.CheckPublicAPIClientOrRetryOrExit(healthcheck.Options{
ControlPlaneNamespace: controlPlaneNamespace,
KubeConfig: kubeconfigPath,
KubeContext: kubeContext,
Impersonate: impersonate,
ImpersonateGroup: impersonateGroup,
APIAddr: apiAddr,
RetryDeadline: time.Now().Add(wait),
})
}
return install(os.Stdout, options)
},
}
cmd.Flags().BoolVar(&skipChecks, "skip-checks", false, `Skip checks for linkerd core control-plane existence`)
cmd.Flags().DurationVar(&wait, "wait", 300*time.Second, "Wait for core control-plane components to be available")
flags.AddValueOptionsFlags(cmd.Flags(), &options)
return cmd
}
func install(w io.Writer, options values.Options) error {
// Create values override
valuesOverrides, err := options.MergeValues(nil)
if err != nil {
return err
}
// TODO: Add any validation logic here
return render(w, valuesOverrides)
}
func render(w io.Writer, valuesOverrides map[string]interface{}) error {
files := []*loader.BufferedFile{
{Name: chartutil.ChartfileName},
{Name: chartutil.ValuesfileName},
}
for _, template := range templatesJaeger {
files = append(files,
&loader.BufferedFile{Name: template},
)
}
var partialFiles []*loader.BufferedFile
for _, template := range charts.L5dPartials {
partialFiles = append(partialFiles,
&loader.BufferedFile{Name: template},
)
}
// Load all jaeger chart files into buffer
if err := charts.FilesReader(static.Templates, "linkerd-jaeger/", files); err != nil {
return err
}
// Load all partial chart files into buffer
if err := charts.FilesReader(partials.Templates, "", partialFiles); err != nil {
return err
}
// Create a Chart obj from the files
chart, err := loader.LoadFiles(append(files, partialFiles...))
if err != nil {
return err
}
vals, err := chartutil.CoalesceValues(chart, valuesOverrides)
if err != nil {
return err
}
vals, err = charts.InsertVersionValues(vals)
if err != nil {
return err
}
// Attach the final values into the `Values` field for rendering to work
renderedTemplates, err := engine.Render(chart, map[string]interface{}{"Values": vals})
if err != nil {
return err
}
// Merge templates and inject
var buf bytes.Buffer
for _, tmpl := range chart.Templates {
t := path.Join(chart.Metadata.Name, tmpl.Name)
if _, err := buf.WriteString(renderedTemplates[t]); err != nil {
return err
}
}
_, err = w.Write(buf.Bytes())
return err
}