mirror of
https://github.com/argoproj/argo-cd.git
synced 2026-02-20 01:28:45 +01:00
Signed-off-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> Signed-off-by: Michael Crenshaw <350466+crenshaw-dev@users.noreply.github.com> Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
204 lines
7.8 KiB
Go
204 lines
7.8 KiB
Go
package utils
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
stderrors "errors"
|
|
"fmt"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
"k8s.io/apimachinery/pkg/api/errors"
|
|
"k8s.io/apimachinery/pkg/api/resource"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
"k8s.io/apimachinery/pkg/conversion"
|
|
"k8s.io/apimachinery/pkg/fields"
|
|
"k8s.io/apimachinery/pkg/labels"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
|
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
|
|
|
|
argov1alpha1 "github.com/argoproj/argo-cd/v3/pkg/apis/application/v1alpha1"
|
|
"github.com/argoproj/argo-cd/v3/util/argo"
|
|
argodiff "github.com/argoproj/argo-cd/v3/util/argo/diff"
|
|
"github.com/argoproj/argo-cd/v3/util/argo/normalizers"
|
|
)
|
|
|
|
// CreateOrUpdate overrides "sigs.k8s.io/controller-runtime" function
|
|
// in sigs.k8s.io/controller-runtime/pkg/controller/controllerutil/controllerutil.go
|
|
// to add equality for argov1alpha1.ApplicationDestination
|
|
// argov1alpha1.ApplicationDestination has a private variable, so the default
|
|
// implementation fails to compare it.
|
|
//
|
|
// CreateOrUpdate creates or updates the given object in the Kubernetes
|
|
// cluster. The object's desired state must be reconciled with the existing
|
|
// state inside the passed in callback MutateFn.
|
|
//
|
|
// The MutateFn is called regardless of creating or updating an object.
|
|
//
|
|
// It returns the executed operation and an error.
|
|
func CreateOrUpdate(ctx context.Context, logCtx *log.Entry, c client.Client, ignoreAppDifferences argov1alpha1.ApplicationSetIgnoreDifferences, ignoreNormalizerOpts normalizers.IgnoreNormalizerOpts, obj *argov1alpha1.Application, f controllerutil.MutateFn) (controllerutil.OperationResult, error) {
|
|
key := client.ObjectKeyFromObject(obj)
|
|
if err := c.Get(ctx, key, obj); err != nil {
|
|
if !errors.IsNotFound(err) {
|
|
return controllerutil.OperationResultNone, err
|
|
}
|
|
if err := mutate(f, key, obj); err != nil {
|
|
return controllerutil.OperationResultNone, err
|
|
}
|
|
if err := c.Create(ctx, obj); err != nil {
|
|
return controllerutil.OperationResultNone, err
|
|
}
|
|
return controllerutil.OperationResultCreated, nil
|
|
}
|
|
|
|
normalizedLive := obj.DeepCopy()
|
|
|
|
// Mutate the live object to match the desired state.
|
|
if err := mutate(f, key, obj); err != nil {
|
|
return controllerutil.OperationResultNone, err
|
|
}
|
|
|
|
// Apply ignoreApplicationDifferences rules to remove ignored fields from both the live and the desired state. This
|
|
// prevents those differences from appearing in the diff and therefore in the patch.
|
|
err := applyIgnoreDifferences(ignoreAppDifferences, normalizedLive, obj, ignoreNormalizerOpts)
|
|
if err != nil {
|
|
return controllerutil.OperationResultNone, fmt.Errorf("failed to apply ignore differences: %w", err)
|
|
}
|
|
|
|
// Normalize to avoid diffing on unimportant differences.
|
|
normalizedLive.Spec = *argo.NormalizeApplicationSpec(&normalizedLive.Spec)
|
|
obj.Spec = *argo.NormalizeApplicationSpec(&obj.Spec)
|
|
|
|
equality := conversion.EqualitiesOrDie(
|
|
func(a, b resource.Quantity) bool {
|
|
// Ignore formatting, only care that numeric value stayed the same.
|
|
// TODO: if we decide it's important, it should be safe to start comparing the format.
|
|
//
|
|
// Uninitialized quantities are equivalent to 0 quantities.
|
|
return a.Cmp(b) == 0
|
|
},
|
|
func(a, b metav1.MicroTime) bool {
|
|
return a.UTC().Equal(b.UTC())
|
|
},
|
|
func(a, b metav1.Time) bool {
|
|
return a.UTC().Equal(b.UTC())
|
|
},
|
|
func(a, b labels.Selector) bool {
|
|
return a.String() == b.String()
|
|
},
|
|
func(a, b fields.Selector) bool {
|
|
return a.String() == b.String()
|
|
},
|
|
func(a, b argov1alpha1.ApplicationDestination) bool {
|
|
return a.Namespace == b.Namespace && a.Name == b.Name && a.Server == b.Server
|
|
},
|
|
)
|
|
|
|
if equality.DeepEqual(normalizedLive, obj) {
|
|
return controllerutil.OperationResultNone, nil
|
|
}
|
|
|
|
patch := client.MergeFrom(normalizedLive)
|
|
if log.IsLevelEnabled(log.DebugLevel) {
|
|
LogPatch(logCtx, patch, obj)
|
|
}
|
|
if err := c.Patch(ctx, obj, patch); err != nil {
|
|
return controllerutil.OperationResultNone, err
|
|
}
|
|
return controllerutil.OperationResultUpdated, nil
|
|
}
|
|
|
|
func LogPatch(logCtx *log.Entry, patch client.Patch, obj *argov1alpha1.Application) {
|
|
patchBytes, err := patch.Data(obj)
|
|
if err != nil {
|
|
logCtx.Errorf("failed to generate patch: %v", err)
|
|
}
|
|
// Get the patch as a plain object so it is easier to work with in json logs.
|
|
var patchObj map[string]any
|
|
err = json.Unmarshal(patchBytes, &patchObj)
|
|
if err != nil {
|
|
logCtx.Errorf("failed to unmarshal patch: %v", err)
|
|
}
|
|
logCtx.WithField("patch", patchObj).Debug("patching application")
|
|
}
|
|
|
|
// mutate wraps a MutateFn and applies validation to its result
|
|
func mutate(f controllerutil.MutateFn, key client.ObjectKey, obj client.Object) error {
|
|
if err := f(); err != nil {
|
|
return fmt.Errorf("error while wrapping using MutateFn: %w", err)
|
|
}
|
|
if newKey := client.ObjectKeyFromObject(obj); key != newKey {
|
|
return stderrors.New("MutateFn cannot mutate object name and/or object namespace")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// applyIgnoreDifferences applies the ignore differences rules to the found application. It modifies the applications in place.
|
|
func applyIgnoreDifferences(applicationSetIgnoreDifferences argov1alpha1.ApplicationSetIgnoreDifferences, found *argov1alpha1.Application, generatedApp *argov1alpha1.Application, ignoreNormalizerOpts normalizers.IgnoreNormalizerOpts) error {
|
|
if len(applicationSetIgnoreDifferences) == 0 {
|
|
return nil
|
|
}
|
|
|
|
generatedAppCopy := generatedApp.DeepCopy()
|
|
diffConfig, err := argodiff.NewDiffConfigBuilder().
|
|
WithDiffSettings(applicationSetIgnoreDifferences.ToApplicationIgnoreDifferences(), nil, false, ignoreNormalizerOpts).
|
|
WithNoCache().
|
|
Build()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to build diff config: %w", err)
|
|
}
|
|
unstructuredFound, err := appToUnstructured(found)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to convert found application to unstructured: %w", err)
|
|
}
|
|
unstructuredGenerated, err := appToUnstructured(generatedApp)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to convert found application to unstructured: %w", err)
|
|
}
|
|
result, err := argodiff.Normalize([]*unstructured.Unstructured{unstructuredFound}, []*unstructured.Unstructured{unstructuredGenerated}, diffConfig)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to normalize application spec: %w", err)
|
|
}
|
|
if len(result.Lives) != 1 {
|
|
return fmt.Errorf("expected 1 normalized application, got %d", len(result.Lives))
|
|
}
|
|
foundJSONNormalized, err := json.Marshal(result.Lives[0].Object)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal normalized app to json: %w", err)
|
|
}
|
|
foundNormalized := &argov1alpha1.Application{}
|
|
err = json.Unmarshal(foundJSONNormalized, &foundNormalized)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to unmarshal normalized app to json: %w", err)
|
|
}
|
|
if len(result.Targets) != 1 {
|
|
return fmt.Errorf("expected 1 normalized application, got %d", len(result.Targets))
|
|
}
|
|
foundNormalized.DeepCopyInto(found)
|
|
generatedJSONNormalized, err := json.Marshal(result.Targets[0].Object)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal normalized app to json: %w", err)
|
|
}
|
|
generatedAppNormalized := &argov1alpha1.Application{}
|
|
err = json.Unmarshal(generatedJSONNormalized, &generatedAppNormalized)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to unmarshal normalized app json to structured app: %w", err)
|
|
}
|
|
generatedAppNormalized.DeepCopyInto(generatedApp)
|
|
// Prohibit jq queries from mutating silly things.
|
|
generatedApp.TypeMeta = generatedAppCopy.TypeMeta
|
|
generatedApp.Name = generatedAppCopy.Name
|
|
generatedApp.Namespace = generatedAppCopy.Namespace
|
|
generatedApp.Operation = generatedAppCopy.Operation
|
|
return nil
|
|
}
|
|
|
|
func appToUnstructured(app client.Object) (*unstructured.Unstructured, error) {
|
|
u, err := runtime.DefaultUnstructuredConverter.ToUnstructured(app)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert app object to unstructured: %w", err)
|
|
}
|
|
return &unstructured.Unstructured{Object: u}, nil
|
|
}
|