Skip to content

Commit b109054

Browse files
pritidesaitekton-robot
authored andcommitted
de-dup order and resource dependencies
In case of a pipeline with a huge list of redundant dependencies, the list is growing exponentially. This way of calculating the list of dependencies is causing extra delay in the validation cycle. This delay sometimes hit the webhook timeout during validation. This is one of the changes being proposed to make the validation cycle efficient and avoid unneccesary delay.
1 parent ce9c916 commit b109054

File tree

3 files changed

+132
-34
lines changed

3 files changed

+132
-34
lines changed

pkg/apis/pipeline/v1beta1/pipeline_types.go

+12-27
Original file line numberDiff line numberDiff line change
@@ -512,44 +512,29 @@ func (pt PipelineTask) Validate(ctx context.Context) (errs *apis.FieldError) {
512512

513513
// Deps returns all other PipelineTask dependencies of this PipelineTask, based on resource usage or ordering
514514
func (pt PipelineTask) Deps() []string {
515-
deps := []string{}
515+
// hold the list of dependencies in a set to avoid duplicates
516+
deps := sets.NewString()
516517

517-
deps = append(deps, pt.resourceDeps()...)
518-
deps = append(deps, pt.orderingDeps()...)
519-
520-
uniqueDeps := sets.NewString()
521-
for _, w := range deps {
522-
if uniqueDeps.Has(w) {
523-
continue
524-
}
525-
uniqueDeps.Insert(w)
526-
}
527-
528-
return uniqueDeps.List()
529-
}
530-
531-
func (pt PipelineTask) resourceDeps() []string {
532-
resourceDeps := []string{}
518+
// add any new dependents from a resource/workspace
533519
if pt.Resources != nil {
534520
for _, rd := range pt.Resources.Inputs {
535-
resourceDeps = append(resourceDeps, rd.From...)
521+
for _, f := range rd.From {
522+
deps.Insert(f)
523+
}
536524
}
537525
}
538526

539-
// Add any dependents from result references.
527+
// add any new dependents from result references - resource dependency
540528
for _, ref := range PipelineTaskResultRefs(&pt) {
541-
resourceDeps = append(resourceDeps, ref.PipelineTask)
529+
deps.Insert(ref.PipelineTask)
542530
}
543531

544-
return resourceDeps
545-
}
546-
547-
func (pt PipelineTask) orderingDeps() []string {
548-
orderingDeps := []string{}
532+
// add any new dependents from runAfter - order dependency
549533
for _, runAfter := range pt.RunAfter {
550-
orderingDeps = append(orderingDeps, runAfter)
534+
deps.Insert(runAfter)
551535
}
552-
return orderingDeps
536+
537+
return deps.List()
553538
}
554539

555540
// PipelineTaskList is a list of PipelineTasks

pkg/apis/pipeline/v1beta1/pipeline_types_test.go

+42-5
Original file line numberDiff line numberDiff line change
@@ -640,13 +640,50 @@ func TestPipelineTaskList_Deps(t *testing.T) {
640640
},
641641
}},
642642
},
643+
}, {
644+
Name: "task-7",
645+
Resources: &PipelineTaskResources{
646+
Inputs: []PipelineTaskInputResource{{
647+
From: []string{"task-1", "task-1"},
648+
}},
649+
},
650+
}, {
651+
Name: "task-8",
652+
WhenExpressions: WhenExpressions{{
653+
Input: "$(tasks.task-3.results.result1)",
654+
Operator: "in",
655+
Values: []string{"foo"},
656+
}, {
657+
Input: "$(tasks.task-3.results.result2)",
658+
Operator: "in",
659+
Values: []string{"foo"},
660+
}},
661+
}, {
662+
Name: "task-9",
663+
Params: []Param{{
664+
Value: ArrayOrString{
665+
Type: "string",
666+
StringVal: "$(tasks.task-4.results.result1)",
667+
}}, {
668+
Value: ArrayOrString{
669+
Type: "string",
670+
StringVal: "$(tasks.task-4.results.result2)",
671+
}},
672+
},
673+
}, {
674+
Name: "task-10",
675+
RunAfter: []string{"task-1", "task-1", "task-1", "task-1"},
643676
}},
644677
expectedDeps: map[string][]string{
645-
"task-2": {"task-1"},
646-
"task-3": {"task-1", "task-2"},
647-
"task-4": {"task-1", "task-2", "task-3"},
648-
"task-5": {"task-1", "task-2", "task-3", "task-4"},
649-
"task-6": {"task-1", "task-2", "task-3", "task-4", "task-5"},
678+
"task-2": {"task-1"},
679+
"task-3": {"task-1", "task-2"},
680+
"task-4": {"task-1", "task-2", "task-3"},
681+
"task-5": {"task-1", "task-2", "task-3", "task-4"},
682+
"task-6": {"task-1", "task-2", "task-3", "task-4", "task-5"},
683+
"task-7": {"task-1"},
684+
"task-8": {"task-3"},
685+
"task-9": {"task-4"},
686+
"task-10": {"task-1"},
650687
},
651688
}}
652689
for _, tc := range pipelines {

pkg/reconciler/pipelinerun/resources/pipelinerunstate_test.go

+78-2
Original file line numberDiff line numberDiff line change
@@ -1308,7 +1308,7 @@ func TestDAGExecutionQueueSequentialRuns(t *testing.T) {
13081308
}
13091309

13101310
func TestPipelineRunState_CompletedOrSkippedDAGTasks(t *testing.T) {
1311-
largePipelineState := buildPipelineStateWithLargeDepencyGraph(t)
1311+
largePipelineState := buildPipelineStateWithLargeDependencyGraph(t)
13121312
tcs := []struct {
13131313
name string
13141314
state PipelineRunState
@@ -1353,6 +1353,14 @@ func TestPipelineRunState_CompletedOrSkippedDAGTasks(t *testing.T) {
13531353
name: "large deps, not started",
13541354
state: largePipelineState,
13551355
expectedNames: []string{},
1356+
}, {
1357+
name: "large deps through params, not started",
1358+
state: buildPipelineStateWithMultipleTaskResults(t, false),
1359+
expectedNames: []string{},
1360+
}, {
1361+
name: "large deps through params and when expressions, not started",
1362+
state: buildPipelineStateWithMultipleTaskResults(t, true),
1363+
expectedNames: []string{},
13561364
}, {
13571365
name: "one-run-started",
13581366
state: oneRunStartedState,
@@ -1386,7 +1394,7 @@ func TestPipelineRunState_CompletedOrSkippedDAGTasks(t *testing.T) {
13861394
}
13871395
}
13881396

1389-
func buildPipelineStateWithLargeDepencyGraph(t *testing.T) PipelineRunState {
1397+
func buildPipelineStateWithLargeDependencyGraph(t *testing.T) PipelineRunState {
13901398
t.Helper()
13911399
var task = &v1beta1.Task{
13921400
ObjectMeta: metav1.ObjectMeta{
@@ -1445,6 +1453,74 @@ func buildPipelineStateWithLargeDepencyGraph(t *testing.T) PipelineRunState {
14451453
return pipelineRunState
14461454
}
14471455

1456+
func buildPipelineStateWithMultipleTaskResults(t *testing.T, includeWhen bool) PipelineRunState {
1457+
t.Helper()
1458+
var task = &v1beta1.Task{
1459+
ObjectMeta: metav1.ObjectMeta{
1460+
Name: "task",
1461+
},
1462+
Spec: v1beta1.TaskSpec{
1463+
Steps: []v1beta1.Step{{
1464+
Name: "step1",
1465+
}},
1466+
},
1467+
}
1468+
var pipelineRunState PipelineRunState
1469+
pipelineRunState = []*ResolvedPipelineTask{{
1470+
PipelineTask: &v1beta1.PipelineTask{
1471+
Name: "t1",
1472+
TaskRef: &v1beta1.TaskRef{Name: "task"},
1473+
},
1474+
TaskRun: nil,
1475+
ResolvedTaskResources: &resources.ResolvedTaskResources{
1476+
TaskSpec: &task.Spec,
1477+
},
1478+
}}
1479+
for i := 2; i < 400; i++ {
1480+
var params []v1beta1.Param
1481+
whenExpressions := v1beta1.WhenExpressions{}
1482+
var alpha byte
1483+
// the task has a reference to multiple task results (a through j) from each parent task - causing a redundant references
1484+
// the task dependents on all predecessors in a graph through params and/or whenExpressions
1485+
for j := 1; j < i; j++ {
1486+
for alpha = 'a'; alpha <= 'j'; alpha++ {
1487+
// include param with task results
1488+
params = append(params, v1beta1.Param{
1489+
Name: fmt.Sprintf("%c", alpha),
1490+
Value: v1beta1.ArrayOrString{
1491+
Type: v1beta1.ParamTypeString,
1492+
StringVal: fmt.Sprintf("$(tasks.t%d.results.%c)", j, alpha),
1493+
},
1494+
})
1495+
}
1496+
if includeWhen {
1497+
for alpha = 'a'; alpha <= 'j'; alpha++ {
1498+
// include when expressions with task results
1499+
whenExpressions = append(whenExpressions, v1beta1.WhenExpression{
1500+
Input: fmt.Sprintf("$(tasks.t%d.results.%c)", j, alpha),
1501+
Operator: selection.In,
1502+
Values: []string{"true"},
1503+
})
1504+
}
1505+
}
1506+
}
1507+
pipelineRunState = append(pipelineRunState, &ResolvedPipelineTask{
1508+
PipelineTask: &v1beta1.PipelineTask{
1509+
Name: fmt.Sprintf("t%d", i),
1510+
Params: params,
1511+
TaskRef: &v1beta1.TaskRef{Name: "task"},
1512+
WhenExpressions: whenExpressions,
1513+
},
1514+
TaskRun: nil,
1515+
ResolvedTaskResources: &resources.ResolvedTaskResources{
1516+
TaskSpec: &task.Spec,
1517+
},
1518+
},
1519+
)
1520+
}
1521+
return pipelineRunState
1522+
}
1523+
14481524
func TestPipelineRunState_GetFinalTasks(t *testing.T) {
14491525
tcs := []struct {
14501526
name string

0 commit comments

Comments
 (0)