From 565a1117484aa0db0024dabbd8a95fc9908ab624 Mon Sep 17 00:00:00 2001 From: Kewei Yang Date: Sun, 8 Oct 2023 09:38:44 +0800 Subject: [PATCH] fix: the pr may lose finallyStartTime when pipeline controller is not synchronized to all current state --- pkg/reconciler/pipelinerun/pipelinerun.go | 6 + .../pipelinerun/pipelinerun_test.go | 251 ++++++++++++++++++ .../pipelinerun/resources/pipelinerunstate.go | 18 ++ 3 files changed, 275 insertions(+) diff --git a/pkg/reconciler/pipelinerun/pipelinerun.go b/pkg/reconciler/pipelinerun/pipelinerun.go index 15fb97aa1c6..b3b8685731f 100644 --- a/pkg/reconciler/pipelinerun/pipelinerun.go +++ b/pkg/reconciler/pipelinerun/pipelinerun.go @@ -799,6 +799,12 @@ func (c *Reconciler) runNextSchedulableTask(ctx context.Context, pr *v1.Pipeline } } + // If FinallyStartTime is not set, and one or more final tasks has been created + // Try to set the FinallyStartTime of this PipelineRun + if pr.Status.FinallyStartTime == nil && pipelineRunFacts.IsFinalTaskStarted() { + c.setFinallyStartedTimeIfNeeded(pr, pipelineRunFacts) + } + for _, rpt := range nextRpts { if rpt.IsFinalTask(pipelineRunFacts) { c.setFinallyStartedTimeIfNeeded(pr, pipelineRunFacts) diff --git a/pkg/reconciler/pipelinerun/pipelinerun_test.go b/pkg/reconciler/pipelinerun/pipelinerun_test.go index f61d0cd0e5d..f7fcd10c341 100644 --- a/pkg/reconciler/pipelinerun/pipelinerun_test.go +++ b/pkg/reconciler/pipelinerun/pipelinerun_test.go @@ -2827,6 +2827,257 @@ status: } } +func TestReconcileWithFinallyStartTime(t *testing.T) { + // TestReconcileWithFinallyStartTime runs "Reconcile" on a PipelineRun with tasks is completed and one or more finally tasks + // may need to be executed. + // It verifies that reconcile is successful, the finallyStartTime can be set correctly. + pipelineFinalTask := parse.MustParseV1Pipeline(t, ` +metadata: + name: test-pipeline-with-finally + namespace: foo +spec: + finally: + - name: finaltask-1 + taskRef: + name: hello-world + tasks: + - name: task1 + taskRef: + name: hello-world +`) + pipelineNotFinalTask := parse.MustParseV1Pipeline(t, ` +metadata: + name: test-pipeline-with-finally + namespace: foo +spec: + tasks: + - name: task1 + taskRef: + name: hello-world +`) + pipelineSkippedFinalTask := parse.MustParseV1Pipeline(t, ` +metadata: + name: test-pipeline-with-finally + namespace: foo +spec: + finally: + - name: finaltask-1 + when: + - input: "$(tasks.task1.status)" + operator: in + values: ["Failure"] + taskRef: + name: hello-world + tasks: + - name: task1 + taskRef: + name: hello-world +`) + + prName := "test-pipeline-run-with-set-finally-start-time" + ts := []*v1.Task{simpleHelloWorldTask} + + tcs := []struct { + name string + trs []*v1.TaskRun + ps []*v1.Pipeline + pr *v1.PipelineRun + wantEvents []string + wantFinallyStartTime bool + }{{ + name: "new final task created", + trs: []*v1.TaskRun{ + getTaskRun( + t, + "test-pipeline-run-with-set-finally-start-time-hello-world", + prName, + "test-pipeline-with-finally", + "hello-world", + corev1.ConditionTrue, + ), + }, + ps: []*v1.Pipeline{pipelineFinalTask}, + pr: parse.MustParseV1PipelineRun(t, fmt.Sprintf(` +metadata: + name: %s + namespace: foo +spec: + pipelineRef: + name: test-pipeline-with-finally +status: + startTime: "2021-12-31T23:40:00Z" + childReferences: + - name: test-pipeline-run-with-set-finally-start-time-hello-world + apiVersion: tekton.dev/v1 + kind: TaskRun + pipelineTaskName: task1 + status: + conditions: + - lastTransitionTime: null + status: "True" + type: Succeeded +`, prName)), + wantEvents: []string{ + "Normal Started", + }, + wantFinallyStartTime: true, + }, { + name: "final task started and the pr not final start time", + trs: []*v1.TaskRun{ + getTaskRun( + t, + "test-pipeline-run-with-set-finally-start-time-hello-world", + prName, + "test-pipeline-with-finally", + "hello-world", + corev1.ConditionTrue, + ), + mustParseTaskRunWithObjectMeta(t, taskRunObjectMeta("test-pipeline-run-with-finally-start-time-finaltask-1", "foo", prName, + "test-pipeline-with-finally", "finaltask-1", false), ` +spec: + serviceAccountName: test-sa + taskRef: + name: hello-world + kind: Task +`)}, + ps: []*v1.Pipeline{pipelineFinalTask}, + pr: parse.MustParseV1PipelineRun(t, fmt.Sprintf(` +metadata: + name: %s + namespace: foo +spec: + pipelineRef: + name: test-pipeline-with-finally + timeouts: + tasks: 5m + pipeline: 20m +status: + startTime: "2021-12-31T23:40:00Z" + childReferences: + - name: test-pipeline-run-with-set-finally-start-time-hello-world + apiVersion: tekton.dev/v1 + kind: TaskRun + pipelineTaskName: task1 + status: + conditions: + - lastTransitionTime: null + status: "True" + type: Succeeded + - name: test-pipeline-run-with-set-finally-start-time-finaltask-1 + apiVersion: tekton.dev/v1 + kind: TaskRun + pipelineTaskName: finaltask-1 + status: + conditions: + - lastTransitionTime: null + status: "Unknown" + type: Succeeded +`, prName)), + wantEvents: []string{ + "Normal Started", + }, + wantFinallyStartTime: true, + }, { + name: "final task not exist", + trs: []*v1.TaskRun{ + getTaskRun( + t, + "test-pipeline-run-with-set-finally-start-time-hello-world", + prName, + "test-pipeline-with-finally", + "hello-world", + corev1.ConditionTrue, + ), + }, + ps: []*v1.Pipeline{pipelineNotFinalTask}, + pr: parse.MustParseV1PipelineRun(t, fmt.Sprintf(` +metadata: + name: %s + namespace: foo +spec: + pipelineRef: + name: test-pipeline-with-finally + timeouts: + tasks: 5m + pipeline: 20m +status: + startTime: "2021-12-31T23:40:00Z" + childReferences: + - name: test-pipeline-run-with-set-finally-start-time-hello-world + apiVersion: tekton.dev/v1 + kind: TaskRun + pipelineTaskName: task1 + status: + conditions: + - lastTransitionTime: null + status: "True" + type: Succeeded +`, prName)), + wantEvents: []string{ + "Normal Succeeded Tasks Completed: 1 (Failed: 0, Cancelled 0), Skipped: 0", + }, + wantFinallyStartTime: false, + }, { + name: "final task skipped", + trs: []*v1.TaskRun{ + getTaskRun( + t, + "test-pipeline-run-with-set-finally-start-time-hello-world", + prName, + "test-pipeline-with-finally", + "hello-world", + corev1.ConditionTrue, + ), + }, + ps: []*v1.Pipeline{pipelineSkippedFinalTask}, + pr: parse.MustParseV1PipelineRun(t, fmt.Sprintf(` +metadata: + name: %s + namespace: foo +spec: + pipelineRef: + name: test-pipeline-with-finally + timeouts: + tasks: 5m + pipeline: 20m +status: + startTime: "2021-12-31T23:40:00Z" + childReferences: + - name: test-pipeline-run-with-set-finally-start-time-hello-world + apiVersion: tekton.dev/v1 + kind: TaskRun + pipelineTaskName: task1 + status: + conditions: + - lastTransitionTime: null + status: "True" + type: Succeeded +`, prName)), + wantEvents: []string{ + "Normal Succeeded Tasks Completed: 1 (Failed: 0, Cancelled 0), Skipped: 1", + }, + wantFinallyStartTime: true, + }} + for _, tc := range tcs { + withOwnerReference(tc.trs, prName) + + d := test.Data{ + PipelineRuns: []*v1.PipelineRun{tc.pr}, + Pipelines: tc.ps, + Tasks: ts, + TaskRuns: tc.trs, + } + prt := newPipelineRunTest(t, d) + defer prt.Cancel() + + reconciledRun, _ := prt.reconcileRun("foo", prName, tc.wantEvents, false) + + if tc.wantFinallyStartTime != (reconciledRun.Status.FinallyStartTime != nil) { + t.Errorf("Expected FinallyStartTime != nil to be %t, but was %t", tc.wantFinallyStartTime, reconciledRun.Status.FinallyStartTime != nil) + } + } +} + func TestReconcileWithoutPVC(t *testing.T) { // TestReconcileWithoutPVC runs "Reconcile" on a PipelineRun that has two unrelated tasks. // It verifies that reconcile is successful and that no PVC is created diff --git a/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go b/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go index 65882f4c987..16997680cbf 100644 --- a/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go +++ b/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go @@ -358,6 +358,24 @@ func (facts *PipelineRunFacts) GetFinalTasks() PipelineRunState { return tasks } +// IsFinalTaskStarted returns true if all DAG pipelineTasks is finished and one or more final tasks have been created. +func (facts *PipelineRunFacts) IsFinalTaskStarted() bool { + // check either pipeline has finished executing all DAG pipelineTasks, + // where "finished executing" means succeeded, failed, or skipped. + if facts.checkDAGTasksDone() { + // return list of tasks with all final tasks + for _, t := range facts.State { + if facts.isFinalTask(t.PipelineTask.Name) { + if len(t.TaskRuns) > 0 || len(t.CustomRuns) > 0 { + return true + } + } + } + } + + return false +} + // GetPipelineConditionStatus will return the Condition that the PipelineRun prName should be // updated with, based on the status of the TaskRuns in state. func (facts *PipelineRunFacts) GetPipelineConditionStatus(ctx context.Context, pr *v1.PipelineRun, logger *zap.SugaredLogger, c clock.PassiveClock) *apis.Condition {