From 088997eb26d22d6e41bbed5999247543cd4e2c1e Mon Sep 17 00:00:00 2001 From: Russell Centanni Date: Tue, 30 Jan 2024 17:45:55 -0500 Subject: [PATCH 1/2] fix: treat pipeline flag defaults consistently when used as command flags or pipeline flags Signed-off-by: Russell Centanni --- cmd/run_pipeline.go | 53 +++++++++++-------------------- pkg/devspace/pipeline/flags.go | 52 ++++++++++++++++++++++++++++++ pkg/devspace/pipeline/pipeline.go | 15 ++++++--- 3 files changed, 80 insertions(+), 40 deletions(-) create mode 100644 pkg/devspace/pipeline/flags.go diff --git a/cmd/run_pipeline.go b/cmd/run_pipeline.go index 048a5662ac..e821497647 100644 --- a/cmd/run_pipeline.go +++ b/cmd/run_pipeline.go @@ -20,7 +20,7 @@ import ( "github.com/loft-sh/devspace/pkg/devspace/hook" "github.com/loft-sh/devspace/pkg/devspace/kill" "github.com/loft-sh/devspace/pkg/devspace/kubectl" - "github.com/loft-sh/devspace/pkg/devspace/pipeline" + pipelinepkg "github.com/loft-sh/devspace/pkg/devspace/pipeline" "github.com/loft-sh/devspace/pkg/devspace/pipeline/types" "github.com/loft-sh/devspace/pkg/devspace/plugin" "github.com/loft-sh/devspace/pkg/devspace/upgrade" @@ -100,51 +100,34 @@ func (cmd *RunPipelineCmd) AddPipelineFlags(f factory.Factory, command *cobra.Co usage = "Flag " + pipelineFlag.Name } - var ok bool if pipelineFlag.Type == "" || pipelineFlag.Type == latest.PipelineFlagTypeBoolean { - val := false - if pipelineFlag.Default != nil { - val, ok = pipelineFlag.Default.(bool) - if !ok { - f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not a boolean", pipelineFlag.Name, pipelineFlag.Default) - continue - } + val, err := pipelinepkg.GetDefaultValue(pipelineFlag) + if err != nil { + f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not a boolean", pipelineFlag.Name, pipelineFlag.Default) } - command.Flags().BoolP(pipelineFlag.Name, pipelineFlag.Short, val, usage) + command.Flags().BoolP(pipelineFlag.Name, pipelineFlag.Short, val.(bool), usage) } else if pipelineFlag.Type == latest.PipelineFlagTypeString { - val := "" - if pipelineFlag.Default != nil { - val, ok = pipelineFlag.Default.(string) - if !ok { - f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not a string", pipelineFlag.Name, pipelineFlag.Default) - continue - } + val, err := pipelinepkg.GetDefaultValue(pipelineFlag) + if err != nil { + f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not a string", pipelineFlag.Name, pipelineFlag.Default) } - command.Flags().StringP(pipelineFlag.Name, pipelineFlag.Short, val, usage) + command.Flags().StringP(pipelineFlag.Name, pipelineFlag.Short, val.(string), usage) } else if pipelineFlag.Type == latest.PipelineFlagTypeInteger { - val := 0 - if pipelineFlag.Default != nil { - val, ok = pipelineFlag.Default.(int) - if !ok { - f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not an integer", pipelineFlag.Name, pipelineFlag.Default) - continue - } + val, err := pipelinepkg.GetDefaultValue(pipelineFlag) + if err != nil { + f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not an integer", pipelineFlag.Name, pipelineFlag.Default) } - command.Flags().IntP(pipelineFlag.Name, pipelineFlag.Short, val, usage) + command.Flags().IntP(pipelineFlag.Name, pipelineFlag.Short, val.(int), usage) } else if pipelineFlag.Type == latest.PipelineFlagTypeStringArray { - val := []string{} - if pipelineFlag.Default != nil { - val, ok = pipelineFlag.Default.([]string) - if !ok { - f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not a string array", pipelineFlag.Name, pipelineFlag.Default) - continue - } + val, err := pipelinepkg.GetDefaultValue(pipelineFlag) + if err != nil { + f.GetLog().Errorf("Error parsing default value for flag %s: %#v is not a string array", pipelineFlag.Name, pipelineFlag.Default) } - command.Flags().StringSliceP(pipelineFlag.Name, pipelineFlag.Short, val, usage) + command.Flags().StringSliceP(pipelineFlag.Name, pipelineFlag.Short, val.([]string), usage) } } } @@ -463,7 +446,7 @@ func runPipeline(ctx devspacecontext.Context, args []string, options *CommandOpt dependencyRegistry := registry.NewDependencyRegistry(ctx.Config().Config().Name, options.DeployOptions.Render) // get deploy pipeline - pipe := pipeline.NewPipeline(ctx.Config().Config().Name, devPodManager, dependencyRegistry, configPipeline, options.Options) + pipe := pipelinepkg.NewPipeline(ctx.Config().Config().Name, devPodManager, dependencyRegistry, configPipeline, options.Options) kill.SetStopFunction(func(message string) { if message != "" { ctx.Log().WriteString(logrus.FatalLevel, "\n"+ansi.Color("fatal", "red+b")+" "+message+"\n") diff --git a/pkg/devspace/pipeline/flags.go b/pkg/devspace/pipeline/flags.go new file mode 100644 index 0000000000..8ad0ff9a5d --- /dev/null +++ b/pkg/devspace/pipeline/flags.go @@ -0,0 +1,52 @@ +package pipeline + +import ( + "fmt" + + "github.com/loft-sh/devspace/pkg/devspace/config/versions/latest" +) + +func GetDefaultValue(pipelineFlag latest.PipelineFlag) (interface{}, error) { + var ok bool + + switch pipelineFlag.Type { + case "", latest.PipelineFlagTypeBoolean: + val := false + if pipelineFlag.Default != nil { + val, ok = pipelineFlag.Default.(bool) + if !ok { + return nil, fmt.Errorf(" default is not a boolean") + } + } + return val, nil + case latest.PipelineFlagTypeString: + val := "" + if pipelineFlag.Default != nil { + val, ok = pipelineFlag.Default.(string) + if !ok { + return nil, fmt.Errorf("default is not a string") + } + } + return val, nil + case latest.PipelineFlagTypeInteger: + val := 0 + if pipelineFlag.Default != nil { + val, ok = pipelineFlag.Default.(int) + if !ok { + return nil, fmt.Errorf("default is not an integer") + } + } + return val, nil + case latest.PipelineFlagTypeStringArray: + val := []string{} + if pipelineFlag.Default != nil { + val, ok = pipelineFlag.Default.([]string) + if !ok { + return nil, fmt.Errorf("default is not a string array") + } + } + return val, nil + } + + return nil, fmt.Errorf("unsupported flag type") +} diff --git a/pkg/devspace/pipeline/pipeline.go b/pkg/devspace/pipeline/pipeline.go index 57f7392104..8458a039ab 100644 --- a/pkg/devspace/pipeline/pipeline.go +++ b/pkg/devspace/pipeline/pipeline.go @@ -145,21 +145,21 @@ func (p *pipeline) WaitDev() error { p.m.Unlock() // wait for children first - errors := []error{} + errs := []error{} for _, child := range children { err := child.WaitDev() if err != nil { - errors = append(errors, err) + errs = append(errs, err) } } // wait for dev pods to finish err := p.devPodManager.Wait() if err != nil { - errors = append(errors, err) + errs = append(errs, err) } - return utilerrors.NewAggregate(errors) + return utilerrors.NewAggregate(errs) } func (p *pipeline) Name() string { @@ -425,7 +425,12 @@ func (p *pipeline) startNewDependency(ctx devspacecontext.Context, dependency ty func applyFlags(ctx devspacecontext.Context, pipeline *latest.Pipeline, setFlags []string) (devspacecontext.Context, error) { newFlags := map[string]string{} for _, flag := range pipeline.Flags { - newFlags[flag.Name] = fmt.Sprintf("%v", flag.Default) + val, err := GetDefaultValue(flag) + if err != nil { + return nil, err + } + + newFlags[flag.Name] = fmt.Sprintf("%v", val) } for _, v := range setFlags { splitted := strings.Split(v, "=") From 99ad69b586096fef0403ef0f59f2534f08899d97 Mon Sep 17 00:00:00 2001 From: Russell Centanni Date: Thu, 8 Feb 2024 08:23:34 -0500 Subject: [PATCH 2/2] fix: pipeline stringArray flag usage ENG-2735 Signed-off-by: Russell Centanni --- docs/pages/configuration/pipelines/README.mdx | 4 +- e2e/tests/pipelines/pipelines.go | 139 ++++++++++++++++++ e2e/tests/pipelines/testdata/flags/dep1.yaml | 12 ++ .../pipelines/testdata/flags/devspace.yaml | 21 +++ pkg/devspace/pipeline/flags.go | 37 ++++- pkg/devspace/pipeline/pipeline.go | 28 +++- 6 files changed, 230 insertions(+), 11 deletions(-) diff --git a/docs/pages/configuration/pipelines/README.mdx b/docs/pages/configuration/pipelines/README.mdx index 252d38823c..644ddb1db0 100644 --- a/docs/pages/configuration/pipelines/README.mdx +++ b/docs/pages/configuration/pipelines/README.mdx @@ -326,8 +326,8 @@ pipelines: short: e type: stringArray run: |- - extraEnv=$(get_flag "env") # Retrieve the value of the `env` flag and store it in a variable - echo ${extraEnv[1]} + extraEnv=($(get_flag "env")) # Retrieve the value of the `env` flag and store it in an array variable + echo ${extraEnv[0]} # Arrays are zero indexed TERMINAL_ENABLED=true if [ $(get_flag "logs") == "true" ]; then # Test if --logs/-l flag is used or not diff --git a/e2e/tests/pipelines/pipelines.go b/e2e/tests/pipelines/pipelines.go index d373a2a636..37b648f359 100644 --- a/e2e/tests/pipelines/pipelines.go +++ b/e2e/tests/pipelines/pipelines.go @@ -75,6 +75,8 @@ var _ = DevSpaceDescribe("pipelines", func() { framework.ExpectLocalFileContentsImmediately("other.txt", "test\n") framework.ExpectLocalFileContentsImmediately("other2.txt", "false\n") framework.ExpectLocalFileContentsImmediately("other3.txt", "true\n") + framework.ExpectLocalFileContentsImmediately("other4-0.txt", "one\n") + framework.ExpectLocalFileContentsImmediately("other4-1.txt", "two\n") framework.ExpectLocalFileContentsImmediately("other-profile.txt", "profile1\n") framework.ExpectLocalFileContentsImmediately("dep1-test.txt", "test\n") framework.ExpectLocalFileContentsImmediately("dep1-test2.txt", "true\n") @@ -85,6 +87,143 @@ var _ = DevSpaceDescribe("pipelines", func() { framework.ExpectLocalFileContentsImmediately("dep1-other-profile.txt", "profile1\n") }) + ginkgo.It("should resolve pipeline override array flags", func() { + tempDir, err := framework.CopyToTempDir("tests/pipelines/testdata/flags") + framework.ExpectNoError(err) + defer framework.CleanupTempDir(initialDir, tempDir) + + ns, err := kubeClient.CreateNamespace("pipelines") + framework.ExpectNoError(err) + defer framework.ExpectDeleteNamespace(kubeClient, ns) + + rootCmd := cmd.NewRootCmd(f) + persistentFlags := rootCmd.PersistentFlags() + globalFlags := flags.SetGlobalFlags(persistentFlags) + globalFlags.NoWarn = true + globalFlags.Namespace = ns + globalFlags.Profiles = []string{"profile1"} + + cmdCtx := values.WithCommandFlags(context.Background(), globalFlags.Flags) + cmdCtx = values.WithFlagsMap(cmdCtx, map[string]string{ + "other": "test", + "other2": "false", + "other3": "true", + "other4": "three four", + }) + + devCmd := &cmd.RunPipelineCmd{ + GlobalFlags: globalFlags, + Pipeline: "other", + Ctx: cmdCtx, + } + err = devCmd.RunDefault(f) + framework.ExpectNoError(err) + + framework.ExpectLocalFileContentsImmediately("other.txt", "test\n") + framework.ExpectLocalFileContentsImmediately("other2.txt", "false\n") + framework.ExpectLocalFileContentsImmediately("other3.txt", "true\n") + framework.ExpectLocalFileContentsImmediately("other-profile.txt", "profile1\n") + framework.ExpectLocalFileContentsImmediately("other4-0.txt", "three\n") + framework.ExpectLocalFileContentsImmediately("other4-1.txt", "four\n") + }) + + ginkgo.It("should resolve pipeline override with --set-flags", func() { + tempDir, err := framework.CopyToTempDir("tests/pipelines/testdata/flags") + framework.ExpectNoError(err) + defer framework.CleanupTempDir(initialDir, tempDir) + + ns, err := kubeClient.CreateNamespace("pipelines") + framework.ExpectNoError(err) + defer framework.ExpectDeleteNamespace(kubeClient, ns) + + rootCmd := cmd.NewRootCmd(f) + persistentFlags := rootCmd.PersistentFlags() + globalFlags := flags.SetGlobalFlags(persistentFlags) + globalFlags.NoWarn = true + globalFlags.Namespace = ns + globalFlags.Profiles = []string{"profile1"} + + cmdCtx := values.WithCommandFlags(context.Background(), globalFlags.Flags) + cmdCtx = values.WithFlagsMap(cmdCtx, map[string]string{}) + + devCmd := &cmd.RunPipelineCmd{ + GlobalFlags: globalFlags, + Pipeline: "other-override", + Ctx: cmdCtx, + } + err = devCmd.RunDefault(f) + framework.ExpectNoError(err) + + framework.ExpectLocalFileContentsImmediately("other.txt", "test\n") + framework.ExpectLocalFileContentsImmediately("other2.txt", "true\n") + framework.ExpectLocalFileContentsImmediately("other3.txt", "true\n") + framework.ExpectLocalFileContentsImmediately("other-profile.txt", "profile1\n") + framework.ExpectLocalFileContentsImmediately("other4-0.txt", "five\n") + framework.ExpectLocalFileContentsImmediately("other4-1.txt", "six\n") + }) + + ginkgo.It("should resolve dependency pipeline flag defaults", func() { + tempDir, err := framework.CopyToTempDir("tests/pipelines/testdata/flags") + framework.ExpectNoError(err) + defer framework.CleanupTempDir(initialDir, tempDir) + + ns, err := kubeClient.CreateNamespace("pipelines") + framework.ExpectNoError(err) + defer framework.ExpectDeleteNamespace(kubeClient, ns) + + rootCmd := cmd.NewRootCmd(f) + persistentFlags := rootCmd.PersistentFlags() + globalFlags := flags.SetGlobalFlags(persistentFlags) + globalFlags.NoWarn = true + globalFlags.Namespace = ns + globalFlags.Profiles = []string{"profile1"} + + cmdCtx := values.WithCommandFlags(context.Background(), globalFlags.Flags) + cmdCtx = values.WithFlagsMap(cmdCtx, map[string]string{}) + + devCmd := &cmd.RunPipelineCmd{ + GlobalFlags: globalFlags, + Pipeline: "arr-dep1", + Ctx: cmdCtx, + } + err = devCmd.RunDefault(f) + framework.ExpectNoError(err) + + framework.ExpectLocalFileContentsImmediately("arr-0.txt", "one") + framework.ExpectLocalFileContentsImmediately("arr-1.txt", "two") + }) + + ginkgo.It("should resolve dependency pipeline flag defaults", func() { + tempDir, err := framework.CopyToTempDir("tests/pipelines/testdata/flags") + framework.ExpectNoError(err) + defer framework.CleanupTempDir(initialDir, tempDir) + + ns, err := kubeClient.CreateNamespace("pipelines") + framework.ExpectNoError(err) + defer framework.ExpectDeleteNamespace(kubeClient, ns) + + rootCmd := cmd.NewRootCmd(f) + persistentFlags := rootCmd.PersistentFlags() + globalFlags := flags.SetGlobalFlags(persistentFlags) + globalFlags.NoWarn = true + globalFlags.Namespace = ns + globalFlags.Profiles = []string{"profile1"} + + cmdCtx := values.WithCommandFlags(context.Background(), globalFlags.Flags) + cmdCtx = values.WithFlagsMap(cmdCtx, map[string]string{}) + + devCmd := &cmd.RunPipelineCmd{ + GlobalFlags: globalFlags, + Pipeline: "arr-dep1-override", + Ctx: cmdCtx, + } + err = devCmd.RunDefault(f) + framework.ExpectNoError(err) + + framework.ExpectLocalFileContentsImmediately("arr-0.txt", "three") + framework.ExpectLocalFileContentsImmediately("arr-1.txt", "") + }) + ginkgo.It("should exec container", func() { tempDir, err := framework.CopyToTempDir("tests/pipelines/testdata/exec_container") framework.ExpectNoError(err) diff --git a/e2e/tests/pipelines/testdata/flags/dep1.yaml b/e2e/tests/pipelines/testdata/flags/dep1.yaml index 9c1cc89611..49e207ce71 100644 --- a/e2e/tests/pipelines/testdata/flags/dep1.yaml +++ b/e2e/tests/pipelines/testdata/flags/dep1.yaml @@ -35,3 +35,15 @@ pipelines: echo $(get_flag test3) > dep1-test2.txt echo $(get_flag profile) > dep1-dev-profile.txt run_pipelines other --set-flag other2=false + + array: + flags: + - name: arr + type: stringArray + default: + - one + - two + run: |- + arr=($(get_flag arr)) + echo -n ${arr[0]} > arr-0.txt + echo -n ${arr[1]} > arr-1.txt diff --git a/e2e/tests/pipelines/testdata/flags/devspace.yaml b/e2e/tests/pipelines/testdata/flags/devspace.yaml index c07db2b1f4..bafee09555 100644 --- a/e2e/tests/pipelines/testdata/flags/devspace.yaml +++ b/e2e/tests/pipelines/testdata/flags/devspace.yaml @@ -15,6 +15,11 @@ pipelines: default: true - name: other3 default: true + - name: other4 + type: stringArray + default: + - one + - two run: |- if get_flag test; then exit 1 @@ -24,6 +29,22 @@ pipelines: echo $(get_flag other2) > other2.txt echo $(get_flag other3) > other3.txt echo $(get_flag profile) > other-profile.txt + + other4=($(get_flag other4)) + echo ${other4[0]} > other4-0.txt + echo ${other4[1]} > other4-1.txt + + other-override: + run: |- + run_pipelines other --set-flag other4=five --set-flag other4=six + + arr-dep1: + run: |- + run_dependency_pipelines dep1 --pipeline array + + arr-dep1-override: + run: |- + run_dependency_pipelines dep1 --pipeline array --set-flag arr=three dev: flags: diff --git a/pkg/devspace/pipeline/flags.go b/pkg/devspace/pipeline/flags.go index 8ad0ff9a5d..312724762b 100644 --- a/pkg/devspace/pipeline/flags.go +++ b/pkg/devspace/pipeline/flags.go @@ -2,6 +2,7 @@ package pipeline import ( "fmt" + "strconv" "github.com/loft-sh/devspace/pkg/devspace/config/versions/latest" ) @@ -31,18 +32,42 @@ func GetDefaultValue(pipelineFlag latest.PipelineFlag) (interface{}, error) { case latest.PipelineFlagTypeInteger: val := 0 if pipelineFlag.Default != nil { - val, ok = pipelineFlag.Default.(int) - if !ok { - return nil, fmt.Errorf("default is not an integer") + switch pipelineFlag.Default.(type) { + case float64: + floatVal, ok := pipelineFlag.Default.(float64) + if !ok { + return nil, fmt.Errorf("default is not an integer") + } + return int(floatVal), nil + case int: + intVal, ok := pipelineFlag.Default.(int) + if !ok { + return nil, fmt.Errorf("default is not an integer") + } + return intVal, nil + case string: + strVal, ok := pipelineFlag.Default.(string) + if !ok { + return nil, fmt.Errorf("default is not an integer") + } + intVal, err := strconv.ParseInt(strVal, 10, 0) + if err != nil { + return nil, err + } + return int(intVal), nil } + return nil, fmt.Errorf("default is not an integer") } return val, nil case latest.PipelineFlagTypeStringArray: val := []string{} if pipelineFlag.Default != nil { - val, ok = pipelineFlag.Default.([]string) - if !ok { - return nil, fmt.Errorf("default is not a string array") + for _, anyVal := range pipelineFlag.Default.([]interface{}) { + strVal, ok := anyVal.(string) + if !ok { + return nil, fmt.Errorf("default is not a string array") + } + val = append(val, strVal) } } return val, nil diff --git a/pkg/devspace/pipeline/pipeline.go b/pkg/devspace/pipeline/pipeline.go index 8458a039ab..e94580910a 100644 --- a/pkg/devspace/pipeline/pipeline.go +++ b/pkg/devspace/pipeline/pipeline.go @@ -423,22 +423,44 @@ func (p *pipeline) startNewDependency(ctx devspacecontext.Context, dependency ty } func applyFlags(ctx devspacecontext.Context, pipeline *latest.Pipeline, setFlags []string) (devspacecontext.Context, error) { - newFlags := map[string]string{} + defaultFlags := map[string]string{} for _, flag := range pipeline.Flags { val, err := GetDefaultValue(flag) if err != nil { return nil, err } - newFlags[flag.Name] = fmt.Sprintf("%v", val) + switch flag.Type { + case latest.PipelineFlagTypeStringArray: + defaultFlags[flag.Name] = fmt.Sprintf("%v", strings.Join(val.([]string), " ")) + default: + defaultFlags[flag.Name] = fmt.Sprintf("%v", val) + } } + + newFlags := map[string]string{} for _, v := range setFlags { splitted := strings.Split(v, "=") if len(splitted) <= 1 { return nil, fmt.Errorf("error parsing flag %s: expected format flag=value", v) } - newFlags[splitted[0]] = strings.Join(splitted[1:], "=") + flagName := splitted[0] + flagVal := strings.Join(splitted[1:], "=") + flagVals := strings.Join(strings.Split(flagVal, ","), " ") + + if newFlags[flagName] != "" { + newFlags[flagName] = strings.Join([]string{newFlags[flagName], flagVals}, " ") + } else { + newFlags[flagName] = flagVals + } + + } + + for name, value := range defaultFlags { + if _, ok := newFlags[name]; !ok { + newFlags[name] = value + } } return ctx.WithContext(values.WithFlagsMap(ctx.Context(), newFlags)), nil