From e3cf329dd11ef17640a3dd54fcd1beea4b348c99 Mon Sep 17 00:00:00 2001 From: Ian Graham Date: Tue, 7 Jan 2025 17:34:47 +0000 Subject: [PATCH] Add resource hash annotation on Deployment --- apis/pipelines/object_hasher.go | 11 ++++ controllers/pipelines/provider_controller.go | 57 ++++++++++++++++---- 2 files changed, 57 insertions(+), 11 deletions(-) diff --git a/apis/pipelines/object_hasher.go b/apis/pipelines/object_hasher.go index b624c2f78..dc7e9f7a0 100644 --- a/apis/pipelines/object_hasher.go +++ b/apis/pipelines/object_hasher.go @@ -2,7 +2,9 @@ package pipelines import ( "crypto/sha1" + "encoding/json" "hash" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "sort" ) @@ -38,6 +40,15 @@ func (oh ObjectHasher) WriteMapField(value map[string]string) { oh.WriteFieldSeparator() } +func (oh ObjectHasher) WriteObject(obj metav1.Object) error { + bytes, err := json.Marshal(obj) + if err != nil { + return err + } + oh.h.Write(bytes) + return nil +} + type KV interface { GetKey() string GetValue() string diff --git a/controllers/pipelines/provider_controller.go b/controllers/pipelines/provider_controller.go index 2ae6874a0..ca6183b06 100644 --- a/controllers/pipelines/provider_controller.go +++ b/controllers/pipelines/provider_controller.go @@ -3,10 +3,11 @@ package pipelines import ( "context" "fmt" + "time" + apierrors "k8s.io/apimachinery/pkg/api/errors" "k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/runtime/schema" - "time" "k8s.io/apimachinery/pkg/runtime" @@ -22,6 +23,12 @@ import ( "sigs.k8s.io/controller-runtime/pkg/log" ) +const ( + OwnerNameLabel = "owner-name" + AppLabel = "app" + ResourceHashAnnotation = "resource-hash" +) + type ProviderReconciler struct { StateHandler[*pipelinesv1.Provider] ResourceReconciler[*pipelinesv1.Provider] @@ -63,7 +70,7 @@ func (r *ProviderReconciler) Reconcile(ctx context.Context, req ctrl.Request) (c return ctrl.Result{}, err } - deploy, err := r.getDeployment(ctx, req.Namespace, provider.Name, *provider) + existingDeployment, err := r.getDeployment(ctx, req.Namespace, provider.Name, *provider) if err != nil && !apierrors.IsNotFound(err) { logger.Error(err, "unable to get existing deployment") return ctrl.Result{}, err @@ -71,11 +78,20 @@ func (r *ProviderReconciler) Reconcile(ctx context.Context, req ctrl.Request) (c logger.Info("desired provider deployment", "deployment", desiredDeployment) - if deploy != nil { - logger.Info("found existing provider deployment", "deployment", deploy) - //TODO: implement updating existing deployment + if existingDeployment != nil { + logger.Info("found existing provider deployment", "deployment", existingDeployment) + if existingDeployment.Annotations != nil && existingDeployment.Annotations[ResourceHashAnnotation] != desiredDeployment.Annotations[ResourceHashAnnotation] { + logger.Info("resource hash mismatch, updating deployment") + existingDeployment.Spec = desiredDeployment.Spec + existingDeployment.SetLabels(desiredDeployment.Labels) + existingDeployment.Annotations[ResourceHashAnnotation] = desiredDeployment.Annotations[ResourceHashAnnotation] + if err = r.EC.Client.Update(ctx, existingDeployment); err != nil { + logger.Error(err, "unable to update provider service deployment", "deployment", desiredDeployment) + return ctrl.Result{}, err + } + } } else { - if err := r.EC.Client.Create(ctx, desiredDeployment); err != nil { + if err = r.EC.Client.Create(ctx, desiredDeployment); err != nil { logger.Error(err, "unable to create provider service deployment") return ctrl.Result{}, err } @@ -101,7 +117,7 @@ func (r *ProviderReconciler) getDeployment(ctx context.Context, namespace string dl := &appsv1.DeploymentList{} err := r.EC.Client.NonCached.List(ctx, dl, &client.ListOptions{ Namespace: namespace, - LabelSelector: labelSelector(map[string]string{"owner-name": fmt.Sprintf("provider-%s", providerName)}), + LabelSelector: labelSelector(map[string]string{OwnerNameLabel: fmt.Sprintf("provider-%s", providerName)}), }) if err != nil { @@ -120,8 +136,8 @@ func labelSelector(labelMap map[string]string) labels.Selector { } func (r *ProviderReconciler) constructDeployment(provider *pipelinesv1.Provider, namespace string, config config.KfpControllerConfigSpec) (*appsv1.Deployment, error) { - matchLabels := map[string]string{"app": fmt.Sprintf("provider-%s", provider.Name)} - ownerLabels := map[string]string{"owner-name": fmt.Sprintf("provider-%s", provider.Name)} + matchLabels := map[string]string{AppLabel: fmt.Sprintf("provider-%s", provider.Name)} + ownerLabels := map[string]string{OwnerNameLabel: fmt.Sprintf("provider-%s", provider.Name)} deploymentLabels := pipelines.MapConcat(pipelines.MapConcat(config.DefaultProviderValues.Labels, matchLabels), ownerLabels) replicas := int32(config.DefaultProviderValues.Replicas) @@ -148,10 +164,14 @@ func (r *ProviderReconciler) constructDeployment(provider *pipelinesv1.Provider, Template: podTemplate, }, } - err := ctrl.SetControllerReference(provider, deployment, r.Scheme) - if err != nil { + + if err := ctrl.SetControllerReference(provider, deployment, r.Scheme); err != nil { + return nil, err + } + if err := setResourceHashAnnotation(deployment); err != nil { return nil, err } + return deployment, nil } @@ -168,3 +188,18 @@ func populateMainContainer(podTemplate v1.PodTemplateSpec, provider *pipelinesv1 } return podTemplate } + +func setResourceHashAnnotation(deployment *appsv1.Deployment) error { + hasher := pipelines.NewObjectHasher() + err := hasher.WriteObject(deployment) + if err != nil { + return err + } + + if deployment.Annotations == nil { + deployment.Annotations = make(map[string]string) + } + deployment.Annotations[ResourceHashAnnotation] = fmt.Sprintf("%x", hasher.Sum()) + + return nil +}