Skip to content

Commit

Permalink
fix(traits): use Comparable matches
Browse files Browse the repository at this point in the history
Reverting apache#4512 which introduced a function diverging the match from the original design
  • Loading branch information
squakez committed Mar 8, 2024
1 parent b986989 commit f004581
Show file tree
Hide file tree
Showing 13 changed files with 228 additions and 113 deletions.
11 changes: 10 additions & 1 deletion docs/modules/ROOT/pages/architecture/traits.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,15 @@ type Trait interface {
IsAllowedInProfile(v1.TraitProfile) bool
Order() int
}
type Comparable interface {
Matches(trait Trait) bool
}
type ComparableTrait interface {
Trait
Comparable
}
----

Each trait will implement this interface. The most important methods that will be invoked by the xref:architecture/operator.adoc[Operator] are `Configure()` and `Apply()`. Basically, the `Configure()` method will set those inputs aforementioned (each trait has its own). The method is in charge to verify also the correctness of those expected parameters, where it makes sense (i.e., a well expected `Kubernetes` resource name). The function can return a `TraitCondition` object containing any informative or warning condition to be attached to the resulting Integration (for instance, traits forcefully altered by the platform, or deprecation notices).
Expand All @@ -59,6 +68,6 @@ The `Order()` method helps in resolving the order of execution of different trai

The `InfluencesKit()`, `IsPlatformTrait()` and `RequiresIntegrationPlatform()` methods are easy to understand. They are used to determine if a trait has to influence an `IntegrationKit` build/initialization, if it's a platform trait (ie, needed by the platform itself) or are requiring the presence of an `IntegrationPlatform`.

The presence of `InfluencesBuild()` will let specify the level of granularity of a trait down to its properties for a rebuild. So, if you need, you can compare the traits properties coming from the `prev` (previous) Integration to decide if it is worth to rebuild an Integration or the trait can reuse the one already provided in `this` version.
For those traits that `InfluencesKit()` you may need to provide a `Matches(trait Trait)` func in order to specify those trait parameters that influences a build. This is required by the platform to decide if it is worth to rebuild an Integration or the trait can reuse the one already provided.

Finally, through the `IsAllowedInProfile()` method we can override the default behavior (allow the trait for any profile). We must specify the profile we expect for this trait to be executed properly.
67 changes: 2 additions & 65 deletions pkg/controller/integration/kits.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,6 @@ package integration

