mirror of
https://github.com/kubernetes/kubernetes.git
synced 2026-06-09 08:55:55 -04:00
2134 lines
72 KiB
Go
2134 lines
72 KiB
Go
/*
|
|
Copyright 2016 The Kubernetes 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 validation
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
|
|
"k8s.io/apimachinery/pkg/api/resource"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/kubernetes/pkg/apis/autoscaling"
|
|
api "k8s.io/kubernetes/pkg/apis/core"
|
|
"k8s.io/utils/ptr"
|
|
)
|
|
|
|
var (
|
|
hpaOpts = func(minReplica int32) HorizontalPodAutoscalerSpecValidationOptions {
|
|
return HorizontalPodAutoscalerSpecValidationOptions{
|
|
MinReplicasLowerBound: minReplica,
|
|
ScaleTargetRefValidationOptions: CrossVersionObjectReferenceValidationOptions{
|
|
AllowInvalidAPIVersion: false,
|
|
AllowEmptyAPIGroup: false,
|
|
},
|
|
ObjectMetricsValidationOptions: CrossVersionObjectReferenceValidationOptions{
|
|
AllowInvalidAPIVersion: false,
|
|
AllowEmptyAPIGroup: true,
|
|
},
|
|
}
|
|
}
|
|
hpaSpecValidationOpts = hpaOpts(1)
|
|
hpaScaleToZeroSpecValidationOpts = hpaOpts(0)
|
|
)
|
|
|
|
func TestValidateScale(t *testing.T) {
|
|
successCases := []autoscaling.Scale{{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "frontend",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.ScaleSpec{
|
|
Replicas: 1,
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "frontend",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.ScaleSpec{
|
|
Replicas: 10,
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "frontend",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.ScaleSpec{
|
|
Replicas: 0,
|
|
},
|
|
}}
|
|
|
|
for _, successCase := range successCases {
|
|
if errs := ValidateScale(&successCase); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
|
|
errorCases := []struct {
|
|
scale autoscaling.Scale
|
|
msg string
|
|
}{{
|
|
scale: autoscaling.Scale{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "frontend",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.ScaleSpec{
|
|
Replicas: -1,
|
|
},
|
|
},
|
|
msg: "must be greater than or equal to 0",
|
|
}}
|
|
|
|
for _, c := range errorCases {
|
|
if errs := ValidateScale(&c.scale); len(errs) == 0 {
|
|
t.Errorf("expected failure for %s", c.msg)
|
|
} else if !strings.Contains(errs[0].Error(), c.msg) {
|
|
t.Errorf("unexpected error: %v, expected: %s", errs[0], c.msg)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidateBehavior(t *testing.T) {
|
|
maxPolicy := autoscaling.MaxPolicySelect
|
|
minPolicy := autoscaling.MinPolicySelect
|
|
disabledPolicy := autoscaling.DisabledPolicySelect
|
|
incorrectPolicy := autoscaling.ScalingPolicySelect("incorrect")
|
|
simplePoliciesList := []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PercentScalingPolicy,
|
|
Value: 10,
|
|
PeriodSeconds: 1,
|
|
}, {
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 1,
|
|
PeriodSeconds: 1800,
|
|
}}
|
|
successCases := []autoscaling.HorizontalPodAutoscalerBehavior{{
|
|
ScaleUp: nil,
|
|
ScaleDown: nil,
|
|
}, {
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](3600),
|
|
SelectPolicy: &minPolicy,
|
|
Policies: simplePoliciesList,
|
|
},
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](0),
|
|
SelectPolicy: &disabledPolicy,
|
|
Policies: simplePoliciesList,
|
|
},
|
|
}, {
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](120),
|
|
SelectPolicy: &maxPolicy,
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 1,
|
|
PeriodSeconds: 2,
|
|
}, {
|
|
Type: autoscaling.PercentScalingPolicy,
|
|
Value: 3,
|
|
PeriodSeconds: 4,
|
|
}, {
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 5,
|
|
PeriodSeconds: 6,
|
|
}, {
|
|
Type: autoscaling.PercentScalingPolicy,
|
|
Value: 7,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](120),
|
|
SelectPolicy: &maxPolicy,
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 1,
|
|
PeriodSeconds: 2,
|
|
}, {
|
|
Type: autoscaling.PercentScalingPolicy,
|
|
Value: 3,
|
|
PeriodSeconds: 4,
|
|
}, {
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 5,
|
|
PeriodSeconds: 6,
|
|
}, {
|
|
Type: autoscaling.PercentScalingPolicy,
|
|
Value: 7,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
}}
|
|
for _, behavior := range successCases {
|
|
hpa := prepareHPAWithBehavior(behavior)
|
|
if errs := ValidateHorizontalPodAutoscaler(&hpa, hpaSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
errorCases := []struct {
|
|
behavior autoscaling.HorizontalPodAutoscalerBehavior
|
|
msg string
|
|
}{{
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
SelectPolicy: &minPolicy,
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleUp.policies: Required value: must specify at least one Policy",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](3601),
|
|
SelectPolicy: &minPolicy,
|
|
Policies: simplePoliciesList,
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleUp.stabilizationWindowSeconds: Invalid value: 3601: must be less than or equal to 3600",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 7,
|
|
PeriodSeconds: 1801,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleUp.policies[0].periodSeconds: Invalid value: 1801: must be less than or equal to 1800",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
SelectPolicy: &incorrectPolicy,
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 7,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: `spec.behavior.scaleUp.selectPolicy: Unsupported value: "incorrect": supported values: "Disabled", "Max", "Min"`,
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.HPAScalingPolicyType("hm"),
|
|
Value: 7,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: `spec.behavior.scaleUp.policies[0].type: Unsupported value: "hm": supported values: "Percent", "Pods"`,
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleUp.policies[0].periodSeconds: Invalid value: 0: must be greater than zero",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleUp.policies[0].value: Invalid value: 0: must be greater than zero",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
PeriodSeconds: -1,
|
|
Value: 1,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleUp.policies[0].periodSeconds: Invalid value: -1: must be greater than zero",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
PeriodSeconds: 1,
|
|
Value: -1,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleUp.policies[0].value: Invalid value: -1: must be greater than zero",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
SelectPolicy: &minPolicy,
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleDown.policies: Required value: must specify at least one Policy",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](3601),
|
|
SelectPolicy: &minPolicy,
|
|
Policies: simplePoliciesList,
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleDown.stabilizationWindowSeconds: Invalid value: 3601: must be less than or equal to 3600",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PercentScalingPolicy,
|
|
Value: 7,
|
|
PeriodSeconds: 1801,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleDown.policies[0].periodSeconds: Invalid value: 1801: must be less than or equal to 1800",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
SelectPolicy: &incorrectPolicy,
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 7,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: `spec.behavior.scaleDown.selectPolicy: Unsupported value: "incorrect": supported values: "Disabled", "Max", "Min"`,
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.HPAScalingPolicyType("hm"),
|
|
Value: 7,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: `spec.behavior.scaleDown.policies[0].type: Unsupported value: "hm": supported values: "Percent", "Pods"`,
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleDown.policies[0].periodSeconds: Invalid value: 0: must be greater than zero",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
PeriodSeconds: 8,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleDown.policies[0].value: Invalid value: 0: must be greater than zero",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
PeriodSeconds: -1,
|
|
Value: 1,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleDown.policies[0].periodSeconds: Invalid value: -1: must be greater than zero",
|
|
}, {
|
|
behavior: autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &autoscaling.HPAScalingRules{
|
|
Policies: []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
PeriodSeconds: 1,
|
|
Value: -1,
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.behavior.scaleDown.policies[0].value: Invalid value: -1: must be greater than zero",
|
|
}}
|
|
for _, c := range errorCases {
|
|
hpa := prepareHPAWithBehavior(c.behavior)
|
|
if errs := ValidateHorizontalPodAutoscaler(&hpa, hpaSpecValidationOpts); len(errs) == 0 {
|
|
t.Errorf("expected failure for %s", c.msg)
|
|
} else if !strings.Contains(errs[0].Error(), c.msg) {
|
|
t.Errorf("unexpected error: %v, expected: %s", errs[0], c.msg)
|
|
}
|
|
}
|
|
}
|
|
|
|
func prepareHPAWithBehavior(b autoscaling.HorizontalPodAutoscalerBehavior) autoscaling.HorizontalPodAutoscaler {
|
|
return autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
ResourceVersion: "1",
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
APIVersion: "apps/v1",
|
|
Kind: "Deployment",
|
|
Name: "mydeployment",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
Behavior: &b,
|
|
},
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscaler(t *testing.T) {
|
|
metricLabelSelector, err := metav1.ParseToLabelSelector("label=value")
|
|
if err != nil {
|
|
t.Errorf("unable to parse label selector: %v", err)
|
|
}
|
|
|
|
successCases := []autoscaling.HorizontalPodAutoscaler{
|
|
{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.PodsMetricSourceType,
|
|
Pods: &autoscaling.PodsMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-container",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-container",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
APIVersion: "networking.k8s.io/v1",
|
|
Name: "my-ingress",
|
|
Kind: "Ingress",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "requests-per-second",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "Deployment",
|
|
Name: "mydeployment",
|
|
APIVersion: "apps/v1",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, successCase := range successCases {
|
|
hpaOpts := hpaSpecValidationOpts
|
|
if successCase.Spec.ScaleTargetRef.Kind == "ReplicationController" {
|
|
hpaOpts.ScaleTargetRefValidationOptions.AllowEmptyAPIGroup = true
|
|
}
|
|
if errs := ValidateHorizontalPodAutoscaler(&successCase, hpaOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
|
|
errorCases := []struct {
|
|
horizontalPodAutoscaler autoscaling.HorizontalPodAutoscaler
|
|
msg string
|
|
}{{
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "scaleTargetRef.kind: Required",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "myrc", APIVersion: "v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "scaleTargetRef.kind: Required",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Kind: "..", Name: "myrc", APIVersion: "v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "scaleTargetRef.kind: Invalid",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Kind: "..", Name: "myrc", APIVersion: "v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "scaleTargetRef.kind: Invalid",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "scaleTargetRef.name: Required",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "scaleTargetRef.name: Required",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "..", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "scaleTargetRef.name: Invalid",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Kind: "Deployment", Name: "mydeployment"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.scaleTargetRef.apiVersion: Invalid value: \"\": apiVersion must specify API group",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{},
|
|
MinReplicas: ptr.To[int32](-1),
|
|
MaxReplicas: 5,
|
|
},
|
|
},
|
|
msg: "must be greater than or equal to 1",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{},
|
|
MinReplicas: ptr.To[int32](7),
|
|
MaxReplicas: 5,
|
|
},
|
|
},
|
|
msg: "must be greater than or equal to `minReplicas`",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "may not set both a target raw value and a target utilization",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "may not set both a target raw value and a target utilization",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a resource name",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a resource name",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: "InvalidResource",
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](70),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "Invalid value: \"InvalidResource\": must be a standard resource type or fully qualified",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](-10),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must be greater than 0",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](-10),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must be greater than 0",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](-10),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a container",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "---***",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.UtilizationMetricType,
|
|
AverageUtilization: ptr.To[int32](-10),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "Invalid value: \"---***\"",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must set either a target raw value or a target utilization",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ContainerResourceMetricSourceType,
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must set either a target raw value or a target utilization",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.PodsMetricSourceType,
|
|
Pods: &autoscaling.PodsMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a metric name",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.PodsMetricSourceType,
|
|
Pods: &autoscaling.PodsMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a positive target averageValue",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must set either a target value or averageValue",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Name: "myrc",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "object.describedObject.kind: Required",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "myrc", Kind: "ReplicationController", APIVersion: "apps/v1/v3"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Name: "myrc",
|
|
Kind: "Deployment",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "spec.scaleTargetRef.apiVersion: Invalid value: \"apps/v1/v3\": unexpected GroupVersion string: apps/v1/v3",
|
|
},
|
|
{
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a metric name",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a metric name",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "foo/../",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "'/'",
|
|
},
|
|
|
|
{
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must set either a target value for metric or a per-pod target",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(-300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must be positive",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(-300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must be positive",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "may not set both a target value for metric and a per-pod target",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: "boogity",
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must be either Utilization, Value, or AverageValue",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a metric target type",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must specify a metric target",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{
|
|
{},
|
|
},
|
|
},
|
|
},
|
|
msg: "must specify a metric source type",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.MetricSourceType("InvalidType"),
|
|
}},
|
|
},
|
|
},
|
|
msg: "type: Unsupported value",
|
|
}, {
|
|
horizontalPodAutoscaler: autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{Name: "mydeployment", Kind: "Deployment", APIVersion: "apps/v1"},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ResourceMetricSourceType,
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
Pods: &autoscaling.PodsMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
},
|
|
msg: "must populate the given metric source only",
|
|
},
|
|
}
|
|
|
|
for _, c := range errorCases {
|
|
errs := ValidateHorizontalPodAutoscaler(&c.horizontalPodAutoscaler, hpaSpecValidationOpts)
|
|
if len(errs) == 0 {
|
|
t.Errorf("expected failure for %q", c.msg)
|
|
} else if !strings.Contains(errs[0].Error(), c.msg) {
|
|
t.Errorf("unexpected error:\n expected: %q\n got: %q", c.msg, errs[0])
|
|
}
|
|
}
|
|
|
|
sourceTypes := map[autoscaling.MetricSourceType]autoscaling.MetricSpec{
|
|
autoscaling.ResourceMetricSourceType: {
|
|
Resource: &autoscaling.ResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
},
|
|
autoscaling.ContainerResourceMetricSourceType: {
|
|
ContainerResource: &autoscaling.ContainerResourceMetricSource{
|
|
Name: api.ResourceCPU,
|
|
Container: "test-application",
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
},
|
|
autoscaling.PodsMetricSourceType: {
|
|
Pods: &autoscaling.PodsMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
},
|
|
autoscaling.ObjectMetricSourceType: {
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Kind: "Deployment",
|
|
Name: "mydeployment",
|
|
APIVersion: "apps/v1",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
for correctType, spec := range sourceTypes {
|
|
for incorrectType := range sourceTypes {
|
|
if correctType == incorrectType {
|
|
continue
|
|
}
|
|
|
|
spec.Type = incorrectType
|
|
|
|
errs := ValidateHorizontalPodAutoscaler(&autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{Name: "myautoscaler", Namespace: metav1.NamespaceDefault},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "Deployment",
|
|
Name: "mydeployment",
|
|
APIVersion: "apps/v1",
|
|
},
|
|
MinReplicas: ptr.To[int32](1),
|
|
MaxReplicas: 5, Metrics: []autoscaling.MetricSpec{spec},
|
|
},
|
|
}, hpaSpecValidationOpts)
|
|
|
|
expectedMsg := "must populate information for the given metric source"
|
|
|
|
if len(errs) == 0 {
|
|
t.Errorf("expected failure with type of %v and spec for %v", incorrectType, correctType)
|
|
} else if !strings.Contains(errs[0].Error(), expectedMsg) {
|
|
t.Errorf("unexpected error: %q, expected %q", errs[0], expectedMsg)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func prepareMinReplicasCases(t *testing.T, minReplicas int32) []autoscaling.HorizontalPodAutoscaler {
|
|
metricLabelSelector, err := metav1.ParseToLabelSelector("label=value")
|
|
if err != nil {
|
|
t.Errorf("unable to parse label selector: %v", err)
|
|
}
|
|
minReplicasCases := []autoscaling.HorizontalPodAutoscaler{{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
ResourceVersion: "theversion",
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "Deployment",
|
|
Name: "mydeployment",
|
|
APIVersion: "apps/v1",
|
|
},
|
|
MinReplicas: ptr.To[int32](minReplicas),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}, {
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
ResourceVersion: "theversion",
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "Deployment",
|
|
Name: "mydeployment",
|
|
APIVersion: "apps/v1",
|
|
},
|
|
MinReplicas: ptr.To[int32](minReplicas),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ExternalMetricSourceType,
|
|
External: &autoscaling.ExternalMetricSource{
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
Selector: metricLabelSelector,
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.AverageValueMetricType,
|
|
AverageValue: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}}
|
|
return minReplicasCases
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerScaleToZeroEnabled(t *testing.T) {
|
|
zeroMinReplicasCases := prepareMinReplicasCases(t, 0)
|
|
for _, successCase := range zeroMinReplicasCases {
|
|
if errs := ValidateHorizontalPodAutoscaler(&successCase, hpaScaleToZeroSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerScaleToZeroDisabled(t *testing.T) {
|
|
zeroMinReplicasCases := prepareMinReplicasCases(t, 0)
|
|
errorMsg := "must be greater than or equal to 1"
|
|
|
|
for _, errorCase := range zeroMinReplicasCases {
|
|
errs := ValidateHorizontalPodAutoscaler(&errorCase, hpaSpecValidationOpts)
|
|
if len(errs) == 0 {
|
|
t.Errorf("expected failure for %q", errorMsg)
|
|
} else if !strings.Contains(errs[0].Error(), errorMsg) {
|
|
t.Errorf("unexpected error: %q, expected: %q", errs[0], errorMsg)
|
|
}
|
|
}
|
|
|
|
nonZeroMinReplicasCases := prepareMinReplicasCases(t, 1)
|
|
|
|
for _, successCase := range nonZeroMinReplicasCases {
|
|
successCase.Spec.MinReplicas = ptr.To[int32](1)
|
|
if errs := ValidateHorizontalPodAutoscaler(&successCase, hpaSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerUpdateScaleToZeroEnabled(t *testing.T) {
|
|
zeroMinReplicasCases := prepareMinReplicasCases(t, 0)
|
|
nonZeroMinReplicasCases := prepareMinReplicasCases(t, 1)
|
|
|
|
for i, zeroCase := range zeroMinReplicasCases {
|
|
nonZeroCase := nonZeroMinReplicasCases[i]
|
|
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(&nonZeroCase, &zeroCase, hpaScaleToZeroSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(&zeroCase, &nonZeroCase, hpaScaleToZeroSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerScaleToZeroUpdateDisabled(t *testing.T) {
|
|
zeroMinReplicasCases := prepareMinReplicasCases(t, 0)
|
|
nonZeroMinReplicasCases := prepareMinReplicasCases(t, 1)
|
|
errorMsg := "must be greater than or equal to 1"
|
|
|
|
for i, zeroCase := range zeroMinReplicasCases {
|
|
nonZeroCase := nonZeroMinReplicasCases[i]
|
|
errs := ValidateHorizontalPodAutoscalerUpdate(&zeroCase, &nonZeroCase, hpaSpecValidationOpts)
|
|
|
|
if len(errs) == 0 {
|
|
t.Errorf("expected failure for %q", errorMsg)
|
|
} else if !strings.Contains(errs[0].Error(), errorMsg) {
|
|
t.Errorf("unexpected error: %q, expected: %q", errs[0], errorMsg)
|
|
}
|
|
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(&nonZeroCase, &zeroCase, hpaSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerConfigurableToleranceEnabled(t *testing.T) {
|
|
policiesList := []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 1,
|
|
PeriodSeconds: 1800,
|
|
}}
|
|
|
|
successCases := []autoscaling.HPAScalingRules{
|
|
{
|
|
Policies: policiesList,
|
|
Tolerance: ptr.To(resource.MustParse("0.1")),
|
|
},
|
|
{
|
|
Policies: policiesList,
|
|
Tolerance: ptr.To(resource.MustParse("10")),
|
|
},
|
|
{
|
|
Policies: policiesList,
|
|
Tolerance: ptr.To(resource.MustParse("0")),
|
|
},
|
|
{
|
|
Policies: policiesList,
|
|
Tolerance: resource.NewMilliQuantity(100, resource.DecimalSI),
|
|
},
|
|
{
|
|
Policies: policiesList,
|
|
Tolerance: resource.NewScaledQuantity(1, resource.Milli),
|
|
},
|
|
{
|
|
Policies: policiesList,
|
|
},
|
|
}
|
|
for _, c := range successCases {
|
|
b := autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &c,
|
|
}
|
|
hpa := prepareHPAWithBehavior(b)
|
|
if errs := ValidateHorizontalPodAutoscaler(&hpa, hpaScaleToZeroSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
|
|
failureCases := []struct {
|
|
rule autoscaling.HPAScalingRules
|
|
msg string
|
|
}{
|
|
{
|
|
rule: autoscaling.HPAScalingRules{},
|
|
msg: "at least one Policy",
|
|
},
|
|
{
|
|
rule: autoscaling.HPAScalingRules{
|
|
Policies: policiesList,
|
|
Tolerance: ptr.To(resource.MustParse("-0.001")),
|
|
},
|
|
msg: "greater than or equal to 0",
|
|
},
|
|
{
|
|
rule: autoscaling.HPAScalingRules{
|
|
Policies: policiesList,
|
|
Tolerance: resource.NewMilliQuantity(-10, resource.DecimalSI),
|
|
},
|
|
msg: "greater than or equal to 0",
|
|
},
|
|
{
|
|
rule: autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](60),
|
|
},
|
|
msg: "at least one Policy",
|
|
},
|
|
{
|
|
rule: autoscaling.HPAScalingRules{
|
|
Tolerance: resource.NewMilliQuantity(1, resource.DecimalSI),
|
|
StabilizationWindowSeconds: ptr.To[int32](60),
|
|
},
|
|
msg: "at least one Policy",
|
|
},
|
|
}
|
|
for _, c := range failureCases {
|
|
b := autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &c.rule,
|
|
}
|
|
hpa := prepareHPAWithBehavior(b)
|
|
errs := ValidateHorizontalPodAutoscaler(&hpa, hpaScaleToZeroSpecValidationOpts)
|
|
if len(errs) != 1 {
|
|
t.Fatalf("expected exactly one error, got: %v", errs)
|
|
}
|
|
if !strings.Contains(errs[0].Error(), c.msg) {
|
|
t.Errorf("unexpected error: %q, expected: %q", errs[0], c.msg)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerConfigurableToleranceDisabled(t *testing.T) {
|
|
maxPolicy := autoscaling.MaxPolicySelect
|
|
policiesList := []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 1,
|
|
PeriodSeconds: 1800,
|
|
}}
|
|
|
|
successCases := []autoscaling.HPAScalingRules{
|
|
{
|
|
Policies: policiesList,
|
|
},
|
|
{
|
|
SelectPolicy: &maxPolicy,
|
|
Policies: policiesList,
|
|
},
|
|
{
|
|
StabilizationWindowSeconds: ptr.To[int32](60),
|
|
Policies: policiesList,
|
|
},
|
|
}
|
|
for _, c := range successCases {
|
|
b := autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleDown: &c,
|
|
}
|
|
hpa := prepareHPAWithBehavior(b)
|
|
if errs := ValidateHorizontalPodAutoscaler(&hpa, hpaSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
|
|
failureCases := []struct {
|
|
rule autoscaling.HPAScalingRules
|
|
msg string
|
|
}{
|
|
{
|
|
rule: autoscaling.HPAScalingRules{},
|
|
msg: "at least one Policy",
|
|
},
|
|
{
|
|
rule: autoscaling.HPAScalingRules{
|
|
StabilizationWindowSeconds: ptr.To[int32](60),
|
|
},
|
|
msg: "at least one Policy",
|
|
},
|
|
}
|
|
for _, c := range failureCases {
|
|
b := autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &c.rule,
|
|
}
|
|
hpa := prepareHPAWithBehavior(b)
|
|
errs := ValidateHorizontalPodAutoscaler(&hpa, hpaSpecValidationOpts)
|
|
if len(errs) != 1 {
|
|
t.Fatalf("expected exactly one error, got: %v", errs)
|
|
}
|
|
if !strings.Contains(errs[0].Error(), c.msg) {
|
|
t.Errorf("unexpected error: %q, expected: %q", errs[0], c.msg)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerUpdateConfigurableToleranceEnabled(t *testing.T) {
|
|
policiesList := []autoscaling.HPAScalingPolicy{{
|
|
Type: autoscaling.PodsScalingPolicy,
|
|
Value: 1,
|
|
PeriodSeconds: 1800,
|
|
}}
|
|
|
|
withToleranceHPA := prepareHPAWithBehavior(autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: policiesList,
|
|
Tolerance: resource.NewMilliQuantity(10, resource.DecimalSI),
|
|
}})
|
|
withoutToleranceHPA := prepareHPAWithBehavior(autoscaling.HorizontalPodAutoscalerBehavior{
|
|
ScaleUp: &autoscaling.HPAScalingRules{
|
|
Policies: policiesList,
|
|
}})
|
|
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(&withToleranceHPA, &withoutToleranceHPA, hpaScaleToZeroSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(&withoutToleranceHPA, &withToleranceHPA, hpaScaleToZeroSpecValidationOpts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerUpdateInvalidHPA(t *testing.T) {
|
|
existingHPA := autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
ResourceVersion: "theversion",
|
|
},
|
|
Spec: autoscaling.HorizontalPodAutoscalerSpec{
|
|
ScaleTargetRef: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
APIVersion: "invalidAPIVersion/v1/v2",
|
|
},
|
|
MinReplicas: ptr.To[int32](3),
|
|
MaxReplicas: 5,
|
|
Metrics: []autoscaling.MetricSpec{{
|
|
Type: autoscaling.ObjectMetricSourceType,
|
|
Object: &autoscaling.ObjectMetricSource{
|
|
DescribedObject: autoscaling.CrossVersionObjectReference{
|
|
Kind: "ReplicationController",
|
|
Name: "myrc",
|
|
APIVersion: "invalidAPIVersion/v1/v2",
|
|
},
|
|
Metric: autoscaling.MetricIdentifier{
|
|
Name: "somemetric",
|
|
},
|
|
Target: autoscaling.MetricTarget{
|
|
Type: autoscaling.ValueMetricType,
|
|
Value: resource.NewMilliQuantity(300, resource.DecimalSI),
|
|
},
|
|
},
|
|
}},
|
|
},
|
|
}
|
|
|
|
opts := HorizontalPodAutoscalerSpecValidationOptions{
|
|
MinReplicasLowerBound: 0,
|
|
ScaleTargetRefValidationOptions: CrossVersionObjectReferenceValidationOptions{
|
|
AllowInvalidAPIVersion: true,
|
|
},
|
|
ObjectMetricsValidationOptions: CrossVersionObjectReferenceValidationOptions{
|
|
AllowInvalidAPIVersion: true,
|
|
},
|
|
}
|
|
modifiedHPAReplicasChanged := existingHPA.DeepCopy()
|
|
modifiedHPAReplicasChanged.Spec.MinReplicas = ptr.To[int32](1)
|
|
// API Version is not modifed, so it should not be checked.
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(modifiedHPAReplicasChanged, &existingHPA, opts); len(errs) != 0 {
|
|
t.Errorf("expected success: %v", errs)
|
|
}
|
|
|
|
modifiedHPAMetricsChanged := existingHPA.DeepCopy()
|
|
modifiedHPAMetricsChanged.Spec.Metrics[0].Object.DescribedObject.APIVersion = "invalidAPIVersion/v1/v3"
|
|
opts.ObjectMetricsValidationOptions.AllowInvalidAPIVersion = false
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(modifiedHPAMetricsChanged, &existingHPA, opts); len(errs) == 0 {
|
|
t.Error("expected error, APIVersion should be checked")
|
|
}
|
|
|
|
modifiedHPAScaleRefChanged := existingHPA.DeepCopy()
|
|
modifiedHPAScaleRefChanged.Spec.ScaleTargetRef.APIVersion = "invalidAPIVersion/v1/v4"
|
|
opts.ScaleTargetRefValidationOptions.AllowInvalidAPIVersion = false
|
|
if errs := ValidateHorizontalPodAutoscalerUpdate(modifiedHPAScaleRefChanged, &existingHPA, opts); len(errs) == 0 {
|
|
t.Error("expected error, APIVersion should be checked")
|
|
}
|
|
}
|
|
|
|
func TestValidateHorizontalPodAutoscalerStatusUpdate(t *testing.T) {
|
|
baseHPA := autoscaling.HorizontalPodAutoscaler{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "myautoscaler",
|
|
Namespace: metav1.NamespaceDefault,
|
|
ResourceVersion: "1",
|
|
},
|
|
}
|
|
|
|
testCases := []struct {
|
|
name string
|
|
status autoscaling.HorizontalPodAutoscalerStatus
|
|
expectError bool
|
|
errContains string
|
|
}{
|
|
{
|
|
name: "valid: no conditions",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 2,
|
|
DesiredReplicas: 3,
|
|
},
|
|
}, {
|
|
name: "valid: condition with nil observedGeneration",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 1,
|
|
DesiredReplicas: 1,
|
|
Conditions: []autoscaling.HorizontalPodAutoscalerCondition{
|
|
{Type: autoscaling.ScalingLimited, Status: autoscaling.ConditionFalse, ObservedGeneration: nil},
|
|
},
|
|
},
|
|
}, {
|
|
name: "valid: condition with zero observedGeneration",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 1,
|
|
DesiredReplicas: 1,
|
|
Conditions: []autoscaling.HorizontalPodAutoscalerCondition{
|
|
{Type: autoscaling.ScalingLimited, Status: autoscaling.ConditionFalse, ObservedGeneration: ptr.To[int64](0)},
|
|
},
|
|
},
|
|
}, {
|
|
name: "valid: condition with positive observedGeneration",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 2,
|
|
DesiredReplicas: 2,
|
|
Conditions: []autoscaling.HorizontalPodAutoscalerCondition{
|
|
{Type: autoscaling.ScalingLimited, Status: autoscaling.ConditionFalse, ObservedGeneration: ptr.To[int64](5)},
|
|
},
|
|
},
|
|
}, {
|
|
name: "valid: multiple conditions all with non-negative observedGeneration",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 1,
|
|
DesiredReplicas: 1,
|
|
Conditions: []autoscaling.HorizontalPodAutoscalerCondition{
|
|
{Type: autoscaling.ScalingActive, Status: autoscaling.ConditionTrue, ObservedGeneration: ptr.To[int64](1)},
|
|
{Type: autoscaling.AbleToScale, Status: autoscaling.ConditionTrue, ObservedGeneration: ptr.To[int64](2)},
|
|
},
|
|
},
|
|
}, {
|
|
name: "invalid: condition with negative observedGeneration",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 1,
|
|
DesiredReplicas: 1,
|
|
Conditions: []autoscaling.HorizontalPodAutoscalerCondition{
|
|
{
|
|
Type: autoscaling.ScalingActive,
|
|
Status: autoscaling.ConditionTrue,
|
|
ObservedGeneration: ptr.To[int64](-1),
|
|
},
|
|
},
|
|
},
|
|
expectError: true,
|
|
errContains: "status.conditions[0].observedGeneration",
|
|
}, {
|
|
name: "invalid: second condition has negative observedGeneration",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 1,
|
|
DesiredReplicas: 1,
|
|
Conditions: []autoscaling.HorizontalPodAutoscalerCondition{
|
|
{Type: autoscaling.ScalingActive, Status: autoscaling.ConditionTrue, ObservedGeneration: ptr.To[int64](1)},
|
|
{Type: autoscaling.AbleToScale, Status: autoscaling.ConditionFalse, ObservedGeneration: ptr.To[int64](-3)},
|
|
},
|
|
},
|
|
expectError: true,
|
|
errContains: "status.conditions[1].observedGeneration",
|
|
}, {
|
|
name: "invalid: negative currentReplicas",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: -1,
|
|
DesiredReplicas: 1,
|
|
},
|
|
expectError: true,
|
|
errContains: "status.currentReplicas",
|
|
}, {
|
|
name: "invalid: negative desiredReplicas",
|
|
status: autoscaling.HorizontalPodAutoscalerStatus{
|
|
CurrentReplicas: 1,
|
|
DesiredReplicas: -1,
|
|
},
|
|
expectError: true,
|
|
errContains: "status.desiredReplicas",
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
newHPA := baseHPA.DeepCopy()
|
|
newHPA.Status = tc.status
|
|
|
|
errs := ValidateHorizontalPodAutoscalerStatusUpdate(newHPA, &baseHPA)
|
|
if tc.expectError {
|
|
if len(errs) == 0 {
|
|
t.Errorf("expected error containing %q, got none", tc.errContains)
|
|
return
|
|
}
|
|
found := false
|
|
for _, err := range errs {
|
|
if strings.Contains(err.Field, tc.errContains) || strings.Contains(err.Error(), tc.errContains) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
t.Errorf("expected error containing %q, got: %v", tc.errContains, errs)
|
|
}
|
|
} else if len(errs) != 0 {
|
|
t.Errorf("expected no errors, got: %v", errs)
|
|
}
|
|
})
|
|
}
|
|
}
|