| /* |
| 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 ( |
| "fmt" |
| "strings" |
| |
| corev1 "k8s.io/api/core/v1" |
| metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" |
| "k8s.io/apimachinery/pkg/labels" |
| "k8s.io/apimachinery/pkg/selection" |
| "k8s.io/utils/pointer" |
| |
| v1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1" |
| traitv1 "github.com/apache/camel-k/v2/pkg/apis/camel/v1/trait" |
| ) |
| |
| const ( |
| affinityTraitID = "affinity" |
| affinityTraitOrder = 1500 |
| ) |
| |
| type affinityTrait struct { |
| BaseTrait |
| traitv1.AffinityTrait `property:",squash"` |
| } |
| |
| func newAffinityTrait() Trait { |
| return &affinityTrait{ |
| BaseTrait: NewBaseTrait(affinityTraitID, affinityTraitOrder), |
| AffinityTrait: traitv1.AffinityTrait{ |
| PodAffinity: pointer.Bool(false), |
| PodAntiAffinity: pointer.Bool(false), |
| }, |
| } |
| } |
| |
| func (t *affinityTrait) Configure(e *Environment) (bool, *TraitCondition, error) { |
| if e.Integration == nil || !pointer.BoolDeref(t.Enabled, false) { |
| return false, nil, nil |
| } |
| |
| if pointer.BoolDeref(t.PodAffinity, false) && pointer.BoolDeref(t.PodAntiAffinity, false) { |
| return false, nil, fmt.Errorf("both pod affinity and pod anti-affinity can't be set simultaneously") |
| } |
| |
| return e.IntegrationInRunningPhases(), nil, nil |
| } |
| |
| func (t *affinityTrait) Apply(e *Environment) error { |
| podSpec := e.GetIntegrationPodSpec() |
| |
| if podSpec == nil { |
| return fmt.Errorf("could not find any integration deployment for %v", e.Integration.Name) |
| } |
| if podSpec.Affinity == nil { |
| podSpec.Affinity = &corev1.Affinity{} |
| } |
| if err := t.addNodeAffinity(e, podSpec); err != nil { |
| return err |
| } |
| if err := t.addPodAffinity(e, podSpec); err != nil { |
| return err |
| } |
| if err := t.addPodAntiAffinity(e, podSpec); err != nil { |
| return err |
| } |
| return nil |
| } |
| |
| func (t *affinityTrait) addNodeAffinity(_ *Environment, podSpec *corev1.PodSpec) error { |
| if len(t.NodeAffinityLabels) == 0 { |
| return nil |
| } |
| |
| nodeSelectorRequirements := make([]corev1.NodeSelectorRequirement, 0) |
| selector, err := labels.Parse(strings.Join(t.NodeAffinityLabels, ",")) |
| if err != nil { |
| return err |
| } |
| requirements, _ := selector.Requirements() |
| for _, r := range requirements { |
| operator, err := operatorToNodeSelectorOperator(r.Operator()) |
| if err != nil { |
| return err |
| } |
| nodeSelectorRequirement := corev1.NodeSelectorRequirement{ |
| Key: r.Key(), |
| Operator: operator, |
| Values: r.Values().List(), |
| } |
| nodeSelectorRequirements = append(nodeSelectorRequirements, nodeSelectorRequirement) |
| } |
| |
| nodeAffinity := &corev1.NodeAffinity{ |
| RequiredDuringSchedulingIgnoredDuringExecution: &corev1.NodeSelector{ |
| NodeSelectorTerms: []corev1.NodeSelectorTerm{ |
| { |
| MatchExpressions: nodeSelectorRequirements, |
| }, |
| }, |
| }, |
| } |
| |
| podSpec.Affinity.NodeAffinity = nodeAffinity |
| return nil |
| } |
| |
| //nolint:dupl |
| func (t *affinityTrait) addPodAffinity(e *Environment, podSpec *corev1.PodSpec) error { |
| if !pointer.BoolDeref(t.PodAffinity, false) && len(t.PodAffinityLabels) == 0 { |
| return nil |
| } |
| |
| labelSelectorRequirements := make([]metav1.LabelSelectorRequirement, 0) |
| if len(t.PodAffinityLabels) > 0 { |
| selector, err := labels.Parse(strings.Join(t.PodAffinityLabels, ",")) |
| if err != nil { |
| return err |
| } |
| requirements, _ := selector.Requirements() |
| for _, r := range requirements { |
| operator, err := operatorToLabelSelectorOperator(r.Operator()) |
| if err != nil { |
| return err |
| } |
| requirement := metav1.LabelSelectorRequirement{ |
| Key: r.Key(), |
| Operator: operator, |
| Values: r.Values().List(), |
| } |
| labelSelectorRequirements = append(labelSelectorRequirements, requirement) |
| } |
| } |
| |
| if pointer.BoolDeref(t.PodAffinity, false) { |
| labelSelectorRequirements = append(labelSelectorRequirements, metav1.LabelSelectorRequirement{ |
| Key: v1.IntegrationLabel, |
| Operator: metav1.LabelSelectorOpIn, |
| Values: []string{ |
| e.Integration.Name, |
| }, |
| }) |
| } |
| |
| podAffinity := &corev1.PodAffinity{ |
| RequiredDuringSchedulingIgnoredDuringExecution: []corev1.PodAffinityTerm{ |
| { |
| LabelSelector: &metav1.LabelSelector{ |
| MatchExpressions: labelSelectorRequirements, |
| }, |
| TopologyKey: "kubernetes.io/hostname", |
| }, |
| }, |
| } |
| |
| podSpec.Affinity.PodAffinity = podAffinity |
| return nil |
| } |
| |
| //nolint:dupl |
| func (t *affinityTrait) addPodAntiAffinity(e *Environment, podSpec *corev1.PodSpec) error { |
| if !pointer.BoolDeref(t.PodAntiAffinity, false) && len(t.PodAntiAffinityLabels) == 0 { |
| return nil |
| } |
| |
| labelSelectorRequirements := make([]metav1.LabelSelectorRequirement, 0) |
| if len(t.PodAntiAffinityLabels) > 0 { |
| selector, err := labels.Parse(strings.Join(t.PodAntiAffinityLabels, ",")) |
| if err != nil { |
| return err |
| } |
| requirements, _ := selector.Requirements() |
| for _, r := range requirements { |
| operator, err := operatorToLabelSelectorOperator(r.Operator()) |
| if err != nil { |
| return err |
| } |
| requirement := metav1.LabelSelectorRequirement{ |
| Key: r.Key(), |
| Operator: operator, |
| Values: r.Values().List(), |
| } |
| labelSelectorRequirements = append(labelSelectorRequirements, requirement) |
| } |
| } |
| |
| if pointer.BoolDeref(t.PodAntiAffinity, false) { |
| labelSelectorRequirements = append(labelSelectorRequirements, metav1.LabelSelectorRequirement{ |
| Key: v1.IntegrationLabel, |
| Operator: metav1.LabelSelectorOpIn, |
| Values: []string{ |
| e.Integration.Name, |
| }, |
| }) |
| } |
| |
| podAntiAffinity := &corev1.PodAntiAffinity{ |
| RequiredDuringSchedulingIgnoredDuringExecution: []corev1.PodAffinityTerm{ |
| { |
| LabelSelector: &metav1.LabelSelector{ |
| MatchExpressions: labelSelectorRequirements, |
| }, |
| TopologyKey: "kubernetes.io/hostname", |
| }, |
| }, |
| } |
| |
| podSpec.Affinity.PodAntiAffinity = podAntiAffinity |
| return nil |
| } |
| |
| func operatorToNodeSelectorOperator(operator selection.Operator) (corev1.NodeSelectorOperator, error) { |
| switch operator { |
| case selection.In, selection.Equals, selection.DoubleEquals: |
| return corev1.NodeSelectorOpIn, nil |
| case selection.NotIn, selection.NotEquals: |
| return corev1.NodeSelectorOpNotIn, nil |
| case selection.Exists: |
| return corev1.NodeSelectorOpExists, nil |
| case selection.DoesNotExist: |
| return corev1.NodeSelectorOpDoesNotExist, nil |
| case selection.GreaterThan: |
| return corev1.NodeSelectorOpGt, nil |
| case selection.LessThan: |
| return corev1.NodeSelectorOpLt, nil |
| } |
| return "", fmt.Errorf("unsupported node selector operator: %s", operator) |
| } |
| |
| func operatorToLabelSelectorOperator(operator selection.Operator) (metav1.LabelSelectorOperator, error) { |
| switch operator { |
| case selection.In, selection.Equals, selection.DoubleEquals: |
| return metav1.LabelSelectorOpIn, nil |
| case selection.NotIn, selection.NotEquals: |
| return metav1.LabelSelectorOpNotIn, nil |
| case selection.Exists: |
| return metav1.LabelSelectorOpExists, nil |
| case selection.DoesNotExist: |
| return metav1.LabelSelectorOpDoesNotExist, nil |
| } |
| return "", fmt.Errorf("unsupported label selector operator: %s", operator) |
| } |