Skip to content

Commit

Permalink
refactor: reduce duplication in sync target functions
Browse files Browse the repository at this point in the history
Signed-off-by: Erik Godding Boye <[email protected]>
  • Loading branch information
erikgb committed Sep 17, 2024
1 parent 6156b4c commit 38dd335
Showing 1 changed file with 76 additions and 118 deletions.
194 changes: 76 additions & 118 deletions pkg/bundle/target.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import (
"strings"

"github.com/go-logr/logr"
corev1 "k8s.io/api/core/v1"
apierrors "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/types"
Expand Down Expand Up @@ -53,91 +54,68 @@ func (b *bundle) syncConfigMapTarget(
resolvedBundle targetData,
shouldExist bool,
) (bool, error) {
configMap := &metav1.PartialObjectMetadata{
targetObj := &metav1.PartialObjectMetadata{
TypeMeta: metav1.TypeMeta{
Kind: "ConfigMap",
APIVersion: "v1",
},
}
err := b.targetCache.Get(ctx, name, configMap)
err := b.targetCache.Get(ctx, name, targetObj)
if err != nil && !apierrors.IsNotFound(err) {
return false, fmt.Errorf("failed to get ConfigMap %s: %w", name, err)
}

// If the ConfigMap exists, but the Bundle is being deleted, delete the ConfigMap.
// If the ConfigMap is not found, and should not exist, we are done.
if apierrors.IsNotFound(err) && !shouldExist {
return false, nil
}

// If the ConfigMap should not exist, but it does, delete it.
// If the ConfigMap exists, but should not, delete it.
if !apierrors.IsNotFound(err) && !shouldExist {
// apply empty patch to remove the key
configMapPatch := coreapplyconfig.
ConfigMap(name.Name, name.Namespace).
WithLabels(map[string]string{
trustapi.BundleLabelKey: bundle.Name,
}).
WithOwnerReferences(
metav1applyconfig.OwnerReference().
WithAPIVersion(trustapi.SchemeGroupVersion.String()).
WithKind(trustapi.BundleKind).
WithName(bundle.GetName()).
WithUID(bundle.GetUID()).
WithBlockOwnerDeletion(true).
WithController(true),
)

if err = b.patchConfigMapResource(ctx, configMapPatch); err != nil {
// Apply empty patch to remove the key(s).
configMapPatch := prepareTargetPatch(coreapplyconfig.ConfigMap(name.Name, name.Namespace), *bundle)
configMap, err := b.patchConfigMap(ctx, configMapPatch)
if err != nil {
return false, fmt.Errorf("failed to patch ConfigMap %s: %w", name, err)
}

// If the ConfigMap is empty, delete it.
if configMap != nil && len(configMap.Data) == 0 && len(configMap.BinaryData) == 0 {
return true, b.client.Delete(ctx, configMap)
}
return true, nil
}

target := bundle.Spec.Target
if target.ConfigMap == nil {
bundleTarget := bundle.Spec.Target
if bundleTarget.ConfigMap == nil {
return false, errors.New("target not defined")
}

// Generated JKS is not deterministic - best we can do here is update if the pem cert has
// changed (hence not checking if JKS matches)
dataHash := fmt.Sprintf("%x", sha256.Sum256([]byte(resolvedBundle.data)))
configmapData := map[string]string{
target.ConfigMap.Key: resolvedBundle.data,
configMapData := map[string]string{
bundleTarget.ConfigMap.Key: resolvedBundle.data,
}
configmapBinData := resolvedBundle.binaryData
configMapBinData := resolvedBundle.binaryData

// If the ConfigMap doesn't exist, create it.
if !apierrors.IsNotFound(err) {
// Exit early if no update is needed
if exit, err := b.needsUpdate(ctx, targetKindConfigMap, log, configMap, bundle, dataHash); err != nil {
if exit, err := b.needsUpdate(ctx, targetKindConfigMap, log, targetObj, bundle, dataHash); err != nil {
return false, err
} else if !exit {
return false, nil
}
}

configMapPatch := coreapplyconfig.
ConfigMap(name.Name, name.Namespace).
WithLabels(map[string]string{
trustapi.BundleLabelKey: bundle.Name,
}).
configMapPatch := prepareTargetPatch(coreapplyconfig.ConfigMap(name.Name, name.Namespace), *bundle).
WithAnnotations(map[string]string{
trustapi.BundleHashAnnotationKey: dataHash,
}).
WithOwnerReferences(
metav1applyconfig.OwnerReference().
WithAPIVersion(trustapi.SchemeGroupVersion.String()).
WithKind(trustapi.BundleKind).
WithName(bundle.GetName()).
WithUID(bundle.GetUID()).
WithBlockOwnerDeletion(true).
WithController(true),
).
WithData(configmapData).
WithBinaryData(configmapBinData)
WithData(configMapData).
WithBinaryData(configMapBinData)

if err = b.patchConfigMapResource(ctx, configMapPatch); err != nil {
if _, err = b.patchConfigMap(ctx, configMapPatch); err != nil {
return false, fmt.Errorf("failed to patch ConfigMap %s: %w", name, err)
}

Expand All @@ -158,93 +136,70 @@ func (b *bundle) syncSecretTarget(
resolvedBundle targetData,
shouldExist bool,
) (bool, error) {
secret := &metav1.PartialObjectMetadata{
targetObj := &metav1.PartialObjectMetadata{
TypeMeta: metav1.TypeMeta{
Kind: "Secret",
APIVersion: "v1",
},
}
err := b.targetCache.Get(ctx, name, secret)
err := b.targetCache.Get(ctx, name, targetObj)
if err != nil && !apierrors.IsNotFound(err) {
return false, fmt.Errorf("failed to get Secret %s: %w", name, err)
}

// If the target obj exists, but the Bundle is being deleted, delete the Secret.
// If the Secret is not found, and should not exist, we are done.
if apierrors.IsNotFound(err) && !shouldExist {
return false, nil
}

// If the Secret should not exist, but it does, delete it.
// If the Secret exists, but should not, delete it.
if !apierrors.IsNotFound(err) && !shouldExist {
// apply empty patch to remove the key
secretPatch := coreapplyconfig.
Secret(name.Name, name.Namespace).
WithLabels(map[string]string{
trustapi.BundleLabelKey: bundle.Name,
}).
WithOwnerReferences(
metav1applyconfig.OwnerReference().
WithAPIVersion(trustapi.SchemeGroupVersion.String()).
WithKind(trustapi.BundleKind).
WithName(bundle.GetName()).
WithUID(bundle.GetUID()).
WithBlockOwnerDeletion(true).
WithController(true),
)

if err = b.patchSecretResource(ctx, secretPatch); err != nil {
return false, fmt.Errorf("failed to patch secret %s: %w", name, err)
// Apply empty patch to remove the key(s).
patch := prepareTargetPatch(coreapplyconfig.Secret(name.Name, name.Namespace), *bundle)
secret, err := b.patchSecret(ctx, patch)
if err != nil {
return false, fmt.Errorf("failed to patch Secret %s: %w", name, err)
}
// If the Secret is empty, delete it.
if secret != nil && len(secret.Data) == 0 {
return true, b.client.Delete(ctx, secret)
}

return true, nil
}

target := bundle.Spec.Target
if target.Secret == nil {
bundleTarget := bundle.Spec.Target
if bundleTarget.Secret == nil {
return false, errors.New("target not defined")
}

// Generated JKS is not deterministic - best we can do here is update if the pem cert has
// changed (hence not checking if JKS matches)
dataHash := fmt.Sprintf("%x", sha256.Sum256([]byte(resolvedBundle.data)))
targetData := map[string][]byte{
target.Secret.Key: []byte(resolvedBundle.data),
secretData := map[string][]byte{
bundleTarget.Secret.Key: []byte(resolvedBundle.data),
}

for k, v := range resolvedBundle.binaryData {
targetData[k] = v
secretData[k] = v
}

// If the Secret doesn't exist, create it.
if !apierrors.IsNotFound(err) {
// Exit early if no update is needed
if exit, err := b.needsUpdate(ctx, targetKindSecret, log, secret, bundle, dataHash); err != nil {
if exit, err := b.needsUpdate(ctx, targetKindSecret, log, targetObj, bundle, dataHash); err != nil {
return false, err
} else if !exit {
return false, nil
}
}

secretPatch := coreapplyconfig.
Secret(name.Name, name.Namespace).
WithLabels(map[string]string{
trustapi.BundleLabelKey: bundle.Name,
}).
secretPatch := prepareTargetPatch(coreapplyconfig.Secret(name.Name, name.Namespace), *bundle).
WithAnnotations(map[string]string{
trustapi.BundleHashAnnotationKey: dataHash,
}).
WithOwnerReferences(
metav1applyconfig.OwnerReference().
WithAPIVersion(trustapi.SchemeGroupVersion.String()).
WithKind(trustapi.BundleKind).
WithName(bundle.GetName()).
WithUID(bundle.GetUID()).
WithBlockOwnerDeletion(true).
WithController(true),
).
WithData(targetData)
WithData(secretData)

if err = b.patchSecretResource(ctx, secretPatch); err != nil {
if _, err = b.patchSecret(ctx, secretPatch); err != nil {
return false, fmt.Errorf("failed to patch Secret %s: %w", name, err)
}

Expand Down Expand Up @@ -350,50 +305,53 @@ func listManagedProperties(configmap *metav1.PartialObjectMetadata, fieldManager
return properties, nil
}

func (b *bundle) patchConfigMapResource(ctx context.Context, applyConfig *coreapplyconfig.ConfigMapApplyConfiguration) error {
func (b *bundle) patchConfigMap(ctx context.Context, applyConfig *coreapplyconfig.ConfigMapApplyConfiguration) (*corev1.ConfigMap, error) {
if b.patchResourceOverwrite != nil {
return b.patchResourceOverwrite(ctx, applyConfig)
return nil, b.patchResourceOverwrite(ctx, applyConfig)
}

configMap, patch, err := ssa_client.GenerateConfigMapPatch(applyConfig)
target, patch, err := ssa_client.GenerateConfigMapPatch(applyConfig)
if err != nil {
return fmt.Errorf("failed to generate patch: %w", err)
}

err = b.client.Patch(ctx, configMap, patch, ssa_client.FieldManager, client.ForceOwnership)
if err != nil {
return err
}

// If the configMap is empty, delete it
if len(configMap.Data) == 0 && len(configMap.BinaryData) == 0 {
return b.client.Delete(ctx, configMap)
return nil, fmt.Errorf("failed to generate patch: %w", err)
}

return nil
return target, b.client.Patch(ctx, target, patch, ssa_client.FieldManager, client.ForceOwnership)
}

func (b *bundle) patchSecretResource(ctx context.Context, applyConfig *coreapplyconfig.SecretApplyConfiguration) error {
func (b *bundle) patchSecret(ctx context.Context, applyConfig *coreapplyconfig.SecretApplyConfiguration) (*corev1.Secret, error) {
if b.patchResourceOverwrite != nil {
return b.patchResourceOverwrite(ctx, applyConfig)
return nil, b.patchResourceOverwrite(ctx, applyConfig)
}

secret, patch, err := ssa_client.GenerateSecretPatch(applyConfig)
target, patch, err := ssa_client.GenerateSecretPatch(applyConfig)
if err != nil {
return fmt.Errorf("failed to generate patch: %w", err)
return nil, fmt.Errorf("failed to generate patch: %w", err)
}

err = b.client.Patch(ctx, secret, patch, ssa_client.FieldManager, client.ForceOwnership)
if err != nil {
return err
}
return target, b.client.Patch(ctx, target, patch, ssa_client.FieldManager, client.ForceOwnership)
}

// If the secret is empty, delete it
if len(secret.Data) == 0 {
return b.client.Delete(ctx, secret)
}
type targetApplyConfiguration[T any] interface {
*coreapplyconfig.ConfigMapApplyConfiguration | *coreapplyconfig.SecretApplyConfiguration

return nil
WithLabels(entries map[string]string) T
WithOwnerReferences(values ...*metav1applyconfig.OwnerReferenceApplyConfiguration) T
}

func prepareTargetPatch[T targetApplyConfiguration[T]](target T, bundle trustapi.Bundle) T {
return target.
WithLabels(map[string]string{
trustapi.BundleLabelKey: bundle.Name,
}).
WithOwnerReferences(
metav1applyconfig.OwnerReference().
WithAPIVersion(trustapi.SchemeGroupVersion.String()).
WithKind(trustapi.BundleKind).
WithName(bundle.GetName()).
WithUID(bundle.GetUID()).
WithBlockOwnerDeletion(true).
WithController(true),
)
}

type targetData struct {
Expand Down

0 comments on commit 38dd335

Please sign in to comment.