Skip to content

Commit

Permalink
bugfix(apache#5758): changing dependencies build order strategy to be…
Browse files Browse the repository at this point in the history
… compatible with the incremental image build algorithm
  • Loading branch information
lsergio committed Aug 14, 2024
1 parent 0b8f885 commit 0fa293d
Show file tree
Hide file tree
Showing 2 changed files with 110 additions and 14 deletions.
104 changes: 99 additions & 5 deletions pkg/apis/camel/v1/build_type_support_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -403,14 +403,108 @@ func TestMatchingBuildsSchedulingSameDependenciesSameRuntime(t *testing.T) {
Items: []Build{buildA, buildB},
}

// ebuilds have the same dependencies, runtime and creation timestamp
// builds have the same dependencies, runtime and creation timestamp
// buildB should wait for buildA

matches, buildMatch := buildList.HasMatchingBuild(&buildA)
assert.False(t, matches)
assert.Nil(t, buildMatch)
matches, buildMatch = buildList.HasMatchingBuild(&buildB)
assert.True(t, matches)
assert.True(t, buildMatch.Name == buildA.Name)
assert.Equal(t, buildA.Name, buildMatch.Name)
}

func TestMatchingBuildsSchedulingMutipleSubsets(t *testing.T) {
timestamp, _ := time.Parse("2006-01-02T15:04:05-0700", "2024-08-09T10:00:00Z")
creationTimestamp := v1.Time{Time: timestamp}
buildA := Build{
ObjectMeta: v1.ObjectMeta{
Name: "buildA",
},
Spec: BuildSpec{
Tasks: []Task{
{
Builder: &BuilderTask{
Dependencies: []string{
"camel:component-a",
},
Runtime: RuntimeSpec{
Version: "3.8.1",
},
},
},
},
},
Status: BuildStatus{
Phase: BuildPhaseScheduling,
},
}
buildB := Build{
ObjectMeta: v1.ObjectMeta{
Name: "buildB",
CreationTimestamp: creationTimestamp,
},
Spec: BuildSpec{
Tasks: []Task{
{
Builder: &BuilderTask{
Dependencies: []string{
"camel:component-a",
"camel:component-b",
},
Runtime: RuntimeSpec{
Version: "3.8.1",
},
},
},
},
},
Status: BuildStatus{
Phase: BuildPhaseScheduling,
},
}
buildC := Build{
ObjectMeta: v1.ObjectMeta{
Name: "buildC",
CreationTimestamp: creationTimestamp,
},
Spec: BuildSpec{
Tasks: []Task{
{
Builder: &BuilderTask{
Dependencies: []string{
"camel:component-a",
"camel:component-b",
"camel:component-c",
},
Runtime: RuntimeSpec{
Version: "3.8.1",
},
},
},
},
},
Status: BuildStatus{
Phase: BuildPhaseScheduling,
},
}

buildList := BuildList{
Items: []Build{buildA, buildB, buildC},
}

// buildA is a subset of buildB, which is a subset of buildC
// buildC should wait for B, which should wait for A

matches, buildMatch := buildList.HasMatchingBuild(&buildA)
assert.False(t, matches)
assert.Nil(t, buildMatch)
matches, buildMatch = buildList.HasMatchingBuild(&buildB)
assert.True(t, matches)
assert.Equal(t, buildA.Name, buildMatch.Name)
matches, buildMatch = buildList.HasMatchingBuild(&buildC)
assert.True(t, matches)
assert.Equal(t, buildB.Name, buildMatch.Name)
}

func TestMatchingBuildsSchedulingFewCommonDependencies(t *testing.T) {
Expand Down Expand Up @@ -475,12 +569,12 @@ func TestMatchingBuildsSchedulingFewCommonDependencies(t *testing.T) {
Items: []Build{buildA, buildB},
}

// builds have only 1 out of 10 required dependencies. they should not match
// buildA is a subset of buildB. should wait for it

matches, buildMatch := buildList.HasMatchingBuild(&buildA)
assert.False(t, matches)
assert.Nil(t, buildMatch)
matches, buildMatch = buildList.HasMatchingBuild(&buildB)
assert.False(t, matches)
assert.Nil(t, buildMatch)
assert.True(t, matches)
assert.Equal(t, buildA.Name, buildMatch.Name)
}
20 changes: 11 additions & 9 deletions pkg/apis/camel/v1/build_types_support.go
Original file line number Diff line number Diff line change
Expand Up @@ -288,6 +288,8 @@ func (bl BuildList) HasMatchingBuild(build *Build) (bool, *Build) {
}
runtimeVersion := build.RuntimeVersion()

var bestBuild *Build = nil
bestBuildCommonDependencies := 0
buildLoop:
for _, b := range bl.Items {
if b.Name == build.Name || b.Status.IsFinished() {
Expand Down Expand Up @@ -323,8 +325,8 @@ buildLoop:
}
}

if commonDependencies < len(required)/2 {
// few common dependencies
if commonDependencies == 0 {
// no common dependencies
continue
}

Expand All @@ -335,23 +337,23 @@ buildLoop:
case BuildPhaseInitialization, BuildPhaseScheduling:
// handle suitable scheduled build

// the build has at least half the dependencies, maybe all of them
if compareBuilds(&b, build) < 0 {
return true, &b
}
if missing == 0 {
// seems like both builds require exactly the same list of dependencies
// additionally check for the creation timestamp
if compareBuilds(&b, build) < 0 {
return true, &b
}
} else {
// some deps are missing, but this build has at least half the deps we need and no extra dependency.
return true, &b
} else if commonDependencies > bestBuildCommonDependencies {
bestBuildCommonDependencies = commonDependencies
bestBuild = &b
continue
}
}
}

if bestBuild != nil {
return true, bestBuild
}
return false, nil
}

Expand Down

0 comments on commit 0fa293d

Please sign in to comment.