pkg/trait/trait_catalog.go (156 lines of code) (raw):

/* Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to You 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 trait import ( "errors" "fmt" "sort" "strings" v1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1" "github.com/apache/camel-k/v2/pkg/client" "github.com/apache/camel-k/v2/pkg/util/log" corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/util/json" ) // Catalog collects all information about traits in one place. type Catalog struct { L log.Logger traits []Trait } // NewCatalog creates a new trait Catalog. func NewCatalog(c client.Client) *Catalog { traitList := make([]Trait, 0, len(FactoryList)) for _, factory := range FactoryList { traitList = append(traitList, factory()) } sort.Slice(traitList, func(i, j int) bool { if traitList[i].Order() != traitList[j].Order() { return traitList[i].Order() < traitList[j].Order() } return string(traitList[i].ID()) < string(traitList[j].ID()) }) catalog := Catalog{ L: log.Log.WithName("trait"), traits: traitList, } for _, t := range catalog.AllTraits() { if c != nil { t.InjectClient(c) } } return &catalog } func (c *Catalog) AllTraits() []Trait { return append([]Trait(nil), c.traits...) } // Traits may depend on the result of previously executed ones, // so care must be taken while changing the lists order. func (c *Catalog) traitsFor(environment *Environment) []Trait { profile := environment.DetermineProfile() return c.TraitsForProfile(profile) } // TraitsForProfile returns all traits associated with a given profile. // // Traits may depend on the result of previously executed ones, // so care must be taken while changing the lists order. func (c *Catalog) TraitsForProfile(profile v1.TraitProfile) []Trait { var res []Trait for _, t := range c.AllTraits() { if t.IsAllowedInProfile(profile) { res = append(res, t) } } return res } func (c *Catalog) apply(environment *Environment) ([]*TraitCondition, *v1.Traits, error) { traitsConditions := []*TraitCondition{} if err := c.Configure(environment); err != nil { return traitsConditions, nil, err } traits := c.traitsFor(environment) environment.ConfiguredTraits = traits applicable := false for _, trait := range traits { if !environment.PlatformInPhase(v1.IntegrationPlatformPhaseReady) && trait.RequiresIntegrationPlatform() { c.L.Debugf("Skipping trait because of missing integration platform: %s", trait.ID()) continue } applicable = true enabled, condition, err := trait.Configure(environment) if condition != nil { traitsConditions = append(traitsConditions, condition) } if err != nil { return traitsConditions, nil, fmt.Errorf("%s trait configuration failed: %w", trait.ID(), err) } if enabled { err = trait.Apply(environment) if err != nil { return traitsConditions, nil, fmt.Errorf("%s trait execution failed: %w", trait.ID(), err) } environment.ExecutedTraits = append(environment.ExecutedTraits, trait) // execute post step processors for _, processor := range environment.PostStepProcessors { err := processor(environment) if err != nil { return traitsConditions, nil, fmt.Errorf("%s trait executing post step action failed: %w", trait.ID(), err) } } } } cs, ts, err := c.executedTraitCondition(environment.ExecutedTraits) if err != nil { return traitsConditions, &ts, err } traitsConditions = append(traitsConditions, cs) if !applicable && environment.PlatformInPhase(v1.IntegrationPlatformPhaseReady) { return traitsConditions, nil, errors.New("no trait can be executed because of no ready platform found") } for _, processor := range environment.PostProcessors { err := processor(environment) if err != nil { return traitsConditions, nil, fmt.Errorf("error executing post processor: %w", err) } } return traitsConditions, &ts, nil } func (c *Catalog) executedTraitCondition(executedTrait []Trait) (*TraitCondition, v1.Traits, error) { var traits v1.Traits var traitMap = make(map[string]map[string]interface{}) traitIds := make([]string, 0) for _, trait := range executedTrait { data, err := json.Marshal(trait) if err != nil { return nil, traits, err } var traitIDMap map[string]interface{} if err := json.Unmarshal(data, &traitIDMap); err != nil { return nil, traits, err } if len(traitIDMap) > 0 { if isAddon(string(trait.ID())) { traitMap["addons"] = map[string]interface{}{ string(trait.ID()): traitIDMap, } } else { traitMap[string(trait.ID())] = traitIDMap } } traitIds = append(traitIds, string(trait.ID())) } traitData, err := json.Marshal(traitMap) if err != nil { return nil, traits, err } if err := json.Unmarshal(traitData, &traits); err != nil { return nil, traits, err } message := fmt.Sprintf("Applied traits: %s", strings.Join(traitIds, ",")) c.L.Debug(message) return NewIntegrationCondition("", v1.IntegrationConditionTraitInfo, corev1.ConditionTrue, TraitConfigurationReason, message), traits, nil } // Deprecated: remove this check when we include the addons traits into regular traits // see https://github.com/apache/camel-k/issues/5787 // isAddon returns true if the trait is an addon. func isAddon(id string) bool { return id == "master" || id == "keda" || id == "3scale" || id == "tracing" || id == "aws-secrets-manager" || id == "azure-key-vault" || id == "gcp-secret-manager" || id == "hashicorp-vault" } // GetTrait returns the trait with the given ID. func (c *Catalog) GetTrait(id string) Trait { for _, t := range c.AllTraits() { if t.ID() == ID(id) { return t } } return nil } type Finder interface { GetTrait(id string) Trait } var _ Finder = &Catalog{}