argo-cd/util/argo/resource_tracking.go
Leonardo Luz Almeida 0cdd44bda7
chore: fix gitops-engine mod name so it can be referenced by external repos (#26407)
Signed-off-by: Leonardo Luz Almeida <leonardo_almeida@intuit.com>
2026-02-12 09:29:40 -05:00

287 lines
9.6 KiB
Go

package argo
import (
"errors"
"fmt"
"regexp"
"strings"
kubeutil "github.com/argoproj/argo-cd/gitops-engine/pkg/utils/kube"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"github.com/argoproj/argo-cd/v3/common"
"github.com/argoproj/argo-cd/v3/pkg/apis/application/v1alpha1"
"github.com/argoproj/argo-cd/v3/util/kube"
)
var (
ErrWrongResourceTrackingFormat = errors.New("wrong resource tracking format, should be <application-name>:<group>/<kind>:<namespace>/<name>")
LabelMaxLength = 63
OkEndPattern = regexp.MustCompile("[a-zA-Z0-9]$")
)
// ResourceTracking defines methods which allow setup and retrieve tracking information to resource
type ResourceTracking interface {
GetAppName(un *unstructured.Unstructured, key string, trackingMethod v1alpha1.TrackingMethod, installationID string) string
GetAppInstance(un *unstructured.Unstructured, trackingMethod v1alpha1.TrackingMethod, installationID string) *AppInstanceValue
SetAppInstance(un *unstructured.Unstructured, key, val, namespace string, trackingMethod v1alpha1.TrackingMethod, instanceID string) error
BuildAppInstanceValue(value AppInstanceValue) string
ParseAppInstanceValue(value string) (*AppInstanceValue, error)
Normalize(config, live *unstructured.Unstructured, labelKey, trackingMethod string) error
RemoveAppInstance(un *unstructured.Unstructured, trackingMethod string) error
}
// AppInstanceValue store information about resource tracking info
type AppInstanceValue struct {
ApplicationName string
Group string
Kind string
Namespace string
Name string
}
type resourceTracking struct{}
func NewResourceTracking() ResourceTracking {
return &resourceTracking{}
}
func IsOldTrackingMethod(trackingMethod string) bool {
return trackingMethod == "" || trackingMethod == string(v1alpha1.TrackingMethodLabel)
}
func (rt *resourceTracking) getAppInstanceValue(un *unstructured.Unstructured, installationID string) *AppInstanceValue {
if installationID != "" && un.GetAnnotations() == nil || un.GetAnnotations()[common.AnnotationInstallationID] != installationID {
return nil
}
appInstanceAnnotation, err := kube.GetAppInstanceAnnotation(un, common.AnnotationKeyAppInstance)
if err != nil {
return nil
}
value, err := rt.ParseAppInstanceValue(appInstanceAnnotation)
if err != nil {
return nil
}
return value
}
// GetAppName retrieve application name base on tracking method
func (rt *resourceTracking) GetAppName(un *unstructured.Unstructured, key string, trackingMethod v1alpha1.TrackingMethod, instanceID string) string {
retrieveAppInstanceValue := func() string {
value := rt.getAppInstanceValue(un, instanceID)
if value != nil {
return value.ApplicationName
}
return ""
}
switch trackingMethod {
case v1alpha1.TrackingMethodLabel:
label, err := kube.GetAppInstanceLabel(un, key)
if err != nil {
return ""
}
return label
case v1alpha1.TrackingMethodAnnotationAndLabel:
return retrieveAppInstanceValue()
case v1alpha1.TrackingMethodAnnotation:
return retrieveAppInstanceValue()
default:
return retrieveAppInstanceValue()
}
}
// GetAppInstance returns the representation of the app instance annotation.
// If the tracking method does not support metadata, or the annotation could
// not be parsed, it returns nil.
func (rt *resourceTracking) GetAppInstance(un *unstructured.Unstructured, trackingMethod v1alpha1.TrackingMethod, instanceID string) *AppInstanceValue {
switch trackingMethod {
case v1alpha1.TrackingMethodAnnotation, v1alpha1.TrackingMethodAnnotationAndLabel:
return rt.getAppInstanceValue(un, instanceID)
default:
return nil
}
}
// UnstructuredToAppInstanceValue will build the AppInstanceValue based
// on the provided unstructured. The given namespace works as a default
// value if the resource's namespace is not defined. It should be the
// Application's target destination namespace.
func UnstructuredToAppInstanceValue(un *unstructured.Unstructured, appName, namespace string) AppInstanceValue {
ns := un.GetNamespace()
if ns == "" {
ns = namespace
}
gvk := un.GetObjectKind().GroupVersionKind()
return AppInstanceValue{
ApplicationName: appName,
Group: gvk.Group,
Kind: gvk.Kind,
Namespace: ns,
Name: un.GetName(),
}
}
// SetAppInstance set label/annotation base on tracking method
func (rt *resourceTracking) SetAppInstance(un *unstructured.Unstructured, key, val, namespace string, trackingMethod v1alpha1.TrackingMethod, instanceID string) error {
setAppInstanceAnnotation := func() error {
appInstanceValue := UnstructuredToAppInstanceValue(un, val, namespace)
if instanceID != "" {
if err := kube.SetAppInstanceAnnotation(un, common.AnnotationInstallationID, instanceID); err != nil {
return err
}
} else {
if err := kube.RemoveAnnotation(un, common.AnnotationInstallationID); err != nil {
return err
}
}
return kube.SetAppInstanceAnnotation(un, common.AnnotationKeyAppInstance, rt.BuildAppInstanceValue(appInstanceValue))
}
switch trackingMethod {
case v1alpha1.TrackingMethodLabel:
err := kube.SetAppInstanceLabel(un, key, val)
if err != nil {
return fmt.Errorf("failed to set app instance label: %w", err)
}
return nil
case v1alpha1.TrackingMethodAnnotation:
return setAppInstanceAnnotation()
case v1alpha1.TrackingMethodAnnotationAndLabel:
err := setAppInstanceAnnotation()
if err != nil {
return err
}
if len(val) > LabelMaxLength {
val = val[:LabelMaxLength]
// Prevent errors if the truncated name ends in a special character.
// See https://github.com/argoproj/argo-cd/issues/18237.
for !OkEndPattern.MatchString(val) {
if len(val) <= 1 {
return errors.New("failed to set app instance label: unable to truncate label to not end with a special character")
}
val = val[:len(val)-1]
}
}
err = kube.SetAppInstanceLabel(un, key, val)
if err != nil {
return fmt.Errorf("failed to set app instance label: %w", err)
}
return nil
default:
return setAppInstanceAnnotation()
}
}
func (rt *resourceTracking) RemoveAppInstance(un *unstructured.Unstructured, trackingMethod string) error {
switch v1alpha1.TrackingMethod(trackingMethod) {
case v1alpha1.TrackingMethodLabel:
if err := kube.RemoveLabel(un, common.LabelKeyAppInstance); err != nil {
return err
}
return nil
case v1alpha1.TrackingMethodAnnotation:
if err := kube.RemoveAnnotation(un, common.AnnotationKeyAppInstance); err != nil {
return err
}
if err := kube.RemoveAnnotation(un, common.AnnotationInstallationID); err != nil {
return err
}
return nil
case v1alpha1.TrackingMethodAnnotationAndLabel:
if err := kube.RemoveAnnotation(un, common.AnnotationKeyAppInstance); err != nil {
return err
}
if err := kube.RemoveAnnotation(un, common.AnnotationInstallationID); err != nil {
return err
}
if err := kube.RemoveLabel(un, common.LabelKeyAppInstance); err != nil {
return err
}
return nil
default:
// By default, only app instance annotations are set and not labels
// hence the default case should be only to remove annotations and not labels
if err := kube.RemoveAnnotation(un, common.AnnotationKeyAppInstance); err != nil {
return err
}
if err := kube.RemoveAnnotation(un, common.AnnotationInstallationID); err != nil {
return err
}
}
return nil
}
// BuildAppInstanceValue build resource tracking id in format <application-name>;<group>/<kind>/<namespace>/<name>
func (rt *resourceTracking) BuildAppInstanceValue(value AppInstanceValue) string {
return fmt.Sprintf("%s:%s/%s:%s/%s", value.ApplicationName, value.Group, value.Kind, value.Namespace, value.Name)
}
// ParseAppInstanceValue parse resource tracking id from format <application-name>:<group>/<kind>:<namespace>/<name> to struct
func (rt *resourceTracking) ParseAppInstanceValue(value string) (*AppInstanceValue, error) {
var appInstanceValue AppInstanceValue
parts := strings.SplitN(value, ":", 3)
appInstanceValue.ApplicationName = parts[0]
if len(parts) != 3 {
return nil, ErrWrongResourceTrackingFormat
}
groupParts := strings.Split(parts[1], "/")
if len(groupParts) != 2 {
return nil, ErrWrongResourceTrackingFormat
}
nsParts := strings.Split(parts[2], "/")
if len(nsParts) != 2 {
return nil, ErrWrongResourceTrackingFormat
}
appInstanceValue.Group = groupParts[0]
appInstanceValue.Kind = groupParts[1]
appInstanceValue.Namespace = nsParts[0]
appInstanceValue.Name = nsParts[1]
return &appInstanceValue, nil
}
// Normalize updates live resource and removes diff caused by missing annotation or extra tracking label.
// The normalization is required to ensure smooth transition to new tracking method.
func (rt *resourceTracking) Normalize(config, live *unstructured.Unstructured, labelKey, trackingMethod string) error {
if IsOldTrackingMethod(trackingMethod) {
return nil
}
if live == nil || config == nil {
return nil
}
label, err := kube.GetAppInstanceLabel(live, labelKey)
if err != nil {
return fmt.Errorf("failed to get app instance label: %w", err)
}
if label == "" {
return nil
}
if kubeutil.IsCRD(live) {
// CRDs don't get tracking annotations.
return nil
}
annotation, err := kube.GetAppInstanceAnnotation(config, common.AnnotationKeyAppInstance)
if err != nil {
return err
}
err = kube.SetAppInstanceAnnotation(live, common.AnnotationKeyAppInstance, annotation)
if err != nil {
return err
}
label, err = kube.GetAppInstanceLabel(config, labelKey)
if err != nil {
return fmt.Errorf("failed to get app instance label: %w", err)
}
if label == "" {
err = kube.RemoveLabel(live, labelKey)
if err != nil {
return fmt.Errorf("failed to remove app instance label: %w", err)
}
}
return nil
}