From 1124ca178387a0cf224bc6f08550c61597527230 Mon Sep 17 00:00:00 2001 From: Daniel Helfand Date: Wed, 19 Feb 2020 17:46:48 -0500 Subject: [PATCH] add containerstate and containername for sidecars --- pkg/apis/pipeline/v1alpha2/taskrun_types.go | 10 +- .../v1alpha2/zz_generated.deepcopy.go | 5 +- pkg/pod/status.go | 6 +- pkg/pod/status_test.go | 108 +++++++++++++- pkg/reconciler/taskrun/taskrun_test.go | 136 +++++++++++++++++- test/builder/sidecar.go | 67 +++++++++ test/builder/task.go | 14 -- test/builder/task_test.go | 20 ++- test/sidecar_test.go | 18 +++ 9 files changed, 351 insertions(+), 33 deletions(-) create mode 100644 test/builder/sidecar.go diff --git a/pkg/apis/pipeline/v1alpha2/taskrun_types.go b/pkg/apis/pipeline/v1alpha2/taskrun_types.go index a5d64bd1c99..4101b3b56c4 100644 --- a/pkg/apis/pipeline/v1alpha2/taskrun_types.go +++ b/pkg/apis/pipeline/v1alpha2/taskrun_types.go @@ -173,7 +173,7 @@ func (tr *TaskRunStatus) SetCondition(newCond *apis.Condition) { } } -// StepState reports the results of running a step in the Task. +// StepState reports the results of running a step in a Task. type StepState struct { corev1.ContainerState Name string `json:"name,omitempty"` @@ -181,10 +181,12 @@ type StepState struct { ImageID string `json:"imageID,omitempty"` } -// SidecarState reports the results of sidecar in the Task. +// SidecarState reports the results of running a sidecar in a Task. type SidecarState struct { - Name string `json:"name,omitempty"` - ImageID string `json:"imageID,omitempty"` + corev1.ContainerState + Name string `json:"name,omitempty"` + ContainerName string `json:"container,omitempty"` + ImageID string `json:"imageID,omitempty"` } // CloudEventDelivery is the target of a cloud event along with the state of diff --git a/pkg/apis/pipeline/v1alpha2/zz_generated.deepcopy.go b/pkg/apis/pipeline/v1alpha2/zz_generated.deepcopy.go index 3e79cb420bf..a7399d8a03c 100644 --- a/pkg/apis/pipeline/v1alpha2/zz_generated.deepcopy.go +++ b/pkg/apis/pipeline/v1alpha2/zz_generated.deepcopy.go @@ -920,6 +920,7 @@ func (in *PipelineTaskRun) DeepCopy() *PipelineTaskRun { // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *SidecarState) DeepCopyInto(out *SidecarState) { *out = *in + in.ContainerState.DeepCopyInto(&out.ContainerState) return } @@ -1402,7 +1403,9 @@ func (in *TaskRunStatusFields) DeepCopyInto(out *TaskRunStatusFields) { if in.Sidecars != nil { in, out := &in.Sidecars, &out.Sidecars *out = make([]SidecarState, len(*in)) - copy(*out, *in) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } } return } diff --git a/pkg/pod/status.go b/pkg/pod/status.go index 02ff4769abe..e55a08607dc 100644 --- a/pkg/pod/status.go +++ b/pkg/pod/status.go @@ -158,8 +158,10 @@ func MakeTaskRunStatus(tr v1alpha1.TaskRun, pod *corev1.Pod, taskSpec v1alpha1.T }) } else if isContainerSidecar(s.Name) { trs.Sidecars = append(trs.Sidecars, v1alpha1.SidecarState{ - Name: trimSidecarPrefix(s.Name), - ImageID: s.ImageID, + ContainerState: *s.State.DeepCopy(), + Name: trimSidecarPrefix(s.Name), + ContainerName: s.Name, + ImageID: s.ImageID, }) } } diff --git a/pkg/pod/status_test.go b/pkg/pod/status_test.go index 9215e116ca8..310867a42c6 100644 --- a/pkg/pod/status_test.go +++ b/pkg/pod/status_test.go @@ -491,8 +491,112 @@ func TestMakeTaskRunStatus(t *testing.T) { ContainerName: "step-running-step", }}, Sidecars: []v1alpha1.SidecarState{{ - Name: "running", - ImageID: "image-id", + ContainerState: corev1.ContainerState{ + Running: &corev1.ContainerStateRunning{}, + }, + Name: "running", + ImageID: "image-id", + ContainerName: "sidecar-running", + }}, + }, + }, + }, { + desc: "with-sidecar-waiting", + podStatus: corev1.PodStatus{ + Phase: corev1.PodRunning, + ContainerStatuses: []corev1.ContainerStatus{{ + Name: "step-waiting-step", + State: corev1.ContainerState{ + Waiting: &corev1.ContainerStateWaiting{ + Reason: "PodInitializing", + Message: "PodInitializing", + }, + }, + }, { + Name: "sidecar-waiting", + ImageID: "image-id", + State: corev1.ContainerState{ + Waiting: &corev1.ContainerStateWaiting{ + Reason: "PodInitializing", + Message: "PodInitializing", + }, + }, + Ready: true, + }}, + }, + want: v1alpha1.TaskRunStatus{ + Status: duckv1beta1.Status{ + Conditions: []apis.Condition{conditionRunning}, + }, + TaskRunStatusFields: v1alpha1.TaskRunStatusFields{ + Steps: []v1alpha1.StepState{{ + ContainerState: corev1.ContainerState{ + Waiting: &corev1.ContainerStateWaiting{ + Reason: "PodInitializing", + Message: "PodInitializing", + }, + }, + Name: "waiting-step", + ContainerName: "step-waiting-step", + }}, + Sidecars: []v1alpha1.SidecarState{{ + ContainerState: corev1.ContainerState{ + Waiting: &corev1.ContainerStateWaiting{ + Reason: "PodInitializing", + Message: "PodInitializing", + }, + }, + Name: "waiting", + ImageID: "image-id", + ContainerName: "sidecar-waiting", + }}, + }, + }, + }, { + desc: "with-sidecar-terminated", + podStatus: corev1.PodStatus{ + Phase: corev1.PodRunning, + ContainerStatuses: []corev1.ContainerStatus{{ + Name: "step-running-step", + State: corev1.ContainerState{ + Running: &corev1.ContainerStateRunning{}, + }, + }, { + Name: "sidecar-error", + ImageID: "image-id", + State: corev1.ContainerState{ + Terminated: &corev1.ContainerStateTerminated{ + ExitCode: 1, + Reason: "Error", + Message: "Error", + }, + }, + Ready: true, + }}, + }, + want: v1alpha1.TaskRunStatus{ + Status: duckv1beta1.Status{ + Conditions: []apis.Condition{conditionRunning}, + }, + TaskRunStatusFields: v1alpha1.TaskRunStatusFields{ + Steps: []v1alpha1.StepState{{ + ContainerState: corev1.ContainerState{ + Running: &corev1.ContainerStateRunning{}, + }, + Name: "running-step", + ContainerName: "step-running-step", + }}, + Sidecars: []v1alpha1.SidecarState{{ + ContainerState: corev1.ContainerState{ + Terminated: &corev1.ContainerStateTerminated{ + ExitCode: 1, + Reason: "Error", + Message: "Error", + }, + }, + Name: "error", + ImageID: "image-id", + ContainerName: "sidecar-error", }}, }, }, diff --git a/pkg/reconciler/taskrun/taskrun_test.go b/pkg/reconciler/taskrun/taskrun_test.go index a81bdcac8dc..c6c54f87ccd 100644 --- a/pkg/reconciler/taskrun/taskrun_test.go +++ b/pkg/reconciler/taskrun/taskrun_test.go @@ -99,6 +99,13 @@ var ( ), )) clustertask = tb.ClusterTask("test-cluster-task", tb.ClusterTaskSpec(simpleStep)) + taskSidecar = tb.Task("test-task-sidecar", "foo", tb.TaskSpec( + tb.Sidecar("sidecar", "image-id"), + )) + taskMultipleSidecars = tb.Task("test-task-sidecar", "foo", tb.TaskSpec( + tb.Sidecar("sidecar", "image-id"), + tb.Sidecar("sidecar2", "image-id"), + )) outputTask = tb.Task("test-output-task", "foo", tb.TaskSpec( simpleStep, tb.TaskInputs( @@ -1145,7 +1152,7 @@ func TestReconcilePodFetchError(t *testing.T) { return true, nil, errors.New("induce failure fetching pods") }) - if err := c.Reconciler.Reconcile(context.Background(), fmt.Sprintf("%s/%s", taskRun.Namespace, taskRun.Name)); err == nil { + if err := c.Reconciler.Reconcile(context.Background(), getRunName(taskRun)); err == nil { t.Fatal("expected error when reconciling a Task for which we couldn't get the corresponding Build Pod but got nil") } } @@ -1196,7 +1203,7 @@ func TestReconcilePodUpdateStatus(t *testing.T) { c := testAssets.Controller clients := testAssets.Clients - if err := c.Reconciler.Reconcile(context.Background(), fmt.Sprintf("%s/%s", taskRun.Namespace, taskRun.Name)); err != nil { + if err := c.Reconciler.Reconcile(context.Background(), getRunName(taskRun)); err != nil { t.Fatalf("Unexpected error when Reconcile() : %v", err) } newTr, err := clients.Pipeline.TektonV1alpha1().TaskRuns(taskRun.Namespace).Get(taskRun.Name, metav1.GetOptions{}) @@ -1219,7 +1226,7 @@ func TestReconcilePodUpdateStatus(t *testing.T) { if _, err := clients.Kube.CoreV1().Pods(taskRun.Namespace).UpdateStatus(pod); err != nil { t.Errorf("Unexpected error while updating build: %v", err) } - if err := c.Reconciler.Reconcile(context.Background(), fmt.Sprintf("%s/%s", taskRun.Namespace, taskRun.Name)); err != nil { + if err := c.Reconciler.Reconcile(context.Background(), getRunName(taskRun)); err != nil { t.Fatalf("Unexpected error when Reconcile(): %v", err) } @@ -1259,7 +1266,7 @@ func TestReconcileOnCompletedTaskRun(t *testing.T) { c := testAssets.Controller clients := testAssets.Clients - if err := c.Reconciler.Reconcile(context.Background(), fmt.Sprintf("%s/%s", taskRun.Namespace, taskRun.Name)); err != nil { + if err := c.Reconciler.Reconcile(context.Background(), getRunName(taskRun)); err != nil { t.Fatalf("Unexpected error when reconciling completed TaskRun : %v", err) } newTr, err := clients.Pipeline.TektonV1alpha1().TaskRuns(taskRun.Namespace).Get(taskRun.Name, metav1.GetOptions{}) @@ -1289,7 +1296,7 @@ func TestReconcileOnCancelledTaskRun(t *testing.T) { c := testAssets.Controller clients := testAssets.Clients - if err := c.Reconciler.Reconcile(context.Background(), fmt.Sprintf("%s/%s", taskRun.Namespace, taskRun.Name)); err != nil { + if err := c.Reconciler.Reconcile(context.Background(), getRunName(taskRun)); err != nil { t.Fatalf("Unexpected error when reconciling completed TaskRun : %v", err) } newTr, err := clients.Pipeline.TektonV1alpha1().TaskRuns(taskRun.Namespace).Get(taskRun.Name, metav1.GetOptions{}) @@ -1377,7 +1384,7 @@ func TestReconcileTimeouts(t *testing.T) { c := testAssets.Controller clients := testAssets.Clients - if err := c.Reconciler.Reconcile(context.Background(), fmt.Sprintf("%s/%s", tc.taskRun.Namespace, tc.taskRun.Name)); err != nil { + if err := c.Reconciler.Reconcile(context.Background(), getRunName(tc.taskRun)); err != nil { t.Fatalf("Unexpected error when reconciling completed TaskRun : %v", err) } newTr, err := clients.Pipeline.TektonV1alpha1().TaskRuns(tc.taskRun.Namespace).Get(tc.taskRun.Name, metav1.GetOptions{}) @@ -1881,3 +1888,120 @@ func TestUpdateTaskRunResourceResult_Errors(t *testing.T) { }) } } + +func TestReconcile_Single_SidecarState(t *testing.T) { + runningState := corev1.ContainerStateRunning{StartedAt: metav1.Time{Time: time.Now()}} + taskRun := tb.TaskRun("test-taskrun-sidecars", "foo", + tb.TaskRunSpec( + tb.TaskRunTaskRef(taskSidecar.Name), + ), + tb.TaskRunStatus( + tb.SidecarState( + tb.SidecarStateName("sidecar"), + tb.SidecarStateImageID("image-id"), + tb.SidecarStateContainerName("sidecar-sidecar"), + tb.SetSidecarStateRunning(runningState), + ), + ), + ) + + d := test.Data{ + TaskRuns: []*v1alpha1.TaskRun{taskRun}, + Tasks: []*v1alpha1.Task{taskSidecar}, + } + + testAssets, cancel := getTaskRunController(t, d) + defer cancel() + clients := testAssets.Clients + + if err := testAssets.Controller.Reconciler.Reconcile(context.Background(), getRunName(taskRun)); err != nil { + t.Errorf("expected no error reconciling valid TaskRun but got %v", err) + } + + getTaskRun, err := clients.Pipeline.TektonV1alpha1().TaskRuns(taskRun.Namespace).Get(taskRun.Name, metav1.GetOptions{}) + if err != nil { + t.Fatalf("Expected completed TaskRun %s to exist but instead got error when getting it: %v", taskRun.Name, err) + } + + expected := v1alpha1.SidecarState{ + Name: "sidecar", + ImageID: "image-id", + ContainerName: "sidecar-sidecar", + ContainerState: corev1.ContainerState{ + Running: &runningState, + }, + } + + if c := cmp.Diff(expected, getTaskRun.Status.Sidecars[0]); c != "" { + t.Errorf("TestReconcile_Single_SidecarState (-want, +got): %s", c) + } +} + +func TestReconcile_Multiple_SidecarStates(t *testing.T) { + runningState := corev1.ContainerStateRunning{StartedAt: metav1.Time{Time: time.Now()}} + waitingState := corev1.ContainerStateWaiting{Reason: "PodInitializing"} + taskRun := tb.TaskRun("test-taskrun-sidecars", "foo", + tb.TaskRunSpec( + tb.TaskRunTaskRef(taskMultipleSidecars.Name), + ), + tb.TaskRunStatus( + tb.SidecarState( + tb.SidecarStateName("sidecar1"), + tb.SidecarStateImageID("image-id"), + tb.SidecarStateContainerName("sidecar-sidecar1"), + tb.SetSidecarStateRunning(runningState), + ), + ), + tb.TaskRunStatus( + tb.SidecarState( + tb.SidecarStateName("sidecar2"), + tb.SidecarStateImageID("image-id"), + tb.SidecarStateContainerName("sidecar-sidecar2"), + tb.SetSidecarStateWaiting(waitingState), + ), + ), + ) + + d := test.Data{ + TaskRuns: []*v1alpha1.TaskRun{taskRun}, + Tasks: []*v1alpha1.Task{taskMultipleSidecars}, + } + + testAssets, cancel := getTaskRunController(t, d) + defer cancel() + clients := testAssets.Clients + + if err := testAssets.Controller.Reconciler.Reconcile(context.Background(), getRunName(taskRun)); err != nil { + t.Errorf("expected no error reconciling valid TaskRun but got %v", err) + } + + getTaskRun, err := clients.Pipeline.TektonV1alpha1().TaskRuns(taskRun.Namespace).Get(taskRun.Name, metav1.GetOptions{}) + if err != nil { + t.Fatalf("Expected completed TaskRun %s to exist but instead got error when getting it: %v", taskRun.Name, err) + } + + expected := []v1alpha1.SidecarState{ + { + Name: "sidecar1", + ImageID: "image-id", + ContainerName: "sidecar-sidecar1", + ContainerState: corev1.ContainerState{ + Running: &runningState, + }, + }, + { + Name: "sidecar2", + ImageID: "image-id", + ContainerName: "sidecar-sidecar2", + ContainerState: corev1.ContainerState{ + Waiting: &waitingState, + }, + }, + } + + for i, sc := range getTaskRun.Status.Sidecars { + if c := cmp.Diff(expected[i], sc); c != "" { + t.Errorf("TestReconcile_Multiple_SidecarStates sidecar%d (-want, +got): %s", i+1, c) + } + } +} diff --git a/test/builder/sidecar.go b/test/builder/sidecar.go new file mode 100644 index 00000000000..e177d97a9dd --- /dev/null +++ b/test/builder/sidecar.go @@ -0,0 +1,67 @@ +/* +Copyright 2020 The Tekton Authors +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + http://www.apache.org/licenses/LICENSE-2.0 +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +package builder + +import ( + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" + corev1 "k8s.io/api/core/v1" +) + +// SidecarStateName sets the name of the Sidecar for the SidecarState. +func SidecarStateName(name string) SidecarStateOp { + return func(s *v1alpha1.SidecarState) { + s.Name = name + } +} + +// SetSidecarStateImageID sets ImageID of Sidecar for SidecarState. +func SidecarStateImageID(imageID string) SidecarStateOp { + return func(s *v1alpha1.SidecarState) { + s.ImageID = imageID + } +} + +// SetSidecarStateImageID sets ImageID of Sidecar for SidecarState. +func SidecarStateContainerName(containerName string) SidecarStateOp { + return func(s *v1alpha1.SidecarState) { + s.ContainerName = containerName + } +} + +// SetSidecarStateTerminated sets Terminated state of a Sidecar. +func SetSidecarStateTerminated(terminated corev1.ContainerStateTerminated) SidecarStateOp { + return func(s *v1alpha1.SidecarState) { + s.ContainerState = corev1.ContainerState{ + Terminated: &terminated, + } + } +} + +// SetSidecarStateRunning sets Running state of a Sidecar. +func SetSidecarStateRunning(running corev1.ContainerStateRunning) SidecarStateOp { + return func(s *v1alpha1.SidecarState) { + s.ContainerState = corev1.ContainerState{ + Running: &running, + } + } +} + +// SetSidecarStateWaiting sets Waiting state of a Sidecar. +func SetSidecarStateWaiting(waiting corev1.ContainerStateWaiting) SidecarStateOp { + return func(s *v1alpha1.SidecarState) { + s.ContainerState = corev1.ContainerState{ + Waiting: &waiting, + } + } +} diff --git a/test/builder/task.go b/test/builder/task.go index ce3f7a83f86..d7652666b25 100644 --- a/test/builder/task.go +++ b/test/builder/task.go @@ -483,20 +483,6 @@ func SetStepStateWaiting(waiting corev1.ContainerStateWaiting) StepStateOp { } } -// SetSidecarStateName sets Name of Sidecar for SidecarState. -func SetSidecarStateName(name string) SidecarStateOp { - return func(s *v1alpha1.SidecarState) { - s.Name = name - } -} - -// SetSidecarStateImageID sets ImageID of Sidecar for SidecarState. -func SetSidecarStateImageID(imageID string) SidecarStateOp { - return func(s *v1alpha1.SidecarState) { - s.ImageID = imageID - } -} - // TaskRunOwnerReference sets the OwnerReference, with specified kind and name, to the TaskRun. func TaskRunOwnerReference(kind, name string, ops ...OwnerReferenceOp) TaskRunOp { return func(tr *v1alpha1.TaskRun) { diff --git a/test/builder/task_test.go b/test/builder/task_test.go index e2e8589c74f..e00fc015516 100644 --- a/test/builder/task_test.go +++ b/test/builder/task_test.go @@ -163,6 +163,7 @@ func TestClusterTask(t *testing.T) { func TestTaskRunWithTaskRef(t *testing.T) { var trueB = true + terminatedState := corev1.ContainerStateTerminated{Reason: "Completed"} taskRun := tb.TaskRun("test-taskrun", "foo", tb.TaskRunOwnerReference("PipelineRun", "test", @@ -203,8 +204,10 @@ func TestTaskRunWithTaskRef(t *testing.T) { tb.StatusCondition(apis.Condition{Type: apis.ConditionSucceeded}), tb.StepState(tb.StateTerminated(127)), tb.SidecarState( - tb.SetSidecarStateName("sidecar"), - tb.SetSidecarStateImageID("ImageID"), + tb.SidecarStateName("sidecar"), + tb.SidecarStateImageID("ImageID"), + tb.SidecarStateContainerName("ContainerName"), + tb.SetSidecarStateTerminated(terminatedState), ), ), ) @@ -285,8 +288,17 @@ func TestTaskRunWithTaskRef(t *testing.T) { Conditions: []apis.Condition{{Type: apis.ConditionSucceeded}}, }, TaskRunStatusFields: v1alpha1.TaskRunStatusFields{ - PodName: "my-pod-name", - Sidecars: []v1alpha2.SidecarState{{Name: "sidecar", ImageID: "ImageID"}}, + PodName: "my-pod-name", + Sidecars: []v1alpha2.SidecarState{{ + Name: "sidecar", + ImageID: "ImageID", + ContainerName: "ContainerName", + ContainerState: corev1.ContainerState{ + Terminated: &corev1.ContainerStateTerminated{ + Reason: "Completed", + }, + }, + }}, Steps: []v1alpha1.StepState{{ContainerState: corev1.ContainerState{ Terminated: &corev1.ContainerStateTerminated{ExitCode: 127}, }}}, diff --git a/test/sidecar_test.go b/test/sidecar_test.go index e65ef9dfc7d..048b499c919 100644 --- a/test/sidecar_test.go +++ b/test/sidecar_test.go @@ -139,6 +139,24 @@ func TestSidecarTaskSupport(t *testing.T) { if !primaryTerminated || !sidecarTerminated { t.Errorf("Either the primary or sidecar containers did not terminate") } + + trCheckSidecarStatus, err := clients.TaskRunClient.Get(sidecarTaskRunName, metav1.GetOptions{}) + if err != nil { + t.Errorf("Error getting TaskRun: %v", err) + } + + sidecarFromStatus := trCheckSidecarStatus.Status.Sidecar[0] + // Check if Sidecar ContainerName for SidecarStatus + if sidecarFromStatus.ContainerName != fmt.Sprintf("sidecar-%s", sidecarContainerName) { + t.Errorf("Sidecar ContainerName should be: %s", sidecarContainerName) + } + + // Check if Sidecar Terminated Status if Present for SidecarStatus + if sidecarFromStatus.Terminated == nil { + t.Errorf("Sidecar container has a nil Terminated status but non-nil is expected.") + } else if sidecarFromStatus.Terminated.Reason != "Completed" { + t.Errorf("Sidecar container has a nil Terminated status but non-nil is expected.") + } }) } }