From 1bb89d394888bb121aa3b3888fc73ce47d6c848e Mon Sep 17 00:00:00 2001 From: Chitrang Patel Date: Fri, 3 Nov 2023 14:38:42 -0400 Subject: [PATCH] Add support for params between Step and StepActions Following the previous [PR](https://github.com/tektoncd/pipeline/pull/7317), which introduced Params to the `StepAction` CRD, this PR integrates `param` usage between `Steps` and `StepActions`. This completes support for params in `StepActions`. This work is part of issue https://github.com/tektoncd/pipeline/issues/7259. --- docs/pipeline-api.md | 32 +- .../pipelineruns/alpha/stepaction-params.yaml | 62 +++ .../v1/taskruns/alpha/stepaction-params.yaml | 68 ++++ pkg/apis/pipeline/v1/container_types.go | 4 + pkg/apis/pipeline/v1/openapi_generated.go | 21 +- pkg/apis/pipeline/v1/swagger.json | 9 + pkg/apis/pipeline/v1/task_validation.go | 6 + pkg/apis/pipeline/v1/task_validation_test.go | 23 ++ pkg/apis/pipeline/v1/zz_generated.deepcopy.go | 7 + .../pipeline/v1beta1/container_conversion.go | 12 + pkg/apis/pipeline/v1beta1/container_types.go | 4 + .../pipeline/v1beta1/openapi_generated.go | 21 +- pkg/apis/pipeline/v1beta1/swagger.json | 9 + .../pipeline/v1beta1/task_conversion_test.go | 3 + pkg/apis/pipeline/v1beta1/task_validation.go | 6 + .../pipeline/v1beta1/task_validation_test.go | 23 ++ .../pipeline/v1beta1/zz_generated.deepcopy.go | 7 + pkg/reconciler/pipelinerun/resources/apply.go | 6 +- pkg/reconciler/taskrun/resources/apply.go | 146 ++++++- .../taskrun/resources/apply_test.go | 56 +++ pkg/reconciler/taskrun/resources/taskref.go | 2 +- pkg/reconciler/taskrun/resources/taskspec.go | 7 + .../taskrun/resources/taskspec_test.go | 364 +++++++++++++++++- .../taskrun/resources/validate_params.go | 38 ++ pkg/reconciler/taskrun/taskrun_test.go | 192 +++++++++ 25 files changed, 1101 insertions(+), 27 deletions(-) create mode 100644 examples/v1/pipelineruns/alpha/stepaction-params.yaml create mode 100644 examples/v1/taskruns/alpha/stepaction-params.yaml diff --git a/docs/pipeline-api.md b/docs/pipeline-api.md index d2f6a4d93b7..3ea5a3c7605 100644 --- a/docs/pipeline-api.md +++ b/docs/pipeline-api.md @@ -1802,7 +1802,7 @@ map[string]string

Params ([]github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.Param alias)

-(Appears on:IncludeParams, Matrix, PipelineRunSpec, PipelineTask, ResolverRef, TaskRunInputs, TaskRunSpec) +(Appears on:IncludeParams, Matrix, PipelineRunSpec, PipelineTask, ResolverRef, Step, TaskRunInputs, TaskRunSpec)

Params is a list of Param

@@ -4427,6 +4427,20 @@ Ref

Contains the reference to an existing StepAction.

+ + +params
+ + +Params + + + + +(Optional) +

Parameters declares parameters passed to this step action.

+ +

StepOutputConfig @@ -10215,7 +10229,7 @@ map[string]string

Params ([]github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.Param alias)

-(Appears on:RunSpec, CustomRunSpec, IncludeParams, Matrix, PipelineRunSpec, PipelineTask, ResolverRef, TaskRunSpec) +(Appears on:RunSpec, CustomRunSpec, IncludeParams, Matrix, PipelineRunSpec, PipelineTask, ResolverRef, Step, TaskRunSpec)

Params is a list of Param

@@ -13227,6 +13241,20 @@ Ref

Contains the reference to an existing StepAction.

+ + +params
+ + +Params + + + + +(Optional) +

Parameters declares parameters passed to this step action.

+ +

StepOutputConfig diff --git a/examples/v1/pipelineruns/alpha/stepaction-params.yaml b/examples/v1/pipelineruns/alpha/stepaction-params.yaml new file mode 100644 index 00000000000..287add81f2d --- /dev/null +++ b/examples/v1/pipelineruns/alpha/stepaction-params.yaml @@ -0,0 +1,62 @@ +apiVersion: tekton.dev/v1alpha1 +kind: StepAction +metadata: + name: step-action +spec: + params: + - name: not + default: "a mysterious GH action" + - name: array-param + type: array + default: + - ho + - ho + - ho + - name: object-param + type: object + properties: + foo: + type: string + bar: + type: string + default: + foo: "default foo" + image: bash:3.2 + args: [ + "echo", + "$(params.array-param[*])", + "$(params.not)", + "$(params.object-param.foo)", + "$(params.object-param.bar)" + ] +--- +apiVersion: tekton.dev/v1 +kind: PipelineRun +metadata: + name: step-action-pipeline-run +spec: + params: + - name: notaction + value: "a github, mysterious or Task action" + - name: arrayparam + value: + - "hello, " + - "hi!!" + - name: objectparam + value: + bar: "chocolate bar" + PipelineSpec: + tasks: + - name: run-action + taskSpec: + steps: + - name: action-runner + ref: + name: step-action + params: + - name: not + value: $(params.notaction) + - name: array-param + value: $(params.arrayparam[*]) + - name: object-param + value: $(params.objectparam[*]) diff --git a/examples/v1/taskruns/alpha/stepaction-params.yaml b/examples/v1/taskruns/alpha/stepaction-params.yaml new file mode 100644 index 00000000000..b416099e417 --- /dev/null +++ b/examples/v1/taskruns/alpha/stepaction-params.yaml @@ -0,0 +1,68 @@ +apiVersion: tekton.dev/v1alpha1 +kind: StepAction +metadata: + name: step-action +spec: + params: + - name: not + default: "a mysterious GH action" + - name: array-param + type: array + default: + - ho + - ho + - ho + - name: object-param + type: object + properties: + foo: + type: string + bar: + type: string + default: + foo: "default foo" + image: bash:3.2 + args: [ + "echo", + "$(params.array-param[*])", + "$(params.not)", + "$(params.object-param.foo)", + "$(params.object-param.bar)" + ] +--- +apiVersion: tekton.dev/v1 +kind: TaskRun +metadata: + name: step-action-run +spec: + params: + - name: notaction + value: "a github, mysterious or Task action" + - name: arrayparam + value: + - "hello, " + - "hi!!" + - name: objectparam + value: + bar: "chocolate bar" + TaskSpec: + #params: + # - name: objectparam + # properties: + # foo: + # type: string + # bar: + # type: string + # default: + # bar: "default bar" + steps: + - name: action-runner + ref: + name: step-action + params: + - name: not + value: $(params.notaction) + - name: array-param + value: $(params.arrayparam[*]) + - name: object-param + value: $(params.objectparam[*]) diff --git a/pkg/apis/pipeline/v1/container_types.go b/pkg/apis/pipeline/v1/container_types.go index 922ac1c8d11..b989fdf055b 100644 --- a/pkg/apis/pipeline/v1/container_types.go +++ b/pkg/apis/pipeline/v1/container_types.go @@ -138,6 +138,10 @@ type Step struct { // Contains the reference to an existing StepAction. //+optional Ref *Ref `json:"ref,omitempty"` + // Parameters declares parameters passed to this step action. + // +optional + // +listType=atomic + Params Params `json:"params,omitempty"` } // Ref can be used to refer to a specific instance of a StepAction. diff --git a/pkg/apis/pipeline/v1/openapi_generated.go b/pkg/apis/pipeline/v1/openapi_generated.go index f21c7a06c0c..a2d11ad69f6 100644 --- a/pkg/apis/pipeline/v1/openapi_generated.go +++ b/pkg/apis/pipeline/v1/openapi_generated.go @@ -2966,12 +2966,31 @@ func schema_pkg_apis_pipeline_v1_Step(ref common.ReferenceCallback) common.OpenA Ref: ref("github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.Ref"), }, }, + "params": { + VendorExtensible: spec.VendorExtensible{ + Extensions: spec.Extensions{ + "x-kubernetes-list-type": "atomic", + }, + }, + SchemaProps: spec.SchemaProps{ + Description: "Parameters declares parameters passed to this step action.", + Type: []string{"array"}, + Items: &spec.SchemaOrArray{ + Schema: &spec.Schema{ + SchemaProps: spec.SchemaProps{ + Default: map[string]interface{}{}, + Ref: ref("github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.Param"), + }, + }, + }, + }, + }, }, Required: []string{"name"}, }, }, Dependencies: []string{ - "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.Ref", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.StepOutputConfig", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.WorkspaceUsage", "k8s.io/api/core/v1.EnvFromSource", "k8s.io/api/core/v1.EnvVar", "k8s.io/api/core/v1.ResourceRequirements", "k8s.io/api/core/v1.SecurityContext", "k8s.io/api/core/v1.VolumeDevice", "k8s.io/api/core/v1.VolumeMount", "k8s.io/apimachinery/pkg/apis/meta/v1.Duration"}, + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.Param", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.Ref", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.StepOutputConfig", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1.WorkspaceUsage", "k8s.io/api/core/v1.EnvFromSource", "k8s.io/api/core/v1.EnvVar", "k8s.io/api/core/v1.ResourceRequirements", "k8s.io/api/core/v1.SecurityContext", "k8s.io/api/core/v1.VolumeDevice", "k8s.io/api/core/v1.VolumeMount", "k8s.io/apimachinery/pkg/apis/meta/v1.Duration"}, } } diff --git a/pkg/apis/pipeline/v1/swagger.json b/pkg/apis/pipeline/v1/swagger.json index 79d8d7aad38..dbc7893ed3f 100644 --- a/pkg/apis/pipeline/v1/swagger.json +++ b/pkg/apis/pipeline/v1/swagger.json @@ -1463,6 +1463,15 @@ "description": "OnError defines the exiting behavior of a container on error can be set to [ continue | stopAndFail ]", "type": "string" }, + "params": { + "description": "Parameters declares parameters passed to this step action.", + "type": "array", + "items": { + "default": {}, + "$ref": "#/definitions/v1.Param" + }, + "x-kubernetes-list-type": "atomic" + }, "ref": { "description": "Contains the reference to an existing StepAction.", "$ref": "#/definitions/v1.Ref" diff --git a/pkg/apis/pipeline/v1/task_validation.go b/pkg/apis/pipeline/v1/task_validation.go index 0eabc27c9a8..4f3ba830c75 100644 --- a/pkg/apis/pipeline/v1/task_validation.go +++ b/pkg/apis/pipeline/v1/task_validation.go @@ -300,6 +300,12 @@ func validateStep(ctx context.Context, s Step, names sets.String) (errs *apis.Fi }) } } else { + if len(s.Params) > 0 { + errs = errs.Also(&apis.FieldError{ + Message: "params cannot be used without Ref", + Paths: []string{"params"}, + }) + } if s.Image == "" { errs = errs.Also(apis.ErrMissingField("Image")) } diff --git a/pkg/apis/pipeline/v1/task_validation_test.go b/pkg/apis/pipeline/v1/task_validation_test.go index 05799e14e2e..d73019120c1 100644 --- a/pkg/apis/pipeline/v1/task_validation_test.go +++ b/pkg/apis/pipeline/v1/task_validation_test.go @@ -534,6 +534,16 @@ func TestTaskSpecStepActionReferenceValidate(t *testing.T) { Name: "stepAction", }, }}, + }, { + name: "valid use of params with Ref", + Steps: []v1.Step{{ + Ref: &v1.Ref{ + Name: "stepAction", + }, + Params: v1.Params{{ + Name: "param", + }}, + }}, }} for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { @@ -1490,6 +1500,19 @@ func TestTaskSpecValidateErrorWithStepActionRef(t *testing.T) { Message: "env cannot be used with Ref", Paths: []string{"steps[0].env"}, }, + }, { + name: "Cannot use params without Ref", + Steps: []v1.Step{{ + Image: "my-image", + Params: v1.Params{{ + Name: "param", + }}, + }}, + enableStepActions: true, + expectedError: apis.FieldError{ + Message: "params cannot be used without Ref", + Paths: []string{"steps[0].params"}, + }, }} for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { diff --git a/pkg/apis/pipeline/v1/zz_generated.deepcopy.go b/pkg/apis/pipeline/v1/zz_generated.deepcopy.go index b1be9d45fe4..838f93869f2 100644 --- a/pkg/apis/pipeline/v1/zz_generated.deepcopy.go +++ b/pkg/apis/pipeline/v1/zz_generated.deepcopy.go @@ -1290,6 +1290,13 @@ func (in *Step) DeepCopyInto(out *Step) { *out = new(Ref) (*in).DeepCopyInto(*out) } + if in.Params != nil { + in, out := &in.Params, &out.Params + *out = make(Params, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } return } diff --git a/pkg/apis/pipeline/v1beta1/container_conversion.go b/pkg/apis/pipeline/v1beta1/container_conversion.go index 0ec816b2548..620a655f124 100644 --- a/pkg/apis/pipeline/v1beta1/container_conversion.go +++ b/pkg/apis/pipeline/v1beta1/container_conversion.go @@ -65,6 +65,12 @@ func (s Step) convertTo(ctx context.Context, sink *v1.Step) { sink.Ref = &v1.Ref{} s.Ref.convertTo(ctx, sink.Ref) } + sink.Params = nil + for _, p := range s.Params { + new := v1.Param{} + p.convertTo(ctx, &new) + sink.Params = append(sink.Params, new) + } } func (s *Step) convertFrom(ctx context.Context, source v1.Step) { @@ -97,6 +103,12 @@ func (s *Step) convertFrom(ctx context.Context, source v1.Step) { newRef.convertFrom(ctx, *source.Ref) s.Ref = &newRef } + s.Params = nil + for _, p := range source.Params { + new := Param{} + new.ConvertFrom(ctx, p) + s.Params = append(s.Params, new) + } } func (s StepTemplate) convertTo(ctx context.Context, sink *v1.StepTemplate) { diff --git a/pkg/apis/pipeline/v1beta1/container_types.go b/pkg/apis/pipeline/v1beta1/container_types.go index 0bf6f502664..74771c8d116 100644 --- a/pkg/apis/pipeline/v1beta1/container_types.go +++ b/pkg/apis/pipeline/v1beta1/container_types.go @@ -232,6 +232,10 @@ type Step struct { // Contains the reference to an existing StepAction. //+optional Ref *Ref `json:"ref,omitempty"` + // Parameters declares parameters passed to this step action. + // +optional + // +listType=atomic + Params Params `json:"params,omitempty"` } // Ref can be used to refer to a specific instance of a StepAction. diff --git a/pkg/apis/pipeline/v1beta1/openapi_generated.go b/pkg/apis/pipeline/v1beta1/openapi_generated.go index ae0e0b1fa93..7f9e3aa5575 100644 --- a/pkg/apis/pipeline/v1beta1/openapi_generated.go +++ b/pkg/apis/pipeline/v1beta1/openapi_generated.go @@ -3920,12 +3920,31 @@ func schema_pkg_apis_pipeline_v1beta1_Step(ref common.ReferenceCallback) common. Ref: ref("github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.Ref"), }, }, + "params": { + VendorExtensible: spec.VendorExtensible{ + Extensions: spec.Extensions{ + "x-kubernetes-list-type": "atomic", + }, + }, + SchemaProps: spec.SchemaProps{ + Description: "Parameters declares parameters passed to this step action.", + Type: []string{"array"}, + Items: &spec.SchemaOrArray{ + Schema: &spec.Schema{ + SchemaProps: spec.SchemaProps{ + Default: map[string]interface{}{}, + Ref: ref("github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.Param"), + }, + }, + }, + }, + }, }, Required: []string{"name"}, }, }, Dependencies: []string{ - "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.Ref", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.StepOutputConfig", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.WorkspaceUsage", "k8s.io/api/core/v1.ContainerPort", "k8s.io/api/core/v1.EnvFromSource", "k8s.io/api/core/v1.EnvVar", "k8s.io/api/core/v1.Lifecycle", "k8s.io/api/core/v1.Probe", "k8s.io/api/core/v1.ResourceRequirements", "k8s.io/api/core/v1.SecurityContext", "k8s.io/api/core/v1.VolumeDevice", "k8s.io/api/core/v1.VolumeMount", "k8s.io/apimachinery/pkg/apis/meta/v1.Duration"}, + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.Param", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.Ref", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.StepOutputConfig", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.WorkspaceUsage", "k8s.io/api/core/v1.ContainerPort", "k8s.io/api/core/v1.EnvFromSource", "k8s.io/api/core/v1.EnvVar", "k8s.io/api/core/v1.Lifecycle", "k8s.io/api/core/v1.Probe", "k8s.io/api/core/v1.ResourceRequirements", "k8s.io/api/core/v1.SecurityContext", "k8s.io/api/core/v1.VolumeDevice", "k8s.io/api/core/v1.VolumeMount", "k8s.io/apimachinery/pkg/apis/meta/v1.Duration"}, } } diff --git a/pkg/apis/pipeline/v1beta1/swagger.json b/pkg/apis/pipeline/v1beta1/swagger.json index 5fc165feec6..4d0b5522d38 100644 --- a/pkg/apis/pipeline/v1beta1/swagger.json +++ b/pkg/apis/pipeline/v1beta1/swagger.json @@ -2070,6 +2070,15 @@ "description": "OnError defines the exiting behavior of a container on error can be set to [ continue | stopAndFail ]", "type": "string" }, + "params": { + "description": "Parameters declares parameters passed to this step action.", + "type": "array", + "items": { + "default": {}, + "$ref": "#/definitions/v1beta1.Param" + }, + "x-kubernetes-list-type": "atomic" + }, "ports": { "description": "List of ports to expose from the Step's container. Exposing a port here gives the system additional information about the network connections a container uses, but is primarily informational. Not specifying a port here DOES NOT prevent that port from being exposed. Any port which is listening on the default \"0.0.0.0\" address inside a container will be accessible from the network. Cannot be updated.\n\nDeprecated: This field will be removed in a future release.", "type": "array", diff --git a/pkg/apis/pipeline/v1beta1/task_conversion_test.go b/pkg/apis/pipeline/v1beta1/task_conversion_test.go index 043b595e2cc..5ec20da9acb 100644 --- a/pkg/apis/pipeline/v1beta1/task_conversion_test.go +++ b/pkg/apis/pipeline/v1beta1/task_conversion_test.go @@ -83,6 +83,9 @@ spec: steps: - ref: name: "step-action" + params: + - name: param1 + value: hello ` remoteStepActionTaskYAML := ` diff --git a/pkg/apis/pipeline/v1beta1/task_validation.go b/pkg/apis/pipeline/v1beta1/task_validation.go index 62944aec850..e623acbe115 100644 --- a/pkg/apis/pipeline/v1beta1/task_validation.go +++ b/pkg/apis/pipeline/v1beta1/task_validation.go @@ -306,6 +306,12 @@ func validateStep(ctx context.Context, s Step, names sets.String) (errs *apis.Fi }) } } else { + if len(s.Params) > 0 { + errs = errs.Also(&apis.FieldError{ + Message: "params cannot be used without Ref", + Paths: []string{"params"}, + }) + } if s.Image == "" { errs = errs.Also(apis.ErrMissingField("Image")) } diff --git a/pkg/apis/pipeline/v1beta1/task_validation_test.go b/pkg/apis/pipeline/v1beta1/task_validation_test.go index f47263654d3..4d09149fa7e 100644 --- a/pkg/apis/pipeline/v1beta1/task_validation_test.go +++ b/pkg/apis/pipeline/v1beta1/task_validation_test.go @@ -538,6 +538,16 @@ func TestTaskSpecStepActionReferenceValidate(t *testing.T) { Name: "stepAction", }, }}, + }, { + name: "valid use of params with Ref", + Steps: []v1beta1.Step{{ + Ref: &v1beta1.Ref{ + Name: "stepAction", + }, + Params: v1beta1.Params{{ + Name: "param", + }}, + }}, }} for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { @@ -1503,6 +1513,19 @@ func TestTaskSpecValidateErrorWithStepActionRef(t *testing.T) { Message: "env cannot be used with Ref", Paths: []string{"steps[0].env"}, }, + }, { + name: "Cannot use params without Ref", + Steps: []v1beta1.Step{{ + Image: "my-image", + Params: v1beta1.Params{{ + Name: "param", + }}, + }}, + enableStepActions: true, + expectedError: apis.FieldError{ + Message: "params cannot be used without Ref", + Paths: []string{"steps[0].params"}, + }, }} for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { diff --git a/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go b/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go index 060142c2e55..923c113107c 100644 --- a/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go +++ b/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go @@ -1762,6 +1762,13 @@ func (in *Step) DeepCopyInto(out *Step) { *out = new(Ref) (*in).DeepCopyInto(*out) } + if in.Params != nil { + in, out := &in.Params, &out.Params + *out = make(Params, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } return } diff --git a/pkg/reconciler/pipelinerun/resources/apply.go b/pkg/reconciler/pipelinerun/resources/apply.go index 39210369f9c..5ed091dd11f 100644 --- a/pkg/reconciler/pipelinerun/resources/apply.go +++ b/pkg/reconciler/pipelinerun/resources/apply.go @@ -367,9 +367,9 @@ func propagateParams(t v1.PipelineTask, stringReplacements map[string]string, ar } } } - t.TaskSpec.TaskSpec = *resources.ApplyReplacements(&t.TaskSpec.TaskSpec, stringReplacementsDup, arrayReplacementsDup) + t.TaskSpec.TaskSpec = *resources.ApplyReplacements(&t.TaskSpec.TaskSpec, stringReplacementsDup, arrayReplacementsDup, objectReplacementsDup) } else { - t.TaskSpec.TaskSpec = *resources.ApplyReplacements(&t.TaskSpec.TaskSpec, stringReplacements, arrayReplacements) + t.TaskSpec.TaskSpec = *resources.ApplyReplacements(&t.TaskSpec.TaskSpec, stringReplacements, arrayReplacements, objectReplacements) } return t } @@ -395,7 +395,7 @@ func PropagateResults(rpt *ResolvedPipelineTask, runStates PipelineRunState) { } } } - rpt.ResolvedTask.TaskSpec = resources.ApplyReplacements(rpt.ResolvedTask.TaskSpec, stringReplacements, arrayReplacements) + rpt.ResolvedTask.TaskSpec = resources.ApplyReplacements(rpt.ResolvedTask.TaskSpec, stringReplacements, arrayReplacements, map[string]map[string]string{}) } // ApplyTaskResultsToPipelineResults applies the results of completed TasksRuns and Runs to a Pipeline's diff --git a/pkg/reconciler/taskrun/resources/apply.go b/pkg/reconciler/taskrun/resources/apply.go index f99ac510e80..9c3273dc3be 100644 --- a/pkg/reconciler/taskrun/resources/apply.go +++ b/pkg/reconciler/taskrun/resources/apply.go @@ -46,14 +46,74 @@ var ( } ) -// ApplyParameters applies the params from a TaskRun.Input.Parameters to a TaskSpec -func ApplyParameters(ctx context.Context, spec *v1.TaskSpec, tr *v1.TaskRun, defaults ...v1.ParamSpec) *v1.TaskSpec { +// applyStepActionParameters applies the params from a TaskRun.Input.Parameters to a TaskSpec +func applyStepActionParameters(ctx context.Context, step *v1.Step, spec *v1.TaskSpec, tr *v1.TaskRun, stepParams v1.Params, defaults []v1.ParamSpec) *v1.Step { + + if stepParams != nil { + stringR, arrayR, objectR := getTaskParameters(ctx, spec, tr, spec.Params...) + stepParams = stepParams.ReplaceVariables(stringR, arrayR, objectR) + } + stringReplacements := map[string]string{} + arrayReplacements := map[string][]string{} + objectReplacements := map[string]map[string]string{} + + // Set all the default stringReplacements + for _, p := range defaults { + if p.Default != nil { + switch p.Default.Type { + case v1.ParamTypeArray: + for _, pattern := range paramPatterns { + for i := 0; i < len(p.Default.ArrayVal); i++ { + stringReplacements[fmt.Sprintf(pattern+"[%d]", p.Name, i)] = p.Default.ArrayVal[i] + } + arrayReplacements[fmt.Sprintf(pattern, p.Name)] = p.Default.ArrayVal + } + case v1.ParamTypeObject: + for _, pattern := range paramPatterns { + objectReplacements[fmt.Sprintf(pattern, p.Name)] = p.Default.ObjectVal + } + for k, v := range p.Default.ObjectVal { + stringReplacements[fmt.Sprintf(objectIndividualVariablePattern, p.Name, k)] = v + } + case v1.ParamTypeString: + fallthrough + default: + for _, pattern := range paramPatterns { + stringReplacements[fmt.Sprintf(pattern, p.Name)] = p.Default.StringVal + } + } + } + } + + // Set and overwrite params with the ones from the Step + stepStrings, stepArrays, stepObjects := paramsFromStep(ctx, stepParams) + for k, v := range stepStrings { + stringReplacements[k] = v + } + for k, v := range stepArrays { + arrayReplacements[k] = v + } + for k, v := range stepObjects { + for key, val := range v { + if objectReplacements != nil && objectReplacements[k] != nil { + objectReplacements[k][key] = val + } + } + } + + container.ApplyStepReplacements(step, stringReplacements, arrayReplacements) + return step +} + +// getTaskParameters applies the params from a TaskRun.Input.Parameters to a TaskSpec +func getTaskParameters(ctx context.Context, spec *v1.TaskSpec, tr *v1.TaskRun, defaults ...v1.ParamSpec) (map[string]string, map[string][]string, map[string]map[string]string) { // This assumes that the TaskRun inputs have been validated against what the Task requests. // stringReplacements is used for standard single-string stringReplacements, while arrayReplacements contains arrays // that need to be further processed. stringReplacements := map[string]string{} arrayReplacements := map[string][]string{} + objectReplacements := map[string]map[string]string{} // Set all the default stringReplacements for _, p := range defaults { @@ -67,6 +127,9 @@ func ApplyParameters(ctx context.Context, spec *v1.TaskSpec, tr *v1.TaskRun, def arrayReplacements[fmt.Sprintf(pattern, p.Name)] = p.Default.ArrayVal } case v1.ParamTypeObject: + for _, pattern := range paramPatterns { + objectReplacements[fmt.Sprintf(pattern, p.Name)] = p.Default.ObjectVal + } for k, v := range p.Default.ObjectVal { stringReplacements[fmt.Sprintf(objectIndividualVariablePattern, p.Name, k)] = v } @@ -80,22 +143,73 @@ func ApplyParameters(ctx context.Context, spec *v1.TaskSpec, tr *v1.TaskRun, def } } // Set and overwrite params with the ones from the TaskRun - trStrings, trArrays := paramsFromTaskRun(ctx, tr) + trStrings, trArrays, trObjects := paramsFromTaskRun(ctx, tr) for k, v := range trStrings { stringReplacements[k] = v } for k, v := range trArrays { arrayReplacements[k] = v } + for k, v := range trObjects { + for key, val := range v { + if objectReplacements != nil { + if objectReplacements[k] != nil { + objectReplacements[k][key] = val + } else { + objectReplacements[k] = v + } + } + } + } + return stringReplacements, arrayReplacements, objectReplacements +} - return ApplyReplacements(spec, stringReplacements, arrayReplacements) +// ApplyParameters applies the params from a TaskRun.Input.Parameters to a TaskSpec +func ApplyParameters(ctx context.Context, spec *v1.TaskSpec, tr *v1.TaskRun, defaults ...v1.ParamSpec) *v1.TaskSpec { + stringReplacements, arrayReplacements, objectReplacements := getTaskParameters(ctx, spec, tr, defaults...) + return ApplyReplacements(spec, stringReplacements, arrayReplacements, objectReplacements) +} + +func paramsFromStep(ctx context.Context, stepParams v1.Params) (map[string]string, map[string][]string, map[string]map[string]string) { + // stringReplacements is used for standard single-string stringReplacements, while arrayReplacements contains arrays + // that need to be further processed. + stringReplacements := map[string]string{} + arrayReplacements := map[string][]string{} + objectReplacements := map[string]map[string]string{} + + for _, p := range stepParams { + switch p.Value.Type { + case v1.ParamTypeArray: + for _, pattern := range paramPatterns { + for i := 0; i < len(p.Value.ArrayVal); i++ { + stringReplacements[fmt.Sprintf(pattern+"[%d]", p.Name, i)] = p.Value.ArrayVal[i] + } + arrayReplacements[fmt.Sprintf(pattern, p.Name)] = p.Value.ArrayVal + } + case v1.ParamTypeObject: + for _, pattern := range paramPatterns { + objectReplacements[fmt.Sprintf(pattern, p.Name)] = p.Value.ObjectVal + } + for k, v := range p.Value.ObjectVal { + stringReplacements[fmt.Sprintf(objectIndividualVariablePattern, p.Name, k)] = v + } + case v1.ParamTypeString: + fallthrough + default: + for _, pattern := range paramPatterns { + stringReplacements[fmt.Sprintf(pattern, p.Name)] = p.Value.StringVal + } + } + } + return stringReplacements, arrayReplacements, objectReplacements } -func paramsFromTaskRun(ctx context.Context, tr *v1.TaskRun) (map[string]string, map[string][]string) { +func paramsFromTaskRun(ctx context.Context, tr *v1.TaskRun) (map[string]string, map[string][]string, map[string]map[string]string) { // stringReplacements is used for standard single-string stringReplacements, while arrayReplacements contains arrays // that need to be further processed. stringReplacements := map[string]string{} arrayReplacements := map[string][]string{} + objectReplacements := map[string]map[string]string{} for _, p := range tr.Spec.Params { switch p.Value.Type { @@ -107,6 +221,9 @@ func paramsFromTaskRun(ctx context.Context, tr *v1.TaskRun) (map[string]string, arrayReplacements[fmt.Sprintf(pattern, p.Name)] = p.Value.ArrayVal } case v1.ParamTypeObject: + for _, pattern := range paramPatterns { + objectReplacements[fmt.Sprintf(pattern, p.Name)] = p.Value.ObjectVal + } for k, v := range p.Value.ObjectVal { stringReplacements[fmt.Sprintf(objectIndividualVariablePattern, p.Name, k)] = v } @@ -119,7 +236,7 @@ func paramsFromTaskRun(ctx context.Context, tr *v1.TaskRun) (map[string]string, } } - return stringReplacements, arrayReplacements + return stringReplacements, arrayReplacements, objectReplacements } func getContextReplacements(taskName string, tr *v1.TaskRun) map[string]string { @@ -135,7 +252,7 @@ func getContextReplacements(taskName string, tr *v1.TaskRun) map[string]string { // ApplyContexts applies the substitution from $(context.(taskRun|task).*) with the specified values. // Uses "" as a default if a value is not available. func ApplyContexts(spec *v1.TaskSpec, taskName string, tr *v1.TaskRun) *v1.TaskSpec { - return ApplyReplacements(spec, getContextReplacements(taskName, tr), map[string][]string{}) + return ApplyReplacements(spec, getContextReplacements(taskName, tr), map[string][]string{}, map[string]map[string]string{}) } // ApplyWorkspaces applies the substitution from paths that the workspaces in declarations mounted to, the @@ -170,7 +287,7 @@ func ApplyWorkspaces(ctx context.Context, spec *v1.TaskSpec, declarations []v1.W stringReplacements[fmt.Sprintf("workspaces.%s.claim", binding.Name)] = "" } } - return ApplyReplacements(spec, stringReplacements, map[string][]string{}) + return ApplyReplacements(spec, stringReplacements, map[string][]string{}, map[string]map[string]string{}) } // applyWorkspaceMountPath accepts a workspace path variable of the form $(workspaces.foo.path) and replaces @@ -204,7 +321,7 @@ func applyWorkspaceMountPath(variable string, spec *v1.TaskSpec, declaration v1. // Replace any remaining instances of the workspace path variable, which should fall // back to the mount path specified in the declaration. stringReplacements[variable] = defaultMountPath - return ApplyReplacements(spec, stringReplacements, emptyArrayReplacements) + return ApplyReplacements(spec, stringReplacements, emptyArrayReplacements, map[string]map[string]string{}) } // ApplyTaskResults applies the substitution from values in results which are referenced in spec as subitems @@ -223,7 +340,7 @@ func ApplyTaskResults(spec *v1.TaskSpec) *v1.TaskSpec { stringReplacements[fmt.Sprintf(pattern, result.Name)] = filepath.Join(pipeline.DefaultResultPath, result.Name) } } - return ApplyReplacements(spec, stringReplacements, map[string][]string{}) + return ApplyReplacements(spec, stringReplacements, map[string][]string{}, map[string]map[string]string{}) } // ApplyStepExitCodePath replaces the occurrences of exitCode path with the absolute tekton internal path @@ -235,7 +352,7 @@ func ApplyStepExitCodePath(spec *v1.TaskSpec) *v1.TaskSpec { stringReplacements[fmt.Sprintf("steps.%s.exitCode.path", pod.StepName(step.Name, i))] = filepath.Join(pipeline.StepsDir, pod.StepName(step.Name, i), "exitCode") } - return ApplyReplacements(spec, stringReplacements, map[string][]string{}) + return ApplyReplacements(spec, stringReplacements, map[string][]string{}, map[string]map[string]string{}) } // ApplyCredentialsPath applies a substitution of the key $(credentials.path) with the path that credentials @@ -244,16 +361,19 @@ func ApplyCredentialsPath(spec *v1.TaskSpec, path string) *v1.TaskSpec { stringReplacements := map[string]string{ "credentials.path": path, } - return ApplyReplacements(spec, stringReplacements, map[string][]string{}) + return ApplyReplacements(spec, stringReplacements, map[string][]string{}, map[string]map[string]string{}) } // ApplyReplacements replaces placeholders for declared parameters with the specified replacements. -func ApplyReplacements(spec *v1.TaskSpec, stringReplacements map[string]string, arrayReplacements map[string][]string) *v1.TaskSpec { +func ApplyReplacements(spec *v1.TaskSpec, stringReplacements map[string]string, arrayReplacements map[string][]string, objectReplacements map[string]map[string]string) *v1.TaskSpec { spec = spec.DeepCopy() // Apply variable expansion to steps fields. steps := spec.Steps for i := range steps { + if steps[i].Params != nil { + steps[i].Params = steps[i].Params.ReplaceVariables(stringReplacements, arrayReplacements, objectReplacements) + } container.ApplyStepReplacements(&steps[i], stringReplacements, arrayReplacements) } diff --git a/pkg/reconciler/taskrun/resources/apply_test.go b/pkg/reconciler/taskrun/resources/apply_test.go index 625041411a6..61bd533afdf 100644 --- a/pkg/reconciler/taskrun/resources/apply_test.go +++ b/pkg/reconciler/taskrun/resources/apply_test.go @@ -175,6 +175,26 @@ var ( }}, } + stepParamTaskSpec = &v1.TaskSpec{ + Params: v1.ParamSpecs{{ + Name: "myObject", + Default: &v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key1": "key1"}, + }, + }}, + Steps: []v1.Step{{ + Name: "foo", + Ref: &v1.Ref{ + Name: "stepAction", + }, + Params: v1.Params{{ + Name: "myObject", + Value: *v1.NewStructuredValues("$(params.myObject[*])"), + }}, + }}, + } + // a taskspec for testing object var in all places i.e. Sidecars, StepTemplate, Steps and Volumns objectParamTaskSpec = &v1.TaskSpec{ Sidecars: []v1.Sidecar{{ @@ -933,6 +953,42 @@ func TestApplyObjectParameters(t *testing.T) { } } +func TestApplyStepParameters(t *testing.T) { + // define the taskrun to test values provided by taskrun can overwrite the values provided in spec's default + tr := &v1.TaskRun{ + Spec: v1.TaskRunSpec{ + Params: []v1.Param{{ + Name: "myObject", + Value: *v1.NewObject(map[string]string{ + "key1": "taskrun-value-for-key1", + }), + }}, + TaskSpec: stepParamTaskSpec, + }, + } + dp := []v1.ParamSpec{{ + Name: "myObject", + Default: &v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key1": "key1"}, + }, + }} + + want := applyMutation(stepParamTaskSpec, func(spec *v1.TaskSpec) { + spec.Steps[0].Params = []v1.Param{{ + Name: "myObject", + Value: v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key1": "taskrun-value-for-key1"}, + }, + }} + }) + got := resources.ApplyParameters(context.Background(), stepParamTaskSpec, tr, dp...) + if d := cmp.Diff(want, got); d != "" { + t.Errorf("ApplyParameters() got diff %s", diff.PrintWantGot(d)) + } +} + func TestApplyWorkspaces(t *testing.T) { names.TestingSeed() ts := &v1.TaskSpec{ diff --git a/pkg/reconciler/taskrun/resources/taskref.go b/pkg/reconciler/taskrun/resources/taskref.go index a22ee654e78..c94f97de332 100644 --- a/pkg/reconciler/taskrun/resources/taskref.go +++ b/pkg/reconciler/taskrun/resources/taskref.go @@ -97,7 +97,7 @@ func GetTaskFunc(ctx context.Context, k8s kubernetes.Interface, tekton clientset return func(ctx context.Context, name string) (*v1.Task, *v1.RefSource, *trustedresources.VerificationResult, error) { var replacedParams v1.Params if ownerAsTR, ok := owner.(*v1.TaskRun); ok { - stringReplacements, arrayReplacements := paramsFromTaskRun(ctx, ownerAsTR) + stringReplacements, arrayReplacements, _ := paramsFromTaskRun(ctx, ownerAsTR) for k, v := range getContextReplacements("", ownerAsTR) { stringReplacements[k] = v } diff --git a/pkg/reconciler/taskrun/resources/taskspec.go b/pkg/reconciler/taskrun/resources/taskspec.go index 4cc85145bb9..174e6b2db08 100644 --- a/pkg/reconciler/taskrun/resources/taskspec.go +++ b/pkg/reconciler/taskrun/resources/taskspec.go @@ -111,6 +111,8 @@ func GetStepActionsData(ctx context.Context, taskSpec v1.TaskSpec, tr *v1.TaskRu return nil, err } stepActionSpec := stepAction.StepActionSpec() + stepActionSpec.SetDefaults(ctx) + s.Image = stepActionSpec.Image if len(stepActionSpec.Command) > 0 { s.Command = stepActionSpec.Command @@ -124,6 +126,11 @@ func GetStepActionsData(ctx context.Context, taskSpec v1.TaskSpec, tr *v1.TaskRu if stepActionSpec.Env != nil { s.Env = stepActionSpec.Env } + if err := validateStepHasStepActionParameters(ctx, s.Params, stepActionSpec.Params); err != nil { + return nil, err + } + s = applyStepActionParameters(ctx, s, &taskSpec, tr, s.Params, stepActionSpec.Params) + s.Params = nil steps = append(steps, *s) } else { steps = append(steps, step) diff --git a/pkg/reconciler/taskrun/resources/taskspec_test.go b/pkg/reconciler/taskrun/resources/taskspec_test.go index 37c174bcb35..9fe10905c59 100644 --- a/pkg/reconciler/taskrun/resources/taskspec_test.go +++ b/pkg/reconciler/taskrun/resources/taskspec_test.go @@ -454,6 +454,292 @@ func TestGetStepActionsData(t *testing.T) { Image: "foo", Command: []string{"ls"}, }}, + }, { + name: "params propagated from taskrun", + tr: &v1.TaskRun{ + ObjectMeta: metav1.ObjectMeta{ + Name: "mytaskrun", + Namespace: "default", + }, + Spec: v1.TaskRunSpec{ + Params: v1.Params{{ + Name: "stringparam", + Value: v1.ParamValue{ + Type: v1.ParamTypeString, + StringVal: "taskrun string param", + }, + }, { + Name: "arrayparam", + Value: v1.ParamValue{ + Type: v1.ParamTypeArray, + ArrayVal: []string{"taskrun", "array", "param"}, + }, + }, { + Name: "objectparam", + Value: v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key": "taskrun object param"}, + }, + }}, + TaskSpec: &v1.TaskSpec{ + Steps: []v1.Step{{ + Ref: &v1.Ref{ + Name: "stepAction", + }, + Params: v1.Params{{ + Name: "string-param", + Value: *v1.NewStructuredValues("$(params.stringparam)"), + }, { + Name: "array-param", + Value: *v1.NewStructuredValues("$(params.arrayparam[*])"), + }, { + Name: "object-param", + Value: *v1.NewStructuredValues("$(params.objectparam[*])"), + }}, + }}, + }, + }, + }, + stepAction: &v1alpha1.StepAction{ + ObjectMeta: metav1.ObjectMeta{ + Name: "stepAction", + Namespace: "default", + }, + Spec: v1alpha1.StepActionSpec{ + Image: "myimage", + Args: []string{"$(params.string-param)", "$(params.array-param[0])", "$(params.array-param[1])", "$(params.array-param[*])", "$(params.object-param.key)"}, + Params: v1.ParamSpecs{{ + Name: "string-param", + Type: v1.ParamTypeString, + }, { + Name: "array-param", + Type: v1.ParamTypeArray, + }, { + Name: "object-param", + Type: v1.ParamTypeObject, + Properties: map[string]v1.PropertySpec{"key": {Type: "string"}}, + }}, + }, + }, + want: []v1.Step{{ + Image: "myimage", + Args: []string{"taskrun string param", "taskrun", "array", "taskrun", "array", "param", "taskrun object param"}, + }}, + }, { + name: "params propagated from taskspec", + tr: &v1.TaskRun{ + ObjectMeta: metav1.ObjectMeta{ + Name: "mytaskrun", + Namespace: "default", + }, + Spec: v1.TaskRunSpec{ + TaskSpec: &v1.TaskSpec{ + Params: v1.ParamSpecs{{ + Name: "stringparam", + Default: &v1.ParamValue{ + Type: v1.ParamTypeString, + StringVal: "taskspec string param", + }, + }, { + Name: "arrayparam", + Default: &v1.ParamValue{ + Type: v1.ParamTypeArray, + ArrayVal: []string{"taskspec", "array", "param"}, + }, + }, { + Name: "objectparam", + Default: &v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key": "taskspec object param"}, + }, + }}, + Steps: []v1.Step{{ + Ref: &v1.Ref{ + Name: "stepAction", + }, + Params: v1.Params{{ + Name: "string-param", + Value: *v1.NewStructuredValues("$(params.stringparam)"), + }, { + Name: "array-param", + Value: *v1.NewStructuredValues("$(params.arrayparam[*])"), + }, { + Name: "object-param", + Value: *v1.NewStructuredValues("$(params.objectparam[*])"), + }}, + }}, + }, + }, + }, + stepAction: &v1alpha1.StepAction{ + ObjectMeta: metav1.ObjectMeta{ + Name: "stepAction", + Namespace: "default", + }, + Spec: v1alpha1.StepActionSpec{ + Image: "myimage", + Args: []string{"$(params.string-param)", "$(params.array-param[0])", "$(params.array-param[1])", "$(params.array-param[*])", "$(params.object-param.key)"}, + Params: v1.ParamSpecs{{ + Name: "string-param", + Type: v1.ParamTypeString, + }, { + Name: "array-param", + Type: v1.ParamTypeArray, + }, { + Name: "object-param", + Type: v1.ParamTypeObject, + Properties: map[string]v1.PropertySpec{"key": {Type: "string"}}, + }}, + }, + }, + want: []v1.Step{{ + Image: "myimage", + Args: []string{"taskspec string param", "taskspec", "array", "taskspec", "array", "param", "taskspec object param"}, + }}, + }, { + name: "params from step action defaults", + tr: &v1.TaskRun{ + ObjectMeta: metav1.ObjectMeta{ + Name: "mytaskrun", + Namespace: "default", + }, + Spec: v1.TaskRunSpec{ + TaskSpec: &v1.TaskSpec{ + Steps: []v1.Step{{ + Ref: &v1.Ref{ + Name: "stepAction", + }, + }}, + }, + }, + }, + stepAction: &v1alpha1.StepAction{ + ObjectMeta: metav1.ObjectMeta{ + Name: "stepAction", + Namespace: "default", + }, + Spec: v1alpha1.StepActionSpec{ + Image: "myimage", + Args: []string{"$(params.string-param)", "$(params.array-param[0])", "$(params.array-param[1])", "$(params.array-param[*])", "$(params.object-param.key)"}, + Params: v1.ParamSpecs{{ + Name: "string-param", + Type: v1.ParamTypeString, + Default: &v1.ParamValue{ + Type: v1.ParamTypeString, + StringVal: "step action string param", + }, + }, { + Name: "array-param", + Type: v1.ParamTypeArray, + Default: &v1.ParamValue{ + Type: v1.ParamTypeArray, + ArrayVal: []string{"step action", "array", "param"}, + }, + }, { + Name: "object-param", + Type: v1.ParamTypeObject, + Properties: map[string]v1.PropertySpec{"key": {Type: "string"}}, + Default: &v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key": "step action object param"}, + }, + }}, + }, + }, + want: []v1.Step{{ + Image: "myimage", + Args: []string{"step action string param", "step action", "array", "step action", "array", "param", "step action object param"}, + }}, + }, { + name: "params propagated partially from taskrun taskspec and stepaction", + tr: &v1.TaskRun{ + ObjectMeta: metav1.ObjectMeta{ + Name: "mytaskrun", + Namespace: "default", + }, + Spec: v1.TaskRunSpec{ + Params: v1.Params{{ + Name: "stringparam", + Value: v1.ParamValue{ + Type: v1.ParamTypeString, + StringVal: "taskrun string param", + }, + }, { + Name: "objectparam", + Value: v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key": "taskrun key"}, + }, + }}, + TaskSpec: &v1.TaskSpec{ + Params: v1.ParamSpecs{{ + Name: "arrayparam", + Default: &v1.ParamValue{ + Type: v1.ParamTypeArray, + ArrayVal: []string{"taskspec", "array", "param"}, + }, + }, { + Name: "objectparam", + Default: &v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key": "key1", "key2": "taskspec key2"}, + }, + }}, + Steps: []v1.Step{{ + Ref: &v1.Ref{ + Name: "stepAction", + }, + Params: v1.Params{{ + Name: "string-param", + Value: *v1.NewStructuredValues("$(params.stringparam)"), + }, { + Name: "array-param", + Value: *v1.NewStructuredValues("$(params.arrayparam[*])"), + }, { + Name: "object-param", + Value: *v1.NewStructuredValues("$(params.objectparam[*])"), + }}, + }}, + }, + }, + }, + stepAction: &v1alpha1.StepAction{ + ObjectMeta: metav1.ObjectMeta{ + Name: "stepAction", + Namespace: "default", + }, + Spec: v1alpha1.StepActionSpec{ + Image: "myimage", + Args: []string{"$(params.string-param)", "$(params.array-param[0])", "$(params.array-param[1])", "$(params.array-param[*])", "$(params.object-param.key)", "$(params.object-param.key2)", "$(params.object-param.key3)"}, + Params: v1.ParamSpecs{{ + Name: "string-param", + Type: v1.ParamTypeString, + Default: &v1.ParamValue{ + Type: v1.ParamTypeString, + StringVal: "step action string param", + }, + }, { + Name: "array-param", + Type: v1.ParamTypeArray, + Default: &v1.ParamValue{ + Type: v1.ParamTypeArray, + ArrayVal: []string{"step action", "array", "param"}, + }, + }, { + Name: "object-param", + Type: v1.ParamTypeObject, + Properties: map[string]v1.PropertySpec{"key": {Type: "string"}}, + Default: &v1.ParamValue{ + Type: v1.ParamTypeObject, + ObjectVal: map[string]string{"key": "step action key1", "key2": "step action key2", "key3": "step action key3"}, + }, + }}, + }, + }, + want: []v1.Step{{ + Image: "myimage", + Args: []string{"taskrun string param", "taskspec", "array", "taskspec", "array", "param", "taskrun key", "taskspec key2", "step action key3"}, + }}, }} for _, tt := range tests { ctx := context.Background() @@ -461,7 +747,7 @@ func TestGetStepActionsData(t *testing.T) { got, err := resources.GetStepActionsData(ctx, *tt.tr.Spec.TaskSpec, tt.tr, tektonclient) if err != nil { - t.Errorf("Did not expect an error but got : %s", err) + t.Fatalf("Did not expect an error but got : %s", err) } if d := cmp.Diff(tt.want, got); d != "" { t.Errorf("the taskSpec did not match what was expected diff: %s", diff.PrintWantGot(d)) @@ -471,10 +757,10 @@ func TestGetStepActionsData(t *testing.T) { func TestGetStepActionsData_Error(t *testing.T) { tests := []struct { - name string - tr *v1.TaskRun - stepActionFunc func(ctx context.Context, ref *v1.Ref) (*v1alpha1.StepAction, *v1.RefSource, error) - expectedError error + name string + tr *v1.TaskRun + stepAction *v1alpha1.StepAction + expectedError error }{{ name: "namespace missing error", tr: &v1.TaskRun{ @@ -491,10 +777,76 @@ func TestGetStepActionsData_Error(t *testing.T) { }, }, }, + stepAction: &v1alpha1.StepAction{}, expectedError: fmt.Errorf("must specify namespace to resolve reference to step action stepActionError"), + }, { + name: "params missing", + tr: &v1.TaskRun{ + ObjectMeta: metav1.ObjectMeta{ + Name: "mytaskrun", + Namespace: "default", + }, + Spec: v1.TaskRunSpec{ + TaskSpec: &v1.TaskSpec{ + Steps: []v1.Step{{ + Ref: &v1.Ref{ + Name: "stepaction", + }, + }}, + }, + }, + }, + stepAction: &v1alpha1.StepAction{ + ObjectMeta: metav1.ObjectMeta{ + Name: "stepaction", + Namespace: "default", + }, + Spec: v1alpha1.StepActionSpec{ + Image: "myimage", + Params: v1.ParamSpecs{{ + Name: "string-param", + Type: v1.ParamTypeString, + }}, + }, + }, + expectedError: fmt.Errorf("non-existent params in Step: [string-param]"), + }, { + name: "params extra", + tr: &v1.TaskRun{ + ObjectMeta: metav1.ObjectMeta{ + Name: "mytaskrun", + Namespace: "default", + }, + Spec: v1.TaskRunSpec{ + TaskSpec: &v1.TaskSpec{ + Steps: []v1.Step{{ + Ref: &v1.Ref{ + Name: "stepaction", + }, + Params: v1.Params{{ + Name: "string-param", + Value: *v1.NewStructuredValues("$(params.stringparam)"), + }}, + }}, + }, + }, + }, + stepAction: &v1alpha1.StepAction{ + ObjectMeta: metav1.ObjectMeta{ + Name: "stepaction", + Namespace: "default", + }, + Spec: v1alpha1.StepActionSpec{ + Image: "myimage", + }, + }, + expectedError: fmt.Errorf("extra params passed by Step to StepAction: [string-param]"), }} for _, tt := range tests { - _, err := resources.GetStepActionsData(context.Background(), *tt.tr.Spec.TaskSpec, tt.tr, nil) + ctx := context.Background() + tektonclient := fake.NewSimpleClientset(tt.stepAction) + + _, err := resources.GetStepActionsData(ctx, *tt.tr.Spec.TaskSpec, tt.tr, tektonclient) if err == nil { t.Fatalf("Expected to get an error but did not find any.") } diff --git a/pkg/reconciler/taskrun/resources/validate_params.go b/pkg/reconciler/taskrun/resources/validate_params.go index 345fe724164..6a75ee001dd 100644 --- a/pkg/reconciler/taskrun/resources/validate_params.go +++ b/pkg/reconciler/taskrun/resources/validate_params.go @@ -1,6 +1,7 @@ package resources import ( + "context" "fmt" v1 "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1" @@ -45,3 +46,40 @@ func ValidateOutOfBoundArrayParams(declarations v1.ParamSpecs, params v1.Params, } return nil } + +func validateStepHasStepActionParameters(ctx context.Context, stepParams v1.Params, stepActionDefaults []v1.ParamSpec) error { + stepActionParams := sets.String{} + requiredStepActionParams := []string{} + for _, sa := range stepActionDefaults { + stepActionParams.Insert(sa.Name) + if sa.Default == nil { + requiredStepActionParams = append(requiredStepActionParams, sa.Name) + } + } + + stepProvidedParams := sets.String{} + extra := []string{} + for _, sp := range stepParams { + stepProvidedParams.Insert(sp.Name) + if !stepActionParams.Has(sp.Name) { + // Extra parameter that is not needed + extra = append(extra, sp.Name) + } + } + if len(extra) > 0 { + return fmt.Errorf("extra params passed by Step to StepAction: %v", extra) + } + + missing := []string{} + + for _, requiredParam := range requiredStepActionParams { + if !stepProvidedParams.Has(requiredParam) { + // Missing required param + missing = append(missing, requiredParam) + } + } + if len(missing) > 0 { + return fmt.Errorf("non-existent params in Step: %v", missing) + } + return nil +} diff --git a/pkg/reconciler/taskrun/taskrun_test.go b/pkg/reconciler/taskrun/taskrun_test.go index b2813dd5a82..fbe2bfa77fe 100644 --- a/pkg/reconciler/taskrun/taskrun_test.go +++ b/pkg/reconciler/taskrun/taskrun_test.go @@ -2856,6 +2856,198 @@ spec: } } +func TestStepActionRefParams(t *testing.T) { + + tests := []struct { + name string + taskRun *v1.TaskRun + stepAction *v1alpha1.StepAction + want []v1.Step + }{{ + name: "params propagated from taskrun", + taskRun: parse.MustParseV1TaskRun(t, ` +metadata: + name: taskrun-with-string-params + namespace: foo +spec: + params: + - name: stringparam + value: "taskrun string param" + - name: arrayparam + value: ["taskrun", "array", "param"] + - name: objectparam + value: + key: taskrun object param + taskSpec: + steps: + - ref: + name: stepAction + name: step1 + params: + - name: string-param + value: $(params.stringparam) + - name: array-param + value: $(params.arrayparam[*]) + - name: object-param + value: $(params.objectparam[*]) +`), + stepAction: parse.MustParseV1alpha1StepAction(t, ` +metadata: + name: stepAction + namespace: foo +spec: + params: + - name: string-param + - name: array-param + type: array + - name: object-param + type: object + properties: + key: + type: string + image: myImage + command: ["echo"] + args: ["$(params.string-param)", "$(params.array-param[0])", "$(params.array-param[1])", "$(params.array-param[*])", "$(params.object-param.key)"] +`), + want: []v1.Step{{ + Image: "myImage", + Command: []string{"echo"}, + Args: []string{"taskrun string param", "taskrun", "array", "taskrun", "array", "param", "taskrun object param"}, + Name: "step1", + }}, + }, { + name: "params from taskspec", + taskRun: parse.MustParseV1TaskRun(t, ` +metadata: + name: taskrun-with-string-params + namespace: foo +spec: + taskSpec: + params: + - name: stringparam + default: "taskspec string param" + - name: arrayparam + default: ["taskspec", "array", "param"] + - name: objectparam + properties: + key: + type: string + default: + key: taskspec object param + steps: + - ref: + name: stepAction + name: step1 + params: + - name: string-param + value: $(params.stringparam) + - name: array-param + value: $(params.arrayparam[*]) + - name: object-param + value: $(params.objectparam[*]) +`), + stepAction: parse.MustParseV1alpha1StepAction(t, ` +metadata: + name: stepAction + namespace: foo +spec: + params: + - name: string-param + - name: array-param + type: array + - name: object-param + type: object + properties: + key: + type: string + image: myImage + command: ["echo"] + args: ["$(params.string-param)", "$(params.array-param[0])", "$(params.array-param[1])", "$(params.array-param[*])", "$(params.object-param.key)"] +`), + want: []v1.Step{{ + Image: "myImage", + Command: []string{"echo"}, + Args: []string{"taskspec string param", "taskspec", "array", "taskspec", "array", "param", "taskspec object param"}, + Name: "step1", + }}, + }, { + name: "params from step action defaults", + taskRun: parse.MustParseV1TaskRun(t, ` +metadata: + name: taskrun-with-string-params + namespace: foo +spec: + taskSpec: + steps: + - ref: + name: stepAction + name: step1 +`), + stepAction: parse.MustParseV1alpha1StepAction(t, ` +metadata: + name: stepAction + namespace: foo +spec: + params: + - name: string-param + type: string + default: "stepaction string param" + - name: array-param + type: array + default: + - stepaction + - array + - param + - name: object-param + type: object + properties: + key: + type: string + default: + key: "stepaction object param" + results: + - name: result + image: myImage + command: ["echo"] + args: ["$(params.string-param)", "$(params.array-param[0])", "$(params.array-param[1])", "$(params.array-param[*])", "$(params.object-param.key)"] +`), + want: []v1.Step{{ + Image: "myImage", + Command: []string{"echo"}, + Args: []string{"stepaction string param", "stepaction", "array", "stepaction", "array", "param", "stepaction object param"}, + Name: "step1", + }}, + }} + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + d := test.Data{ + TaskRuns: []*v1.TaskRun{tt.taskRun}, + StepActions: []*v1alpha1.StepAction{tt.stepAction}, + ConfigMaps: []*corev1.ConfigMap{ + { + ObjectMeta: metav1.ObjectMeta{Name: config.GetFeatureFlagsConfigName(), Namespace: system.Namespace()}, + Data: map[string]string{ + "enable-step-actions": "true", + }, + }, + }, + } + testAssets, cancel := getTaskRunController(t, d) + defer cancel() + createServiceAccount(t, testAssets, "default", tt.taskRun.Namespace) + c := testAssets.Controller + if err := c.Reconciler.Reconcile(testAssets.Ctx, getRunName(tt.taskRun)); err == nil { + t.Fatalf("Could not reconcile the taskrun: %v", err) + } + getTaskRun, _ := testAssets.Clients.Pipeline.TektonV1().TaskRuns(tt.taskRun.Namespace).Get(testAssets.Ctx, tt.taskRun.Name, metav1.GetOptions{}) + got := getTaskRun.Status.TaskSpec.Steps + if c := cmp.Diff(tt.want, got); c != "" { + t.Errorf("TestStepActionRefParams errored with: %s", diff.PrintWantGot(c)) + } + }) + } +} + func TestExpandMountPath(t *testing.T) { expectedMountPath := "/temppath/replaced" expectedReplacedArgs := fmt.Sprintf("replacedArgs - %s", expectedMountPath)