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{}