Skip to content

Commit

Permalink
Cleanup component dependencies when removed
Browse files Browse the repository at this point in the history
* if component is removed from management object
  the related helmreleases, providertemplates and
  helmcharts are deleted now
* small code amends
* test coverage for the mgmt-ctrl raised
  from 5% to 45%

Closes #553
  • Loading branch information
zerospiel committed Oct 31, 2024
1 parent 992d221 commit f46f509
Show file tree
Hide file tree
Showing 3 changed files with 358 additions and 76 deletions.
201 changes: 129 additions & 72 deletions internal/controller/management_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,15 +41,6 @@ import (
"github.com/Mirantis/hmc/internal/utils"
)

// Those are only needed for the initial installation
var enforcedValues = map[string]any{
"controller": map[string]any{
"createManagement": false,
"createTemplateManagement": false,
"createRelease": false,
},
}

// ManagementReconciler reconciles a Management object
type ManagementReconciler struct {
client.Client
Expand Down Expand Up @@ -87,92 +78,135 @@ func (r *ManagementReconciler) Update(ctx context.Context, management *hmc.Manag

if controllerutil.AddFinalizer(management, hmc.ManagementFinalizer) {
if err := r.Client.Update(ctx, management); err != nil {
l.Error(err, "Failed to update Management finalizers")
l.Error(err, "failed to update Management finalizers")
return ctrl.Result{}, err
}
return ctrl.Result{}, nil
}

if err := r.ensureTemplateManagement(ctx, management); err != nil {
l.Error(err, "Failed to ensure TemplateManagement is created")
if err := r.cleanupRemovedComponents(ctx, management); err != nil {
l.Error(err, "failed to cleanup removed components")
return ctrl.Result{}, err
}

release := &hmc.Release{}
if err := r.Client.Get(ctx, client.ObjectKey{Name: management.Spec.Release}, release); err != nil {
l.Error(err, "failed to get Release object")
if err := r.ensureTemplateManagement(ctx, management); err != nil {
l.Error(err, "failed to ensure TemplateManagement is created")
return ctrl.Result{}, err
}

var (
errs error

detectedProviders = hmc.Providers{}
detectedComponents = make(map[string]hmc.ComponentStatus)
detectedContracts = make(map[string]hmc.CompatibilityContracts)
)

err := r.enableAdditionalComponents(ctx, management)
if err != nil {
if err := r.enableAdditionalComponents(ctx, management); err != nil { // TODO (zerospiel): i wonder, do we need to reflect these changes and changes from the `wrappedComponents` in the spec?
l.Error(err, "failed to enable additional HMC components")
return ctrl.Result{}, err
}

components, err := wrappedComponents(management, release)
components, err := getWrappedComponents(ctx, r.Client, management)
if err != nil {
l.Error(err, "failed to wrap HMC components")
return ctrl.Result{}, err
}

var (
errs error

statusAccumulator = &mgmtStatusAccumulator{
providers: hmc.Providers{},
components: make(map[string]hmc.ComponentStatus),
compatibilityContracts: make(map[string]hmc.CompatibilityContracts),
}
)
for _, component := range components {
template := &hmc.ProviderTemplate{}
err := r.Get(ctx, client.ObjectKey{
Name: component.Template,
}, template)
if err != nil {
l.V(1).Info("reconciling components", "component", component)
template := new(hmc.ProviderTemplate)
if err := r.Get(ctx, client.ObjectKey{Name: component.Template}, template); err != nil {
errMsg := fmt.Sprintf("Failed to get ProviderTemplate %s: %s", component.Template, err)
updateComponentsStatus(detectedComponents, &detectedProviders, detectedContracts, component.helmReleaseName, component.Template, template.Status.Providers, template.Status.CAPIContracts, errMsg)
updateComponentsStatus(statusAccumulator, component, nil, errMsg)
errs = errors.Join(errs, errors.New(errMsg))

continue
}

if !template.Status.Valid {
errMsg := fmt.Sprintf("Template %s is not marked as valid", component.Template)
updateComponentsStatus(detectedComponents, &detectedProviders, detectedContracts, component.helmReleaseName, component.Template, template.Status.Providers, template.Status.CAPIContracts, errMsg)
updateComponentsStatus(statusAccumulator, component, nil, errMsg)
errs = errors.Join(errs, errors.New(errMsg))

continue
}

_, _, err = helm.ReconcileHelmRelease(ctx, r.Client, component.helmReleaseName, r.SystemNamespace, helm.ReconcileHelmReleaseOpts{
if _, _, err := helm.ReconcileHelmRelease(ctx, r.Client, component.helmReleaseName, r.SystemNamespace, helm.ReconcileHelmReleaseOpts{
Values: component.Config,
ChartRef: template.Status.ChartRef,
DependsOn: component.dependsOn,
TargetNamespace: component.targetNamespace,
CreateNamespace: component.createNamespace,
})
if err != nil {
errMsg := fmt.Sprintf("error reconciling HelmRelease %s/%s: %s", r.SystemNamespace, component.Template, err)
updateComponentsStatus(detectedComponents, &detectedProviders, detectedContracts, component.helmReleaseName, component.Template, template.Status.Providers, template.Status.CAPIContracts, errMsg)
}); err != nil {
errMsg := fmt.Sprintf("Failed to reconcile HelmRelease %s/%s: %s", r.SystemNamespace, component.helmReleaseName, err)
updateComponentsStatus(statusAccumulator, component, nil, errMsg)
errs = errors.Join(errs, errors.New(errMsg))

continue
}
updateComponentsStatus(detectedComponents, &detectedProviders, detectedContracts, component.helmReleaseName, component.Template, template.Status.Providers, template.Status.CAPIContracts, "")

updateComponentsStatus(statusAccumulator, component, template, "")
}

management.Status.AvailableProviders = statusAccumulator.providers
management.Status.CAPIContracts = statusAccumulator.compatibilityContracts
management.Status.Components = statusAccumulator.components
management.Status.ObservedGeneration = management.Generation
management.Status.AvailableProviders = detectedProviders
management.Status.CAPIContracts = detectedContracts
management.Status.Components = detectedComponents
management.Status.Release = management.Spec.Release

if err := r.Status().Update(ctx, management); err != nil {
errs = errors.Join(errs, fmt.Errorf("failed to update status for Management %s: %w",
management.Name, err))
errs = errors.Join(errs, fmt.Errorf("failed to update status for Management %s: %w", management.Name, err))
}

if errs != nil {
l.Error(errs, "Multiple errors during Management reconciliation")
return ctrl.Result{}, errs
}

return ctrl.Result{}, nil
}

func (r *ManagementReconciler) cleanupRemovedComponents(ctx context.Context, management *hmc.Management) error {
var (
errs error
l = ctrl.LoggerFrom(ctx)
)

for compName, compSt := range management.Status.Components {
if compName == hmc.CoreCAPIName ||
compName == hmc.CoreHMCName ||
slices.ContainsFunc(management.Spec.Providers, func(newComp hmc.Provider) bool { return compName == newComp.Name }) {
continue
}

l.Info("Found component to remove", "component_name", compName)

// removing providertemplate will also delete its ownees, including HelmCharts
p := new(hmc.ProviderTemplate)
p.SetGroupVersionKind(hmc.GroupVersion.WithKind(hmc.ProviderTemplateKind))
p.SetName(compSt.Template)
if err := r.Client.Delete(ctx, p); client.IgnoreNotFound(err) != nil {
errs = errors.Join(errs, fmt.Errorf("failed to delete %s %s: %w", p.GroupVersionKind(), p.Name, err))
continue
}
l.Info("Removed ProviderTemplate", "reference", client.ObjectKeyFromObject(p).String())

hr := new(fluxv2.HelmRelease)
hr.SetGroupVersionKind(fluxv2.GroupVersion.WithKind(fluxv2.HelmReleaseKind))
hr.SetName(compName)
hr.SetNamespace(r.SystemNamespace)
if err := r.Client.Delete(ctx, hr); client.IgnoreNotFound(err) != nil {
errs = errors.Join(errs, fmt.Errorf("failed to delete %s %s: %w", hr.GroupVersionKind(), client.ObjectKeyFromObject(hr), err))
continue
}
l.Info("Removed HelmRelease", "reference", client.ObjectKeyFromObject(hr).String())
}

return errs
}

func (r *ManagementReconciler) ensureTemplateManagement(ctx context.Context, mgmt *hmc.Management) error {
l := ctrl.LoggerFrom(ctx)
if !r.CreateTemplateManagement {
Expand Down Expand Up @@ -295,6 +329,16 @@ func applyHMCDefaults(config *apiextensionsv1.JSON) (*apiextensionsv1.JSON, erro
return nil, err
}
}

// Those are only needed for the initial installation
enforcedValues := map[string]any{
"controller": map[string]any{
"createManagement": false,
"createTemplateManagement": false,
"createRelease": false,
},
}

chartutil.CoalesceTables(values, enforcedValues)
raw, err := json.Marshal(values)
if err != nil {
Expand All @@ -303,10 +347,16 @@ func applyHMCDefaults(config *apiextensionsv1.JSON) (*apiextensionsv1.JSON, erro
return &apiextensionsv1.JSON{Raw: raw}, nil
}

func wrappedComponents(mgmt *hmc.Management, release *hmc.Release) ([]component, error) {
func getWrappedComponents(ctx context.Context, cl client.Client, mgmt *hmc.Management) ([]component, error) {
if mgmt.Spec.Core == nil {
return nil, nil
}

release := &hmc.Release{}
if err := cl.Get(ctx, client.ObjectKey{Name: mgmt.Spec.Release}, release); err != nil {
return nil, fmt.Errorf("failed to get Release %s: %w", mgmt.Spec.Release, err)
}

components := make([]component, 0, len(mgmt.Spec.Providers)+2)
hmcComp := component{Component: mgmt.Spec.Core.HMC, helmReleaseName: hmc.CoreHMCName}
if hmcComp.Template == "" {
Expand Down Expand Up @@ -358,9 +408,8 @@ func (r *ManagementReconciler) enableAdditionalComponents(ctx context.Context, m
config := make(map[string]any)

if hmcComponent.Config != nil {
err := json.Unmarshal(hmcComponent.Config.Raw, &config)
if err != nil {
return fmt.Errorf("failed to unmarshal HMC config into map[string]any: %v", err)
if err := json.Unmarshal(hmcComponent.Config.Raw, &config); err != nil {
return fmt.Errorf("failed to unmarshal HMC config into map[string]any: %w", err)
}
}

Expand All @@ -384,13 +433,15 @@ func (r *ManagementReconciler) enableAdditionalComponents(ctx context.Context, m
capiOperatorValues = v
}

err := certmanager.VerifyAPI(ctx, r.Config, r.SystemNamespace)
if err != nil {
return fmt.Errorf("failed to check in the cert-manager API is installed: %v", err)
if r.Config != nil {
if err := certmanager.VerifyAPI(ctx, r.Config, r.SystemNamespace); err != nil {
return fmt.Errorf("failed to check in the cert-manager API is installed: %w", err)
}

l.Info("Cert manager is installed, enabling the HMC admission webhook")
admissionWebhookValues["enabled"] = true
}
l.Info("Cert manager is installed, enabling the HMC admission webhook")

admissionWebhookValues["enabled"] = true
config["admissionWebhook"] = admissionWebhookValues

// Enable HMC capi operator only if it was not explicitly disabled in the config to
Expand All @@ -407,37 +458,43 @@ func (r *ManagementReconciler) enableAdditionalComponents(ctx context.Context, m

updatedConfig, err := json.Marshal(config)
if err != nil {
return fmt.Errorf("failed to marshal HMC config: %v", err)
}
hmcComponent.Config = &apiextensionsv1.JSON{
Raw: updatedConfig,
return fmt.Errorf("failed to marshal HMC config: %w", err)
}

hmcComponent.Config = &apiextensionsv1.JSON{Raw: updatedConfig}

return nil
}

type mgmtStatusAccumulator struct {
components map[string]hmc.ComponentStatus
compatibilityContracts map[string]hmc.CompatibilityContracts
providers hmc.Providers
}

func updateComponentsStatus(
components map[string]hmc.ComponentStatus,
providers *hmc.Providers,
capiContracts map[string]hmc.CompatibilityContracts,
componentName string,
templateName string,
templateProviders hmc.Providers,
templateContracts hmc.CompatibilityContracts,
stAcc *mgmtStatusAccumulator,
comp component,
template *hmc.ProviderTemplate,
err string,
) {
components[componentName] = hmc.ComponentStatus{
if stAcc == nil {
return
}

stAcc.components[comp.helmReleaseName] = hmc.ComponentStatus{
Error: err,
Success: err == "",
Template: templateName,
Template: comp.Component.Template,
}

if err == "" {
*providers = append(*providers, templateProviders...)
slices.Sort(*providers)
*providers = slices.Compact(*providers)
if err == "" && template != nil {
stAcc.providers = append(stAcc.providers, template.Status.Providers...)
slices.Sort(stAcc.providers)
stAcc.providers = slices.Compact(stAcc.providers)

for _, v := range templateProviders {
capiContracts[v] = templateContracts
for _, v := range template.Status.Providers {
stAcc.compatibilityContracts[v] = template.Status.CAPIContracts
}
}
}
Expand Down
Loading

0 comments on commit f46f509

Please sign in to comment.