From 17da685bbff3baee75e529b13c19d8f047015395 Mon Sep 17 00:00:00 2001 From: Jeroen Op 't Eynde Date: Mon, 12 Apr 2021 16:22:42 +0200 Subject: [PATCH] perf(export): only call FindEnvs once (#553) --- cmd/tk/export.go | 23 ++++++++++------------- pkg/tanka/export.go | 7 ++++--- pkg/tanka/parallel.go | 43 ++++++++++++++++++++----------------------- 3 files changed, 34 insertions(+), 39 deletions(-) diff --git a/cmd/tk/export.go b/cmd/tk/export.go index 0560074b2..f13048e90 100644 --- a/cmd/tk/export.go +++ b/cmd/tk/export.go @@ -2,13 +2,11 @@ package main import ( "fmt" - "path/filepath" "github.com/go-clix/cli" - "github.com/pkg/errors" - "github.com/grafana/tanka/pkg/jsonnet/jpath" "github.com/grafana/tanka/pkg/process" + "github.com/grafana/tanka/pkg/spec/v1alpha1" "github.com/grafana/tanka/pkg/tanka" ) @@ -62,15 +60,10 @@ func exportCmd() *cli.Command { Parallelism: *parallel, } - var paths []string + var exportEnvs []*v1alpha1.Environment for _, path := range args[1:] { // find possible environments if *recursive { - rootDir, err := jpath.FindRoot(path) - if err != nil { - return errors.Wrap(err, "resolving jpath") - } - // get absolute path to Environment envs, err := tanka.FindEnvs(path, tanka.FindOpts{Selector: opts.Selector}) if err != nil { @@ -78,13 +71,17 @@ func exportCmd() *cli.Command { } for _, env := range envs { - paths = append(paths, filepath.Join(rootDir, env.Metadata.Namespace)) + if opts.Opts.Name != "" && opts.Opts.Name != env.Metadata.Name { + continue + } + exportEnvs = append(exportEnvs, env) } continue } // validate environment - if _, err := tanka.Peek(path, opts.Opts); err != nil { + env, err := tanka.Peek(path, opts.Opts) + if err != nil { switch err.(type) { case tanka.ErrMultipleEnvs: fmt.Println("Please use --name to export a single environment or --recursive to export multiple environments.") @@ -94,11 +91,11 @@ func exportCmd() *cli.Command { } } - paths = append(paths, path) + exportEnvs = append(exportEnvs, env) } // export them - return tanka.ExportEnvironments(paths, args[0], &opts) + return tanka.ExportEnvironments(exportEnvs, args[0], &opts) } return cmd } diff --git a/pkg/tanka/export.go b/pkg/tanka/export.go index 5466a2721..8916bd0ac 100644 --- a/pkg/tanka/export.go +++ b/pkg/tanka/export.go @@ -15,6 +15,7 @@ import ( "k8s.io/apimachinery/pkg/labels" "github.com/grafana/tanka/pkg/kubernetes/manifest" + "github.com/grafana/tanka/pkg/spec/v1alpha1" ) // BelRune is a string of the Ascii character BEL which made computers ring in ancient times @@ -43,7 +44,7 @@ type ExportEnvOpts struct { Parallelism int } -func ExportEnvironments(paths []string, to string, opts *ExportEnvOpts) error { +func ExportEnvironments(envs []*v1alpha1.Environment, to string, opts *ExportEnvOpts) error { // Keep track of which file maps to which environment fileToEnv := map[string]string{} @@ -57,7 +58,7 @@ func ExportEnvironments(paths []string, to string, opts *ExportEnvOpts) error { } // get all environments for paths - envs, err := parallelLoadEnvironments(paths, parallelOpts{ + loadedEnvs, err := parallelLoadEnvironments(envs, parallelOpts{ Opts: opts.Opts, Selector: opts.Selector, Parallelism: opts.Parallelism, @@ -66,7 +67,7 @@ func ExportEnvironments(paths []string, to string, opts *ExportEnvOpts) error { return err } - for _, env := range envs { + for _, env := range loadedEnvs { // get the manifests loaded, err := LoadManifests(env, opts.Opts.Filters) if err != nil { diff --git a/pkg/tanka/parallel.go b/pkg/tanka/parallel.go index 2d23f5ea1..4fb0a1cc7 100644 --- a/pkg/tanka/parallel.go +++ b/pkg/tanka/parallel.go @@ -2,10 +2,14 @@ package tanka import ( "fmt" + "log" + "path/filepath" "k8s.io/apimachinery/pkg/labels" + "github.com/grafana/tanka/pkg/jsonnet/jpath" "github.com/grafana/tanka/pkg/spec/v1alpha1" + "github.com/pkg/errors" ) const defaultParallelism = 8 @@ -17,22 +21,9 @@ type parallelOpts struct { } // parallelLoadEnvironments evaluates multiple environments in parallel -func parallelLoadEnvironments(paths []string, opts parallelOpts) ([]*v1alpha1.Environment, error) { +func parallelLoadEnvironments(envs []*v1alpha1.Environment, opts parallelOpts) ([]*v1alpha1.Environment, error) { jobsCh := make(chan parallelJob) - list := make(map[string]string) - for _, path := range paths { - envs, err := FindEnvs(path, FindOpts{Selector: opts.Selector, JsonnetOpts: opts.JsonnetOpts}) - if err != nil { - return nil, err - } - for _, env := range envs { - if opts.Name != "" && opts.Name != env.Metadata.Name { - continue - } - list[env.Metadata.Name] = path - } - } - outCh := make(chan parallelOut, len(list)) + outCh := make(chan parallelOut, len(envs)) if opts.Parallelism <= 0 { opts.Parallelism = defaultParallelism @@ -42,7 +33,7 @@ func parallelLoadEnvironments(paths []string, opts parallelOpts) ([]*v1alpha1.En go parallelWorker(jobsCh, outCh) } - for name, path := range list { + for _, env := range envs { o := opts.Opts // TODO: This is required because the map[string]string in here is not @@ -52,32 +43,37 @@ func parallelLoadEnvironments(paths []string, opts parallelOpts) ([]*v1alpha1.En // to Tanka workflow thus being able to handle such cases o.JsonnetOpts = o.JsonnetOpts.Clone() - o.Name = name + o.Name = env.Metadata.Name + path := env.Metadata.Namespace + rootDir, err := jpath.FindRoot(path) + if err != nil { + return nil, errors.Wrap(err, "finding root") + } jobsCh <- parallelJob{ - path: path, + path: filepath.Join(rootDir, path), opts: o, } } close(jobsCh) - var envs []*v1alpha1.Environment + var outenvs []*v1alpha1.Environment var errors []error - for i := 0; i < len(list); i++ { + for i := 0; i < len(envs); i++ { out := <-outCh if out.err != nil { errors = append(errors, out.err) continue } if opts.Selector == nil || opts.Selector.Empty() || opts.Selector.Matches(out.env.Metadata) { - envs = append(envs, out.env) + outenvs = append(outenvs, out.env) } } if len(errors) != 0 { - return envs, ErrParallel{errors: errors} + return outenvs, ErrParallel{errors: errors} } - return envs, nil + return outenvs, nil } type parallelJob struct { @@ -92,6 +88,7 @@ type parallelOut struct { func parallelWorker(jobsCh <-chan parallelJob, outCh chan parallelOut) { for job := range jobsCh { + log.Printf("Loading %s from %s", job.opts.Name, job.path) env, err := LoadEnvironment(job.path, job.opts) if err != nil { err = fmt.Errorf("%s:\n %w", job.path, err)