Skip to content

Commit

Permalink
Merge pull request #4128 from shysank/feature/4020
Browse files Browse the repository at this point in the history
✨ Support range of values for unhealthy machines in machine health check spec
  • Loading branch information
k8s-ci-robot authored Mar 10, 2021
2 parents febfe5b + 7d53242 commit d18ddcd
Show file tree
Hide file tree
Showing 10 changed files with 341 additions and 45 deletions.
31 changes: 29 additions & 2 deletions api/v1alpha3/conversion.go
Original file line number Diff line number Diff line change
Expand Up @@ -152,13 +152,36 @@ func (dst *MachineDeploymentList) ConvertFrom(srcRaw conversion.Hub) error {
func (src *MachineHealthCheck) ConvertTo(dstRaw conversion.Hub) error {
dst := dstRaw.(*v1alpha4.MachineHealthCheck)

return Convert_v1alpha3_MachineHealthCheck_To_v1alpha4_MachineHealthCheck(src, dst, nil)
if err := Convert_v1alpha3_MachineHealthCheck_To_v1alpha4_MachineHealthCheck(src, dst, nil); err != nil {
return err
}

// Manually restore data.
restored := &v1alpha4.MachineHealthCheck{}
if ok, err := utilconversion.UnmarshalData(src, restored); err != nil || !ok {
return err
}

if restored.Spec.UnhealthyRange != nil {
dst.Spec.UnhealthyRange = restored.Spec.UnhealthyRange
}

return nil
}

func (dst *MachineHealthCheck) ConvertFrom(srcRaw conversion.Hub) error {
src := srcRaw.(*v1alpha4.MachineHealthCheck)

return Convert_v1alpha4_MachineHealthCheck_To_v1alpha3_MachineHealthCheck(src, dst, nil)
if err := Convert_v1alpha4_MachineHealthCheck_To_v1alpha3_MachineHealthCheck(src, dst, nil); err != nil {
return err
}

// Preserve Hub data on down-conversion except for metadata
if err := utilconversion.MarshalData(src, dst); err != nil {
return err
}

return nil
}

func (src *MachineHealthCheckList) ConvertTo(dstRaw conversion.Hub) error {
Expand All @@ -181,3 +204,7 @@ func Convert_v1alpha3_Bootstrap_To_v1alpha4_Bootstrap(in *Bootstrap, out *v1alph
func Convert_v1alpha4_MachineRollingUpdateDeployment_To_v1alpha3_MachineRollingUpdateDeployment(in *v1alpha4.MachineRollingUpdateDeployment, out *MachineRollingUpdateDeployment, s apiconversion.Scope) error {
return autoConvert_v1alpha4_MachineRollingUpdateDeployment_To_v1alpha3_MachineRollingUpdateDeployment(in, out, s)
}

func Convert_v1alpha4_MachineHealthCheckSpec_To_v1alpha3_MachineHealthCheckSpec(in *v1alpha4.MachineHealthCheckSpec, out *MachineHealthCheckSpec, s apiconversion.Scope) error {
return autoConvert_v1alpha4_MachineHealthCheckSpec_To_v1alpha3_MachineHealthCheckSpec(in, out, s)
}
1 change: 1 addition & 0 deletions api/v1alpha3/conversion_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ func TestFuzzyConversion(t *testing.T) {
t.Run("for Machine", utilconversion.FuzzTestFunc(scheme, &v1alpha4.Machine{}, &Machine{}, BootstrapFuzzFuncs))
t.Run("for MachineSet", utilconversion.FuzzTestFunc(scheme, &v1alpha4.MachineSet{}, &MachineSet{}, BootstrapFuzzFuncs))
t.Run("for MachineDeployment", utilconversion.FuzzTestFunc(scheme, &v1alpha4.MachineDeployment{}, &MachineDeployment{}, BootstrapFuzzFuncs))
t.Run("for MachineHealthCheckSpec", utilconversion.FuzzTestFunc(scheme, &v1alpha4.MachineHealthCheck{}, &MachineHealthCheck{}))
}

func BootstrapFuzzFuncs(_ runtimeserializer.CodecFactory) []interface{} {
Expand Down
40 changes: 28 additions & 12 deletions api/v1alpha3/zz_generated.conversion.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

9 changes: 9 additions & 0 deletions api/v1alpha4/machinehealthcheck_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,15 @@ type MachineHealthCheckSpec struct {
// +optional
MaxUnhealthy *intstr.IntOrString `json:"maxUnhealthy,omitempty"`

// Any further remediation is only allowed if the number of machines selected by "selector" as not healthy
// is within the range of "UnhealthyRange". Takes precedence over MaxUnhealthy.
// Eg. "[3-5]" - This means that remediation will be allowed only when:
// (a) there are at least 3 unhealthy machines (and)
// (b) there are at most 5 unhealthy machines
// +optional
// +kubebuilder:validation:Pattern=^\[[0-9]+-[0-9]+\]$
UnhealthyRange *string `json:"unhealthyRange,omitempty"`

// Machines older than this duration without a node will be considered to have
// failed and will be remediated.
// +optional
Expand Down
5 changes: 5 additions & 0 deletions api/v1alpha4/zz_generated.deepcopy.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 4 additions & 0 deletions config/crd/bases/cluster.x-k8s.io_machinehealthchecks.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -322,6 +322,10 @@ spec:
type: object
minItems: 1
type: array
unhealthyRange:
description: 'Any further remediation is only allowed if the number of machines selected by "selector" as not healthy is within the range of "UnhealthyRange". Takes precedence over MaxUnhealthy. Eg. "[3-5]" - This means that remediation will be allowed only when: (a) there are at least 3 unhealthy machines (and) (b) there are at most 5 unhealthy machines'
pattern: ^\[[0-9]+-[0-9]+\]$
type: string
required:
- clusterName
- selector
Expand Down
99 changes: 75 additions & 24 deletions controllers/machinehealthcheck_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import (
"context"
"fmt"
"sort"
"strconv"
"strings"
"time"

Expand Down Expand Up @@ -212,21 +213,41 @@ func (r *MachineHealthCheckReconciler) reconcile(ctx context.Context, logger log
healthy, unhealthy, nextCheckTimes := r.healthCheckTargets(targets, logger, m.Spec.NodeStartupTimeout.Duration)
m.Status.CurrentHealthy = int32(len(healthy))

var unhealthyLimitKey, unhealthyLimitValue interface{}

// check MHC current health against MaxUnhealthy
if !isAllowedRemediation(m) {
remediationAllowed, remediationCount, err := isAllowedRemediation(m)
if err != nil {
return ctrl.Result{}, errors.Wrapf(err, "error checking if remediation is allowed")
}

if !remediationAllowed {
var message string

if m.Spec.UnhealthyRange == nil {
unhealthyLimitKey = "max unhealthy"
unhealthyLimitValue = m.Spec.MaxUnhealthy
message = fmt.Sprintf("Remediation is not allowed, the number of not started or unhealthy machines exceeds maxUnhealthy (total: %v, unhealthy: %v, maxUnhealthy: %v)",
totalTargets,
len(unhealthy),
m.Spec.MaxUnhealthy)
} else {
unhealthyLimitKey = "unhealthy range"
unhealthyLimitValue = *m.Spec.UnhealthyRange
message = fmt.Sprintf("Remediation is not allowed, the number of not started or unhealthy machines does not fall within the range (total: %v, unhealthy: %v, unhealthyRange: %v)",
totalTargets,
len(unhealthy),
*m.Spec.UnhealthyRange)
}

logger.V(3).Info(
"Short-circuiting remediation",
"total target", totalTargets,
"max unhealthy", m.Spec.MaxUnhealthy,
unhealthyLimitKey, unhealthyLimitValue,
"unhealthy targets", len(unhealthy),
)
message := fmt.Sprintf("Remediation is not allowed, the number of not started or unhealthy machines exceeds maxUnhealthy (total: %v, unhealthy: %v, maxUnhealthy: %v)",
totalTargets,
len(unhealthy),
m.Spec.MaxUnhealthy,
)

// Remediation not allowed, the number of not started or unhealthy machines exceeds maxUnhealthy
// Remediation not allowed, the number of not started or unhealthy machines either exceeds maxUnhealthy (or) not within unhealthyRange
m.Status.RemediationsAllowed = 0
conditions.Set(m, &clusterv1.Condition{
Type: clusterv1.RemediationAllowedCondition,
Expand Down Expand Up @@ -258,17 +279,12 @@ func (r *MachineHealthCheckReconciler) reconcile(ctx context.Context, logger log
logger.V(3).Info(
"Remediations are allowed",
"total target", totalTargets,
"max unhealthy", m.Spec.MaxUnhealthy,
unhealthyLimitKey, unhealthyLimitValue,
"unhealthy targets", len(unhealthy),
)

maxUnhealthy, err := getMaxUnhealthy(m)
if err != nil {
return ctrl.Result{}, errors.Wrapf(err, "Failed to get value for maxUnhealthy")
}

// Remediation is allowed so maxUnhealthy - unhealthyMachineCount >= 0
m.Status.RemediationsAllowed = int32(maxUnhealthy - unhealthyMachineCount(m))
// Remediation is allowed so unhealthyMachineCount is within unhealthyRange (or) maxUnhealthy - unhealthyMachineCount >= 0
m.Status.RemediationsAllowed = remediationCount
conditions.MarkTrue(m, clusterv1.RemediationAllowedCondition)

errList := r.PatchUnhealthyTargets(ctx, logger, unhealthy, cluster, m)
Expand Down Expand Up @@ -519,21 +535,56 @@ func (r *MachineHealthCheckReconciler) watchClusterNodes(ctx context.Context, cl
}

// isAllowedRemediation checks the value of the MaxUnhealthy field to determine
// whether remediation should be allowed or not
func isAllowedRemediation(mhc *clusterv1.MachineHealthCheck) bool {
// TODO(JoelSpeed): return an error from isAllowedRemediation when maxUnhealthy
// is nil, we expect it to be defaulted always.
if mhc.Spec.MaxUnhealthy == nil {
return true
// returns whether remediation should be allowed or not, the remediation count, and error if any
func isAllowedRemediation(mhc *clusterv1.MachineHealthCheck) (bool, int32, error) {
var remediationAllowed bool
var remediationCount int32
if mhc.Spec.UnhealthyRange != nil {
min, max, err := getUnhealthyRange(mhc)
if err != nil {
return false, 0, err
}
unhealthyMachineCount := unhealthyMachineCount(mhc)
remediationAllowed = unhealthyMachineCount >= min && unhealthyMachineCount <= max
remediationCount = int32(max - unhealthyMachineCount)
return remediationAllowed, remediationCount, nil
}

maxUnhealthy, err := getMaxUnhealthy(mhc)
if err != nil {
return false
return false, 0, err
}

// Remediation is not allowed if unhealthy is above maxUnhealthy
return unhealthyMachineCount(mhc) <= maxUnhealthy
unhealthyMachineCount := unhealthyMachineCount(mhc)
remediationAllowed = unhealthyMachineCount <= maxUnhealthy
remediationCount = int32(maxUnhealthy - unhealthyMachineCount)
return remediationAllowed, remediationCount, nil
}

// getUnhealthyRange parses an integer range and returns the min and max values
// Eg. [2-5] will return (2,5,nil)
func getUnhealthyRange(mhc *clusterv1.MachineHealthCheck) (int, int, error) {
// remove '[' and ']'
unhealthyRange := (*(mhc.Spec.UnhealthyRange))[1 : len(*mhc.Spec.UnhealthyRange)-1]

parts := strings.Split(unhealthyRange, "-")

min, err := strconv.ParseUint(parts[0], 10, 32)
if err != nil {
return 0, 0, err
}

max, err := strconv.ParseUint(parts[1], 10, 32)
if err != nil {
return 0, 0, err
}

if max < min {
return 0, 0, errors.Errorf("max value %d cannot be less than min value %d for unhealthyRange", max, min)
}

return int(min), int(max), nil
}

func getMaxUnhealthy(mhc *clusterv1.MachineHealthCheck) (int, error) {
Expand Down
Loading

0 comments on commit d18ddcd

Please sign in to comment.