linkerd2/jaeger/cmd/install.go

177 lines
4.1 KiB
Go

package cmd
import (
"bytes"
"context"
"fmt"
"io"
"os"
"path"
"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"
"github.com/linkerd/linkerd2/pkg/k8s"
"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"
kerrors "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
var (
templatesJaeger = []string{
"templates/namespace.yaml",
"templates/proxy-mutator.yaml",
"templates/rbac.yaml",
"templates/tracing.yaml",
}
)
func newCmdInstall() *cobra.Command {
var skipChecks bool
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 -`,
RunE: func(cmd *cobra.Command, args []string) error {
if !skipChecks {
// Ensure there is a Linkerd installation.
exists, err := checkIfLinkerdExists(cmd.Context())
if err != nil {
return fmt.Errorf("could not check for Linkerd existence: %s", err)
}
if !exists {
return fmt.Errorf("could not find a Linkerd installation")
}
}
return install(os.Stdout, options)
},
}
cmd.Flags().BoolVar(
&skipChecks, "skip-checks", false,
`Skip checks for namespace existence`,
)
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, "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
}
func checkIfLinkerdExists(ctx context.Context) (bool, error) {
kubeAPI, err := k8s.NewAPI(kubeconfigPath, kubeContext, impersonate, impersonateGroup, 0)
if err != nil {
return false, err
}
_, err = kubeAPI.CoreV1().Namespaces().Get(ctx, controlPlaneNamespace, metav1.GetOptions{})
if err != nil {
if kerrors.IsNotFound(err) {
return false, nil
}
return false, err
}
_, _, err = healthcheck.FetchCurrentConfiguration(ctx, kubeAPI, controlPlaneNamespace)
if err != nil {
if kerrors.IsNotFound(err) {
return false, nil
}
return false, err
}
return true, nil
}