v2/api/kusto/customizations/cluster_extensions.go (50 lines of code) (raw):
/*
* Copyright (c) Microsoft Corporation.
* Licensed under the MIT license.
*/
package customizations
import (
"context"
"fmt"
"strings"
"github.com/go-logr/logr"
"github.com/rotisserie/eris"
"sigs.k8s.io/controller-runtime/pkg/conversion"
kusto "github.com/Azure/azure-service-operator/v2/api/kusto/v1api20230815/storage"
"github.com/Azure/azure-service-operator/v2/internal/genericarmclient"
"github.com/Azure/azure-service-operator/v2/internal/resolver"
"github.com/Azure/azure-service-operator/v2/internal/set"
"github.com/Azure/azure-service-operator/v2/pkg/genruntime"
"github.com/Azure/azure-service-operator/v2/pkg/genruntime/extensions"
)
var _ extensions.PreReconciliationChecker = &ClusterExtension{}
// clusterTerminalStates is the set of provisioning states that are considered terminal for a cluster.
// These are all listed lowercase, so we can do a case-insensitive match.
var clusterTerminalStates = set.Make(
"succeeded",
"failed",
"canceled",
)
// PreReconcileCheck is called before the reconciliation of the resource to see if the cluster
// is in a state that will allow reconciliation to proceed.
// If a cluster has a provisioningState not in the set above, it will reject any attempt to PUT a
// new state out of hand; so there's no point in even trying. This is true even if the PUT we're
// doing will have no effect on the state of the cluster.
func (ext *ClusterExtension) PreReconcileCheck(
ctx context.Context,
obj genruntime.MetaObject,
owner genruntime.MetaObject,
resourceResolver *resolver.Resolver,
armClient *genericarmclient.GenericClient,
log logr.Logger,
next extensions.PreReconcileCheckFunc,
) (extensions.PreReconcileCheckResult, error) {
// This has to be the current hub storage version. It will need to be updated
// if the hub storage version changes.
cluster, ok := obj.(*kusto.Cluster)
if !ok {
return extensions.PreReconcileCheckResult{},
eris.Errorf("cannot run on unknown resource type %T, expected *kusto.Cluster", obj)
}
// Type assert that we are the hub type. This will fail to compile if
// the hub type has been changed but this extension has not
var _ conversion.Hub = cluster
// If the cluster is in a state that will reject any PUT, then we should skip reconciliation
// as there's no point in even trying.
// This allows us to "play nice with others" and not use up request quota attempting to make changes when we
// already know those attempts will fail.
state := cluster.Status.ProvisioningState
if state != nil && clusterProvisioningStateBlocksReconciliation(state) {
return extensions.BlockReconcile(
fmt.Sprintf("Cluster is in provisioning state %q", *state)),
nil
}
return next(ctx, obj, owner, resourceResolver, armClient, log)
}
func clusterProvisioningStateBlocksReconciliation(provisioningState *string) bool {
if provisioningState == nil {
return false
}
return !clusterTerminalStates.Contains(strings.ToLower(*provisioningState))
}