import (
"context"
"fmt"
"reflect"

k8serrors "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/labels"
Expand Down Expand Up @@ -132,7 +130,7 @@ func integrationMatches(ctx context.Context, c client.Client, integration *v1.In
return false, err
}

if match, err := hasMatchingTraits(itc, ikc); !match || err != nil {
if match, err := trait.HasMatchingTraits(itc, ikc); !match || err != nil {
ilog.Debug("Integration and integration-kit traits do not match", "integration", integration.Name, "integration-kit", kit.Name, "namespace", integration.Namespace)
return false, err
}
Expand Down Expand Up @@ -195,7 +193,7 @@ func kitMatches(kit *v1.IntegrationKit, target *v1.IntegrationKit) (bool, error)
return false, err
}

if match, err := hasMatchingTraits(c1, c2); !match || err != nil {
if match, err := trait.HasMatchingTraits(c1, c2); !match || err != nil {
return false, err
}
if !util.StringSliceContains(kit.Spec.Dependencies, target.Spec.Dependencies) {
Expand All @@ -205,67 +203,6 @@ func kitMatches(kit *v1.IntegrationKit, target *v1.IntegrationKit) (bool, error)
return true, nil
}

func hasMatchingTraits(traitMap trait.Options, kitTraitMap trait.Options) (bool, error) {
catalog := trait.NewCatalog(nil)

for _, t := range catalog.AllTraits() {
if t == nil {
continue
}

id := string(t.ID())
it, ok1 := traitMap.Get(id)
kt, ok2 := kitTraitMap.Get(id)

if !ok1 && !ok2 {
continue
}

if t.InfluencesKit() && t.InfluencesBuild(it, kt) {
if ct, ok := t.(trait.ComparableTrait); ok {
// if it's match trait use its matches method to determine the match
if match, err := matchesComparableTrait(ct, it, kt); !match || err != nil {
return false, err
}
} else {
if !matchesTrait(it, kt) {
return false, nil
}
}
}
}

return true, nil
}

func matchesComparableTrait(ct trait.ComparableTrait, it map[string]interface{}, kt map[string]interface{}) (bool, error) {
t1 := reflect.New(reflect.TypeOf(ct).Elem()).Interface()
if err := trait.ToTrait(it, &t1); err != nil {
return false, err
}

t2 := reflect.New(reflect.TypeOf(ct).Elem()).Interface()
if err := trait.ToTrait(kt, &t2); err != nil {
return false, err
}

ct2, ok := t2.(trait.ComparableTrait)
if !ok {
return false, fmt.Errorf("type assertion failed: %v", t2)
}
tt1, ok := t1.(trait.Trait)
if !ok {
return false, fmt.Errorf("type assertion failed: %v", t1)
}

return ct2.Matches(tt1), nil
}

func matchesTrait(it map[string]interface{}, kt map[string]interface{}) bool {
// perform exact match on the two trait maps
return reflect.DeepEqual(it, kt)
}

func hasMatchingSourcesForNative(it *v1.Integration, kit *v1.IntegrationKit) bool {
if len(it.UserDefinedSources()) != len(kit.Spec.Sources) {
return false
Expand Down
15 changes: 0 additions & 15 deletions pkg/controller/integration/kits_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,6 @@ import (
traitv1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1/trait"

"github.com/apache/camel-k/v2/pkg/trait"
"github.com/apache/camel-k/v2/pkg/util/log"
"github.com/apache/camel-k/v2/pkg/util/test"

"github.com/stretchr/testify/assert"
Expand Down Expand Up @@ -94,10 +93,6 @@ func TestLookupKitForIntegration_DiscardKitsInError(t *testing.T) {

require.NoError(t, err)

a := buildKitAction{}
a.InjectLogger(log.Log)
a.InjectClient(c)

kits, err := lookupKitsForIntegration(context.TODO(), c, &v1.Integration{
TypeMeta: metav1.TypeMeta{
APIVersion: v1.SchemeGroupVersion.String(),
Expand Down Expand Up @@ -221,10 +216,6 @@ func TestLookupKitForIntegration_DiscardKitsWithIncompatibleTraits(t *testing.T)

require.NoError(t, err)

a := buildKitAction{}
a.InjectLogger(log.Log)
a.InjectClient(c)

kits, err := lookupKitsForIntegration(context.TODO(), c, &v1.Integration{
TypeMeta: metav1.TypeMeta{
APIVersion: v1.SchemeGroupVersion.String(),
Expand Down Expand Up @@ -288,9 +279,6 @@ func TestHasMatchingTraits_KitNoTraitShouldNotBePicked(t *testing.T) {
},
}

a := buildKitAction{}
a.InjectLogger(log.Log)

ok, err := integrationAndKitHaveSameTraits(integration, kit)
require.NoError(t, err)
assert.False(t, ok)
Expand Down Expand Up @@ -339,9 +327,6 @@ func TestHasMatchingTraits_KitSameTraitShouldBePicked(t *testing.T) {
},
}

a := buildKitAction{}
a.InjectLogger(log.Log)

ok, err := integrationAndKitHaveSameTraits(integration, kit)
require.NoError(t, err)
assert.True(t, ok)
Expand Down
22 changes: 19 additions & 3 deletions pkg/trait/builder.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ package trait
import (
"fmt"
"regexp"
"slices"
"sort"
"strconv"
"strings"
Expand Down Expand Up @@ -56,9 +57,24 @@ func (t *builderTrait) InfluencesKit() bool {
return true
}

// InfluencesBuild overrides base class method.
func (t *builderTrait) InfluencesBuild(this, prev map[string]interface{}) bool {
return true
func (t *builderTrait) Matches(trait Trait) bool {
otherTrait, ok := trait.(*builderTrait)
if !ok {
return false
}
if t.BaseImage != otherTrait.BaseImage || len(t.Properties) != len(otherTrait.Properties) {
return false
}
// More sofisticated check if len is the same. Sort and compare via slices equal func.
// Altough the Matches func is used as a support for comparison, it makes sense
// to copy the properties and avoid possible inconsistencies caused by the sorting operation.
srtThisProps := make([]string, len(t.Properties))
srtOtheProps := make([]string, len(otherTrait.Properties))
copy(srtThisProps, t.Properties)
copy(srtOtheProps, otherTrait.Properties)
slices.Sort(srtThisProps)
slices.Sort(srtOtheProps)
return slices.Equal(srtThisProps, srtOtheProps)
}

func (t *builderTrait) Configure(e *Environment) (bool, *TraitCondition, error) {
Expand Down
30 changes: 30 additions & 0 deletions pkg/trait/builder_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"

v1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1"
traitv1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1/trait"
"github.com/apache/camel-k/v2/pkg/util/camel"
"github.com/apache/camel-k/v2/pkg/util/defaults"
"github.com/apache/camel-k/v2/pkg/util/kubernetes"
Expand Down Expand Up @@ -603,3 +604,32 @@ func tasksByName(tasks []v1.Task) []string {
}
return pipelineTasks
}

func TestBuilderMatches(t *testing.T) {
t1 := builderTrait{
BasePlatformTrait: NewBasePlatformTrait("builder", 600),
BuilderTrait: traitv1.BuilderTrait{
OrderStrategy: "dependencies",
},
}
t2 := builderTrait{
BasePlatformTrait: NewBasePlatformTrait("builder", 600),
BuilderTrait: traitv1.BuilderTrait{
OrderStrategy: "dependencies",
},
}
assert.True(t, t1.Matches(&t2))
// This is a property that does not influence the build
t2.OrderStrategy = "fifo"
assert.True(t, t1.Matches(&t2))
// Changing properties which influences build
t1.Properties = []string{"hello=world"}
assert.False(t, t1.Matches(&t2))
t2.Properties = []string{"hello=world"}
assert.True(t, t1.Matches(&t2))
t1.Properties = []string{"hello=world", "weare=theworld"}
assert.False(t, t1.Matches(&t2))
// should detect swap
t2.Properties = []string{"weare=theworld", "hello=world"}
assert.True(t, t1.Matches(&t2))
}
10 changes: 7 additions & 3 deletions pkg/trait/camel.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,9 +53,13 @@ func (t *camelTrait) InfluencesKit() bool {
return true
}

// InfluencesBuild only when the runtime has changed.
func (t *camelTrait) InfluencesBuild(this, prev map[string]interface{}) bool {
return this["runtimeVersion"] != prev["runtimeVersion"]
func (t *camelTrait) Matches(trait Trait) bool {
otherTrait, ok := trait.(*camelTrait)
if !ok {
return false
}

return otherTrait.RuntimeVersion == t.RuntimeVersion
}

func (t *camelTrait) Configure(e *Environment) (bool, *TraitCondition, error) {
Expand Down
22 changes: 22 additions & 0 deletions pkg/trait/camel_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"

v1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1"
traitv1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1/trait"
"github.com/apache/camel-k/v2/pkg/util/camel"
"github.com/apache/camel-k/v2/pkg/util/kubernetes"
"github.com/apache/camel-k/v2/pkg/util/test"
Expand Down Expand Up @@ -208,3 +209,24 @@ func TestApplyCamelTraitWithSources(t *testing.T) {
"content": "XML Source Code",
}, sourceCm.Data)
}

func TestCamelMatches(t *testing.T) {
t1 := camelTrait{
BasePlatformTrait: NewBasePlatformTrait("camel", 600),
CamelTrait: traitv1.CamelTrait{
RuntimeVersion: "1.2.3",
},
}
t2 := camelTrait{
BasePlatformTrait: NewBasePlatformTrait("camel", 600),
CamelTrait: traitv1.CamelTrait{
RuntimeVersion: "1.2.3",
},
}

assert.True(t, t1.Matches(&t2))
t1.Properties = []string{"hello=world"}
assert.True(t, t1.Matches(&t2))
t2.RuntimeVersion = "3.2.1"
assert.False(t, t1.Matches(&t2))
}
20 changes: 10 additions & 10 deletions pkg/trait/quarkus.go
Original file line number Diff line number Diff line change
Expand Up @@ -109,23 +109,14 @@ func (t *quarkusTrait) InfluencesKit() bool {
return true
}

// InfluencesBuild overrides base class method.
func (t *quarkusTrait) InfluencesBuild(this, prev map[string]interface{}) bool {
return true
}

var _ ComparableTrait = &quarkusTrait{}

func (t *quarkusTrait) Matches(trait Trait) bool {
qt, ok := trait.(*quarkusTrait)
if !ok {
return false
}

if len(t.Modes) == 0 && len(qt.Modes) != 0 && !qt.containsMode(traitv1.JvmQuarkusMode) {
return false
}

for _, md := range t.Modes {
if md == traitv1.JvmQuarkusMode && len(qt.Modes) == 0 {
continue
Expand All @@ -135,8 +126,17 @@ func (t *quarkusTrait) Matches(trait Trait) bool {
}
return false
}
// We need to check if the native base image used is the same
thisNativeBaseImage := t.NativeBaseImage
if thisNativeBaseImage == "" {
thisNativeBaseImage = QuarkusNativeDefaultBaseImageName
}
otherNativeBaseImage := qt.NativeBaseImage
if otherNativeBaseImage == "" {
otherNativeBaseImage = QuarkusNativeDefaultBaseImageName
}

return true
return thisNativeBaseImage == otherNativeBaseImage
}

func (t *quarkusTrait) Configure(e *Environment) (bool, *TraitCondition, error) {
Expand Down
28 changes: 28 additions & 0 deletions pkg/trait/quarkus_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -225,3 +225,31 @@ func TestGetLanguageSettingsWithLoaders(t *testing.T) {
assert.Equal(t, languageSettings{native: false, sourcesRequiredAtBuildTime: true}, getLanguageSettings(environment, v1.LanguageKotlin))
assert.Equal(t, languageSettings{native: true, sourcesRequiredAtBuildTime: false}, getLanguageSettings(environment, v1.LanguageJavaShell))
}

func TestQuarkusMatches(t *testing.T) {
qt := quarkusTrait{
BasePlatformTrait: NewBasePlatformTrait("quarkus", 600),
QuarkusTrait: traitv1.QuarkusTrait{
Modes: []traitv1.QuarkusMode{traitv1.JvmQuarkusMode},
},
}
qt2 := quarkusTrait{
BasePlatformTrait: NewBasePlatformTrait("quarkus", 600),
QuarkusTrait: traitv1.QuarkusTrait{
Modes: []traitv1.QuarkusMode{traitv1.JvmQuarkusMode},
NativeBaseImage: QuarkusNativeDefaultBaseImageName,
},
}

assert.True(t, qt.Matches(&qt2))
qt2.Modes = append(qt2.Modes, traitv1.NativeQuarkusMode)
assert.True(t, qt.Matches(&qt2))
qt2.Modes = []traitv1.QuarkusMode{traitv1.NativeQuarkusMode}
assert.False(t, qt.Matches(&qt2))
qt2.Modes = nil
assert.True(t, qt.Matches(&qt2))
qt2.Modes = []traitv1.QuarkusMode{}
assert.True(t, qt.Matches(&qt2))
qt2.NativeBaseImage = "docker.io/my-new-native-base"
assert.False(t, qt.Matches(&qt2))
}
5 changes: 0 additions & 5 deletions pkg/trait/registry.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,11 +54,6 @@ func (t *registryTrait) InfluencesKit() bool {
return true
}

// InfluencesBuild overrides base class method.
func (t *registryTrait) InfluencesBuild(this, prev map[string]interface{}) bool {
return true
}

func (t *registryTrait) Configure(e *Environment) (bool, *TraitCondition, error) {
// disabled by default
if e.IntegrationKit == nil || !pointer.BoolDeref(t.Enabled, false) {
Expand Down
Loading

0 comments on commit f004581

Please sign in to comment.