From 64fbfdb6caab19b086c9579ddf5b2312ffad2ce8 Mon Sep 17 00:00:00 2001 From: John Belamaric Date: Tue, 13 Jun 2023 16:17:39 -0700 Subject: [PATCH 1/5] Add a check for deltas before saving --- controllers/pkg/porch/util/packagerevision.go | 35 +++++++++++++++++++ .../generic-specializer/reconciler.go | 23 +++++++++++- 2 files changed, 57 insertions(+), 1 deletion(-) create mode 100644 controllers/pkg/porch/util/packagerevision.go diff --git a/controllers/pkg/porch/util/packagerevision.go b/controllers/pkg/porch/util/packagerevision.go new file mode 100644 index 00000000..42b885e1 --- /dev/null +++ b/controllers/pkg/porch/util/packagerevision.go @@ -0,0 +1,35 @@ +/* + Copyright 2023 The Nephio Authors. + + Licensed 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 util + +import ( + "context" + "crypto/sha1" + "encoding/hex" + + porchv1alpha1 "github.com/GoogleContainerTools/kpt/porch/api/porch/v1alpha1" + "sigs.k8s.io/kustomize/kyaml/yaml" +) + +func PackageRevisionResourcesHash(prr *porchv1alpha1.PackageRevisionResources) (string, error) { + b, err := yaml.Marshal(prr.Spec.Resources) + if err != nil { + return "", err + } + hash := sha1.Sum(b) + return hex.EncodeToString(hash[:]), nil +} diff --git a/controllers/pkg/reconcilers/generic-specializer/reconciler.go b/controllers/pkg/reconcilers/generic-specializer/reconciler.go index 6ba3358b..84ecc77a 100644 --- a/controllers/pkg/reconcilers/generic-specializer/reconciler.go +++ b/controllers/pkg/reconcilers/generic-specializer/reconciler.go @@ -148,12 +148,20 @@ func (r *reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu r.l.Error(err, "cannot get package revision resources") return ctrl.Result{}, errors.Wrap(err, "cannot get package revision resources") } + + beforeHash, err := util.PackageRevisionResourcesHash(prr) + if err != nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate pre-reconcile hash: %s", err.Error())) + r.l.Error(err, "cannot calculate pre-reconcile hash") + return ctrl.Result{}, nil + } + // get resourceList from resources rl, err := kptrl.GetResourceList(prr.Spec.Resources) if err != nil { r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot get resourceList: %s", err.Error())) r.l.Error(err, "cannot get resourceList") - return ctrl.Result{}, errors.Wrap(err, "cannot get resourceList") + return ctrl.Result{}, nil } if porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.ipamFor)) { @@ -261,6 +269,19 @@ func (r *reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu return ctrl.Result{}, nil } pr.Status.Conditions = porchcondition.GetPorchConditions(kptf.GetConditions()) + + afterHash, err := util.PackageRevisionResourcesHash(prr) + if err != nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate post-reconcile hash: %s", err.Error())) + r.l.Error(err, "cannot calculate post-reconcile hash") + return ctrl.Result{}, nil + } + + if beforeHash == afterHash { + r.recorder.Event(pr, corev1.EventTypeNormal, "Skipping", "no change needed") + return ctrl.Result{}, nil + } + if err = r.porchClient.Update(ctx, prr); err != nil { return ctrl.Result{}, err } From 7fc3a2608c25b56e4bf051afb41fb7aacdd9409f Mon Sep 17 00:00:00 2001 From: John Belamaric Date: Tue, 13 Jun 2023 16:59:19 -0700 Subject: [PATCH 2/5] Fix build errors --- controllers/pkg/porch/util/packagerevision.go | 1 - controllers/pkg/reconcilers/generic-specializer/reconciler.go | 4 ++-- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/controllers/pkg/porch/util/packagerevision.go b/controllers/pkg/porch/util/packagerevision.go index 42b885e1..0e47a11c 100644 --- a/controllers/pkg/porch/util/packagerevision.go +++ b/controllers/pkg/porch/util/packagerevision.go @@ -17,7 +17,6 @@ package util import ( - "context" "crypto/sha1" "encoding/hex" diff --git a/controllers/pkg/reconcilers/generic-specializer/reconciler.go b/controllers/pkg/reconcilers/generic-specializer/reconciler.go index 84ecc77a..431a6dcc 100644 --- a/controllers/pkg/reconcilers/generic-specializer/reconciler.go +++ b/controllers/pkg/reconcilers/generic-specializer/reconciler.go @@ -149,7 +149,7 @@ func (r *reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu return ctrl.Result{}, errors.Wrap(err, "cannot get package revision resources") } - beforeHash, err := util.PackageRevisionResourcesHash(prr) + beforeHash, err := porchutil.PackageRevisionResourcesHash(prr) if err != nil { r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate pre-reconcile hash: %s", err.Error())) r.l.Error(err, "cannot calculate pre-reconcile hash") @@ -270,7 +270,7 @@ func (r *reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu } pr.Status.Conditions = porchcondition.GetPorchConditions(kptf.GetConditions()) - afterHash, err := util.PackageRevisionResourcesHash(prr) + afterHash, err := porchutil.PackageRevisionResourcesHash(prr) if err != nil { r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate post-reconcile hash: %s", err.Error())) r.l.Error(err, "cannot calculate post-reconcile hash") From 8f16fdfbd0ef3c13a7f0bdf7fc076bb7beb68c2f Mon Sep 17 00:00:00 2001 From: John Belamaric Date: Tue, 13 Jun 2023 17:08:58 -0700 Subject: [PATCH 3/5] Use sha512 --- controllers/pkg/porch/util/packagerevision.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/controllers/pkg/porch/util/packagerevision.go b/controllers/pkg/porch/util/packagerevision.go index 0e47a11c..c4944977 100644 --- a/controllers/pkg/porch/util/packagerevision.go +++ b/controllers/pkg/porch/util/packagerevision.go @@ -17,7 +17,7 @@ package util import ( - "crypto/sha1" + "crypto/sha512" "encoding/hex" porchv1alpha1 "github.com/GoogleContainerTools/kpt/porch/api/porch/v1alpha1" @@ -29,6 +29,6 @@ func PackageRevisionResourcesHash(prr *porchv1alpha1.PackageRevisionResources) ( if err != nil { return "", err } - hash := sha1.Sum(b) + hash := sha512.Sum512(b) return hex.EncodeToString(hash[:]), nil } From eccdd1a1f024c458f05151134b169ea1648a8a47 Mon Sep 17 00:00:00 2001 From: John Belamaric Date: Tue, 13 Jun 2023 17:58:32 -0700 Subject: [PATCH 4/5] Cleaner indentation --- .../generic-specializer/reconciler.go | 251 +++++++++--------- 1 file changed, 128 insertions(+), 123 deletions(-) diff --git a/controllers/pkg/reconcilers/generic-specializer/reconciler.go b/controllers/pkg/reconcilers/generic-specializer/reconciler.go index 431a6dcc..e604d0f4 100644 --- a/controllers/pkg/reconcilers/generic-specializer/reconciler.go +++ b/controllers/pkg/reconcilers/generic-specializer/reconciler.go @@ -138,154 +138,159 @@ func (r *reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu // we just check for forResource conditions and we dont care if it is satisfied already // this allows us to refresh the allocation. - if porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.ipamFor)) || - porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.vlanFor)) { - - // get package revision resourceList - prr := &porchv1alpha1.PackageRevisionResources{} - if err := r.porchClient.Get(ctx, req.NamespacedName, prr); err != nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot get package revision resources: %s", err.Error())) - r.l.Error(err, "cannot get package revision resources") - return ctrl.Result{}, errors.Wrap(err, "cannot get package revision resources") - } + hasIpam := porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.ipamFor)) + hasVlan := porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.vlanFor)) + + if !hasIpam && !hasVlan { + // we do not care about it + return ctrl.Result{}, nil + } - beforeHash, err := porchutil.PackageRevisionResourcesHash(prr) + // get package revision resourceList + prr := &porchv1alpha1.PackageRevisionResources{} + if err := r.porchClient.Get(ctx, req.NamespacedName, prr); err != nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot get package revision resources: %s", err.Error())) + r.l.Error(err, "cannot get package revision resources") + return ctrl.Result{}, errors.Wrap(err, "cannot get package revision resources") + } + + beforeHash, err := porchutil.PackageRevisionResourcesHash(prr) + if err != nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate pre-reconcile hash: %s", err.Error())) + r.l.Error(err, "cannot calculate pre-reconcile hash") + return ctrl.Result{}, nil + } + + // get resourceList from resources + rl, err := kptrl.GetResourceList(prr.Spec.Resources) + if err != nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot get resourceList: %s", err.Error())) + r.l.Error(err, "cannot get resourceList") + return ctrl.Result{}, nil + } + + if porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.ipamFor)) { + // run the function SDK + _, err = r.ipamkrmfn.Process(rl) if err != nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate pre-reconcile hash: %s", err.Error())) - r.l.Error(err, "cannot calculate pre-reconcile hash") + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("ipam function: %s", err.Error())) + r.l.Error(err, "ipam function run failed") return ctrl.Result{}, nil } - - // get resourceList from resources - rl, err := kptrl.GetResourceList(prr.Spec.Resources) + r.l.Info("ipam specializer fn run successfull") + } + if porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.vlanFor)) { + // run the function SDK + _, err = r.vlankrmfn.Process(rl) if err != nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot get resourceList: %s", err.Error())) - r.l.Error(err, "cannot get resourceList") + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("vlan function: %s", err.Error())) + r.l.Error(err, "vlan function run failed") return ctrl.Result{}, nil } + r.l.Info("vlan specializer fn run successfull") + } + workloadClusterObjs := rl.Items.Where(fn.IsGroupVersionKind(infrav1alpha1.WorkloadClusterGroupVersionKind)) + clusterName := r.getClusterName(workloadClusterObjs) + + // We want to process the functions to refresh the claims + // but if the package is in publish state the updates cannot be done + // so we stop here + if porchv1alpha1.LifecycleIsPublished(pr.Spec.Lifecycle) { + r.recorder.Event(pr, corev1.EventTypeNormal, "CannotRefreshClaims", "package is published, no update possible") + r.l.Info("package is published, no updates possible", + "repo", pr.Spec.RepositoryName, + "package", pr.Spec.PackageName, + "rev", pr.Spec.Revision, + "clusterName", clusterName, + ) + return ctrl.Result{}, nil + } - if porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.ipamFor)) { - // run the function SDK - _, err = r.ipamkrmfn.Process(rl) + for _, o := range rl.Items { + // TBD what if we create new resources + // update only the resource we act upon + if o.GetAPIVersion() == r.ipamFor.APIVersion && o.GetKind() == r.ipamFor.Kind { + prr.Spec.Resources[o.GetAnnotation(kioutil.PathAnnotation)] = o.String() + // Debug + alloc, err := kubeobject.NewFromKubeObject[ipamv1alpha1.IPClaim](o) if err != nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("ipam function: %s", err.Error())) - r.l.Error(err, "ipam function run failed") - return ctrl.Result{}, nil + r.l.Error(err, "cannot get extended kubeobject") + continue } - r.l.Info("ipam specializer fn run successfull") - } - if porchcondition.HasSpecificTypeConditions(pr.Status.Conditions, kptfilelibv1.GetConditionType(&r.vlanFor)) { - // run the function SDK - _, err = r.vlankrmfn.Process(rl) + ipAlloc, err := alloc.GetGoStruct() if err != nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("vlan function: %s", err.Error())) - r.l.Error(err, "vlan function run failed") - return ctrl.Result{}, nil + r.l.Error(err, "cannot get gostruct from kubeobject") + continue } - r.l.Info("vlan specializer fn run successfull") + r.l.Info("generic specializer ip allocation", "clusterName", clusterName, "status", ipAlloc.Status) } - workloadClusterObjs := rl.Items.Where(fn.IsGroupVersionKind(infrav1alpha1.WorkloadClusterGroupVersionKind)) - clusterName := r.getClusterName(workloadClusterObjs) - - // We want to process the functions to refresh the claims - // but if the package is in publish state the updates cannot be done - // so we stop here - if porchv1alpha1.LifecycleIsPublished(pr.Spec.Lifecycle) { - r.recorder.Event(pr, corev1.EventTypeNormal, "CannotRefreshClaims", "package is published, no update possible") - r.l.Info("package is published, no updates possible", - "repo", pr.Spec.RepositoryName, - "package", pr.Spec.PackageName, - "rev", pr.Spec.Revision, - "clusterName", clusterName, - ) - return ctrl.Result{}, nil + if o.GetAPIVersion() == r.vlanFor.APIVersion && o.GetKind() == r.vlanFor.Kind { + prr.Spec.Resources[o.GetAnnotation(kioutil.PathAnnotation)] = o.String() + // Debug + alloc, err := kubeobject.NewFromKubeObject[vlanv1alpha1.VLANClaim](o) + if err != nil { + r.l.Error(err, "cannot get extended kubeobject") + continue + } + vlanAlloc, err := alloc.GetGoStruct() + if err != nil { + r.l.Error(err, "cannot get gostruct from kubeobject") + continue + } + r.l.Info("generic specializer vlan allocation", "cluserName", clusterName, "status", vlanAlloc.Status) } - - for _, o := range rl.Items { - // TBD what if we create new resources - // update only the resource we act upon - if o.GetAPIVersion() == r.ipamFor.APIVersion && o.GetKind() == r.ipamFor.Kind { - prr.Spec.Resources[o.GetAnnotation(kioutil.PathAnnotation)] = o.String() - // Debug - alloc, err := kubeobject.NewFromKubeObject[ipamv1alpha1.IPClaim](o) - if err != nil { - r.l.Error(err, "cannot get extended kubeobject") - continue - } - ipAlloc, err := alloc.GetGoStruct() - if err != nil { - r.l.Error(err, "cannot get gostruct from kubeobject") - continue - } - r.l.Info("generic specializer ip allocation", "clusterName", clusterName, "status", ipAlloc.Status) + if o.GetAPIVersion() == "kpt.dev/v1" && o.GetKind() == "Kptfile" { + prr.Spec.Resources[o.GetAnnotation(kioutil.PathAnnotation)] = o.String() + kptf, err := kubeobject.NewFromKubeObject[kptv1.KptFile](o) + if err != nil { + r.l.Error(err, "cannot get extended kubeobject") + continue } - if o.GetAPIVersion() == r.vlanFor.APIVersion && o.GetKind() == r.vlanFor.Kind { - prr.Spec.Resources[o.GetAnnotation(kioutil.PathAnnotation)] = o.String() - // Debug - alloc, err := kubeobject.NewFromKubeObject[vlanv1alpha1.VLANClaim](o) - if err != nil { - r.l.Error(err, "cannot get extended kubeobject") - continue - } - vlanAlloc, err := alloc.GetGoStruct() - if err != nil { - r.l.Error(err, "cannot get gostruct from kubeobject") - continue - } - r.l.Info("generic specializer vlan allocation", "cluserName", clusterName, "status", vlanAlloc.Status) + kptfile, err := kptf.GetGoStruct() + if err != nil { + r.l.Error(err, "cannot get gostruct from kubeobject") + continue } - if o.GetAPIVersion() == "kpt.dev/v1" && o.GetKind() == "Kptfile" { - prr.Spec.Resources[o.GetAnnotation(kioutil.PathAnnotation)] = o.String() - kptf, err := kubeobject.NewFromKubeObject[kptv1.KptFile](o) - if err != nil { - r.l.Error(err, "cannot get extended kubeobject") - continue - } - kptfile, err := kptf.GetGoStruct() - if err != nil { - r.l.Error(err, "cannot get gostruct from kubeobject") - continue - } - for _, c := range kptfile.Status.Conditions { - if strings.HasPrefix(c.Type, kptfilelibv1.GetConditionType(&r.vlanFor)+".") || - strings.HasPrefix(c.Type, kptfilelibv1.GetConditionType(&r.ipamFor)+".") { - r.l.Info("generic specializer conditions", "cluserName", clusterName, "status", c.Status, "condition", c.Type) - } + for _, c := range kptfile.Status.Conditions { + if strings.HasPrefix(c.Type, kptfilelibv1.GetConditionType(&r.vlanFor)+".") || + strings.HasPrefix(c.Type, kptfilelibv1.GetConditionType(&r.ipamFor)+".") { + r.l.Info("generic specializer conditions", "cluserName", clusterName, "status", c.Status, "condition", c.Type) } } } + } - kptfile := rl.Items.GetRootKptfile() - if kptfile == nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", "mandatory Kptfile is missing") - r.l.Error(fmt.Errorf("mandatory Kptfile is missing from the package"), "") - return ctrl.Result{}, nil - } + kptfile := rl.Items.GetRootKptfile() + if kptfile == nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", "mandatory Kptfile is missing") + r.l.Error(fmt.Errorf("mandatory Kptfile is missing from the package"), "") + return ctrl.Result{}, nil + } - kptf, err := kptfilelibv1.New(rl.Items.GetRootKptfile().String()) - if err != nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", "cannot unmarshal Kptfile") - r.l.Error(err, "cannot unmarshal kptfile") - return ctrl.Result{}, nil - } - pr.Status.Conditions = porchcondition.GetPorchConditions(kptf.GetConditions()) + kptf, err := kptfilelibv1.New(rl.Items.GetRootKptfile().String()) + if err != nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", "cannot unmarshal Kptfile") + r.l.Error(err, "cannot unmarshal kptfile") + return ctrl.Result{}, nil + } + pr.Status.Conditions = porchcondition.GetPorchConditions(kptf.GetConditions()) - afterHash, err := porchutil.PackageRevisionResourcesHash(prr) - if err != nil { - r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate post-reconcile hash: %s", err.Error())) - r.l.Error(err, "cannot calculate post-reconcile hash") - return ctrl.Result{}, nil - } + afterHash, err := porchutil.PackageRevisionResourcesHash(prr) + if err != nil { + r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", fmt.Sprintf("cannot calculate post-reconcile hash: %s", err.Error())) + r.l.Error(err, "cannot calculate post-reconcile hash") + return ctrl.Result{}, nil + } - if beforeHash == afterHash { - r.recorder.Event(pr, corev1.EventTypeNormal, "Skipping", "no change needed") - return ctrl.Result{}, nil - } + if beforeHash == afterHash { + r.recorder.Event(pr, corev1.EventTypeNormal, "Skipping", "no change needed") + return ctrl.Result{}, nil + } - if err = r.porchClient.Update(ctx, prr); err != nil { - return ctrl.Result{}, err - } + if err = r.porchClient.Update(ctx, prr); err != nil { + return ctrl.Result{}, err } + return ctrl.Result{}, nil } From 94be99fa72edf99f5c823b47871ed59350a67061 Mon Sep 17 00:00:00 2001 From: John Belamaric Date: Wed, 14 Jun 2023 06:48:59 -0700 Subject: [PATCH 5/5] Comment out code I think may not be used --- .../pkg/reconcilers/generic-specializer/reconciler.go | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/controllers/pkg/reconcilers/generic-specializer/reconciler.go b/controllers/pkg/reconcilers/generic-specializer/reconciler.go index e604d0f4..925dc9df 100644 --- a/controllers/pkg/reconcilers/generic-specializer/reconciler.go +++ b/controllers/pkg/reconcilers/generic-specializer/reconciler.go @@ -254,12 +254,15 @@ func (r *reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu for _, c := range kptfile.Status.Conditions { if strings.HasPrefix(c.Type, kptfilelibv1.GetConditionType(&r.vlanFor)+".") || strings.HasPrefix(c.Type, kptfilelibv1.GetConditionType(&r.ipamFor)+".") { - r.l.Info("generic specializer conditions", "cluserName", clusterName, "status", c.Status, "condition", c.Type) + r.l.Info("generic specializer conditions", "clusterName", clusterName, "status", c.Status, "condition", c.Type) } } } } + /* + * jbelamaric: Since we never save the PR, I don't think this does anything? + * kptfile := rl.Items.GetRootKptfile() if kptfile == nil { r.recorder.Event(pr, corev1.EventTypeWarning, "ReconcileError", "mandatory Kptfile is missing") @@ -274,6 +277,7 @@ func (r *reconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Resu return ctrl.Result{}, nil } pr.Status.Conditions = porchcondition.GetPorchConditions(kptf.GetConditions()) + */ afterHash, err := porchutil.PackageRevisionResourcesHash(prr) if err != nil {