From af8c8dc6b2e38ced639294b62fcb04577e382037 Mon Sep 17 00:00:00 2001 From: poy Date: Sun, 1 Dec 2019 07:12:50 -0700 Subject: [PATCH] Adds Env to TaskRunSpec Adds a `Env` field to `TaskRunSpec` to allow a user to set environment variables on each step container. This mimics Knative-Build's `TemplateInstantiationSpec`'s `Env` field. Setting environment variables is useful for when a `TaskRun` author may not know what variables are required ahead of time (e.g., Buildpacks). fixes #1606 --- Gopkg.lock | 2 - docs/taskruns.md | 17 +++++++ pkg/apis/pipeline/v1alpha1/taskrun_types.go | 4 +- .../v1alpha1/zz_generated.deepcopy.go | 7 +++ pkg/pod/pod.go | 8 +-- pkg/pod/pod_test.go | 50 ++++++++++++++++++- 6 files changed, 81 insertions(+), 7 deletions(-) diff --git a/Gopkg.lock b/Gopkg.lock index a8e7111b83f..5788a63544a 100644 --- a/Gopkg.lock +++ b/Gopkg.lock @@ -1356,10 +1356,8 @@ "github.com/google/go-containerregistry/pkg/v1", "github.com/google/go-containerregistry/pkg/v1/empty", "github.com/google/go-containerregistry/pkg/v1/layout", - "github.com/google/go-containerregistry/pkg/v1/partial", "github.com/google/go-containerregistry/pkg/v1/random", "github.com/google/go-containerregistry/pkg/v1/remote", - "github.com/google/go-containerregistry/pkg/v1/types", "github.com/hashicorp/go-multierror", "github.com/hashicorp/golang-lru", "github.com/jenkins-x/go-scm/scm", diff --git a/docs/taskruns.md b/docs/taskruns.md index 01de3efc7ac..d768405dcd0 100644 --- a/docs/taskruns.md +++ b/docs/taskruns.md @@ -17,6 +17,7 @@ A `TaskRun` runs until all `steps` have completed or until a failure occurs. - [Providing resources](#providing-resources) - [Overriding where resources are copied from](#overriding-where-resources-are-copied-from) - [Service Account](#service-account) + - [Environment Variables](#environment-variables) - [Pod Template](#pod-template) - [Status](#status) - [Steps](#steps) @@ -55,6 +56,8 @@ following fields: `timeout` is empty, the default timeout will be applied. If the value is set to 0, there is no timeout. You can also follow the instruction [here](#Configuring-default-timeout) to configure the default timeout. + - [`Env`] - Specifies [environment variables](#environment-variables) that + will be available to each step container. - [`podTemplate`](#pod-template) - Specifies a subset of [`PodSpec`](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.15/#pod-v1-core) configuration that will be used as the basis for the `Task` pod. @@ -167,6 +170,20 @@ of the `TaskRun` resource object. For examples and more information about specifying service accounts, see the [`ServiceAccount`](./auth.md) reference topic. +### Environment Variables + +Specifies environment variables to be set on each step container. They are an +array of type [`corev1.EnvVar`](https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.16/#envvar-v1-core). + +```yaml +spec: + env: + - name: FOO + value: bar + - name: BAZ + value: foo +``` + ## Pod Template Specifies a subset of diff --git a/pkg/apis/pipeline/v1alpha1/taskrun_types.go b/pkg/apis/pipeline/v1alpha1/taskrun_types.go index 1a676a7cbaf..06515436d36 100644 --- a/pkg/apis/pipeline/v1alpha1/taskrun_types.go +++ b/pkg/apis/pipeline/v1alpha1/taskrun_types.go @@ -48,9 +48,11 @@ type TaskRunSpec struct { // Refer Go's ParseDuration documentation for expected format: https://golang.org/pkg/time/#ParseDuration // +optional Timeout *metav1.Duration `json:"timeout,omitempty"` - // PodTemplate holds pod specific configuration PodTemplate PodTemplate `json:"podTemplate,omitempty"` + // Env, if specified will provide variables to all task steps. + // +optional + Env []corev1.EnvVar `json:"env,omitempty"` } // TaskRunSpecStatus defines the taskrun spec status the user can provide diff --git a/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go b/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go index 2d57103828e..44fede027a0 100644 --- a/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go +++ b/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go @@ -1717,6 +1717,13 @@ func (in *TaskRunSpec) DeepCopyInto(out *TaskRunSpec) { **out = **in } in.PodTemplate.DeepCopyInto(&out.PodTemplate) + if in.Env != nil { + in, out := &in.Env, &out.Env + *out = make([]v1.EnvVar, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } return } diff --git a/pkg/pod/pod.go b/pkg/pod/pod.go index 75b84259111..5cd9a7f6271 100644 --- a/pkg/pod/pod.go +++ b/pkg/pod/pod.go @@ -144,10 +144,12 @@ func MakePod(images pipeline.Images, taskRun *v1alpha1.TaskRun, taskSpec v1alpha } // Add implicit env vars. - // They're prepended to the list, so that if the user specified any - // themselves their value takes precedence. + // Append to an empty list to ensure we don't alter implicitEnvVars. + // Precedence: step.Env > taskRun.Spec.Env > implicitEnvVars for i, s := range stepContainers { - env := append(implicitEnvVars, s.Env...) + env := append([]corev1.EnvVar{}, implicitEnvVars...) + env = append(env, taskRun.Spec.Env...) + env = append(env, s.Env...) stepContainers[i].Env = env } diff --git a/pkg/pod/pod_test.go b/pkg/pod/pod_test.go index a4d3b3f8191..37379ffd819 100644 --- a/pkg/pod/pod_test.go +++ b/pkg/pod/pod_test.go @@ -573,7 +573,55 @@ script-heredoc-randomly-generated-6nl7g }}, Volumes: append(implicitVolumes, scriptsVolume, toolsVolume, downwardVolume), }, - }} { + }, { + desc: "env is set", + ts: v1alpha1.TaskSpec{ + Steps: []v1alpha1.Step{{Container: corev1.Container{ + Name: "step-with-env", + Image: "image", + Command: []string{"cmd"}, // avoid entrypoint lookup. + }}}, + }, + trs: v1alpha1.TaskRunSpec{ + Env: []corev1.EnvVar{ + {Name: "FOO", Value: "bar"}, + }, + }, + want: &corev1.PodSpec{ + RestartPolicy: corev1.RestartPolicyNever, + InitContainers: []corev1.Container{placeToolsInit}, + Containers: []corev1.Container{{ + Name: "step-step-with-env", + Image: "image", + Command: []string{"/tekton/tools/entrypoint"}, + Args: []string{ + "-wait_file", + "/tekton/downward/ready", + "-wait_file_content", + "-post_file", + "/tekton/tools/0", + "-entrypoint", + "cmd", + "--", + }, + // Append to an empty slice to avoid manipulating + // implicitEnvVars, but keep user variables last to ensure + // precedence. + Env: append(append([]corev1.EnvVar{}, implicitEnvVars...), corev1.EnvVar{Name: "FOO", Value: "bar"}), + VolumeMounts: append([]corev1.VolumeMount{toolsMount, downwardMount}, implicitVolumeMounts...), + WorkingDir: workspaceDir, + Resources: corev1.ResourceRequirements{ + Requests: corev1.ResourceList{ + corev1.ResourceCPU: resource.MustParse("0"), + corev1.ResourceMemory: resource.MustParse("0"), + corev1.ResourceEphemeralStorage: resource.MustParse("0"), + }, + }, + }}, + Volumes: append(implicitVolumes, toolsVolume, downwardVolume), + }, + }, + } { t.Run(c.desc, func(t *testing.T) { names.TestingSeed() kubeclient := fakek8s.NewSimpleClientset(