Merge pull request #112925 from SergeyKanzhelev/addValueOfLimitToErr
added limit value to the pod validation error to simplify debugging
This commit is contained in:
@@ -20,7 +20,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
"k8s.io/apimachinery/pkg/api/resource"
|
"k8s.io/apimachinery/pkg/api/resource"
|
||||||
"k8s.io/apimachinery/pkg/util/sets"
|
"k8s.io/apimachinery/pkg/util/sets"
|
||||||
"k8s.io/apimachinery/pkg/util/validation/field"
|
"k8s.io/apimachinery/pkg/util/validation/field"
|
||||||
@@ -60,9 +60,9 @@ func ValidateResourceRequirements(requirements *v1.ResourceRequirements, fldPath
|
|||||||
if exists {
|
if exists {
|
||||||
// For GPUs, not only requests can't exceed limits, they also can't be lower, i.e. must be equal.
|
// For GPUs, not only requests can't exceed limits, they also can't be lower, i.e. must be equal.
|
||||||
if quantity.Cmp(limitQuantity) != 0 && !v1helper.IsOvercommitAllowed(resourceName) {
|
if quantity.Cmp(limitQuantity) != 0 && !v1helper.IsOvercommitAllowed(resourceName) {
|
||||||
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be equal to %s limit", resourceName)))
|
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be equal to %s limit of %s", resourceName, limitQuantity.String())))
|
||||||
} else if quantity.Cmp(limitQuantity) > 0 {
|
} else if quantity.Cmp(limitQuantity) > 0 {
|
||||||
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be less than or equal to %s limit", resourceName)))
|
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be less than or equal to %s limit of %s", resourceName, limitQuantity.String())))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -17,9 +17,10 @@ limitations under the License.
|
|||||||
package validation
|
package validation
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
"k8s.io/apimachinery/pkg/api/resource"
|
"k8s.io/apimachinery/pkg/api/resource"
|
||||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
"k8s.io/apimachinery/pkg/util/sets"
|
"k8s.io/apimachinery/pkg/util/sets"
|
||||||
@@ -91,6 +92,8 @@ func TestValidateResourceRequirements(t *testing.T) {
|
|||||||
errorCase := []struct {
|
errorCase := []struct {
|
||||||
name string
|
name string
|
||||||
requirements v1.ResourceRequirements
|
requirements v1.ResourceRequirements
|
||||||
|
skipLimitValueCheck bool
|
||||||
|
skipRequestValueCheck bool
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "Resources with Requests Larger Than Limits",
|
name: "Resources with Requests Larger Than Limits",
|
||||||
@@ -114,6 +117,7 @@ func TestValidateResourceRequirements(t *testing.T) {
|
|||||||
v1.ResourceName("my.org"): resource.MustParse("10m"),
|
v1.ResourceName("my.org"): resource.MustParse("10m"),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
skipRequestValueCheck: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Invalid Resources with Limits",
|
name: "Invalid Resources with Limits",
|
||||||
@@ -122,17 +126,44 @@ func TestValidateResourceRequirements(t *testing.T) {
|
|||||||
v1.ResourceName("my.org"): resource.MustParse("9m"),
|
v1.ResourceName("my.org"): resource.MustParse("9m"),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
skipLimitValueCheck: true,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
for _, tc := range errorCase {
|
for _, tc := range errorCase {
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
if errs := ValidateResourceRequirements(&tc.requirements, field.NewPath("resources")); len(errs) == 0 {
|
errs := ValidateResourceRequirements(&tc.requirements, field.NewPath("resources"))
|
||||||
|
if len(errs) == 0 {
|
||||||
t.Errorf("expected error")
|
t.Errorf("expected error")
|
||||||
}
|
}
|
||||||
|
validateNamesAndValuesInDescription(t, tc.requirements.Limits, errs, tc.skipLimitValueCheck, "limit")
|
||||||
|
validateNamesAndValuesInDescription(t, tc.requirements.Requests, errs, tc.skipRequestValueCheck, "request")
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func validateNamesAndValuesInDescription(t *testing.T, r v1.ResourceList, errs field.ErrorList, skipValueTest bool, rl string) {
|
||||||
|
for name, value := range r {
|
||||||
|
containsName := false
|
||||||
|
containsValue := false
|
||||||
|
|
||||||
|
for _, e := range errs {
|
||||||
|
if strings.Contains(e.Error(), name.String()) {
|
||||||
|
containsName = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.Contains(e.Error(), value.String()) {
|
||||||
|
containsValue = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !containsName {
|
||||||
|
t.Errorf("error must contain %s name", rl)
|
||||||
|
}
|
||||||
|
if !containsValue && !skipValueTest {
|
||||||
|
t.Errorf("error must contain %s value", rl)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestValidateContainerResourceName(t *testing.T) {
|
func TestValidateContainerResourceName(t *testing.T) {
|
||||||
successCase := []struct {
|
successCase := []struct {
|
||||||
name string
|
name string
|
||||||
|
@@ -5936,9 +5936,9 @@ func ValidateResourceRequirements(requirements *core.ResourceRequirements, podCl
|
|||||||
if exists {
|
if exists {
|
||||||
// For non overcommitable resources, not only requests can't exceed limits, they also can't be lower, i.e. must be equal.
|
// For non overcommitable resources, not only requests can't exceed limits, they also can't be lower, i.e. must be equal.
|
||||||
if quantity.Cmp(limitQuantity) != 0 && !helper.IsOvercommitAllowed(resourceName) {
|
if quantity.Cmp(limitQuantity) != 0 && !helper.IsOvercommitAllowed(resourceName) {
|
||||||
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be equal to %s limit", resourceName)))
|
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be equal to %s limit of %s", resourceName, limitQuantity.String())))
|
||||||
} else if quantity.Cmp(limitQuantity) > 0 {
|
} else if quantity.Cmp(limitQuantity) > 0 {
|
||||||
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be less than or equal to %s limit", resourceName)))
|
allErrs = append(allErrs, field.Invalid(reqPath, quantity.String(), fmt.Sprintf("must be less than or equal to %s limit of %s", resourceName, limitQuantity.String())))
|
||||||
}
|
}
|
||||||
} else if !helper.IsOvercommitAllowed(resourceName) {
|
} else if !helper.IsOvercommitAllowed(resourceName) {
|
||||||
allErrs = append(allErrs, field.Required(limPath, "Limit must be set for non overcommitable resources"))
|
allErrs = append(allErrs, field.Required(limPath, "Limit must be set for non overcommitable resources"))
|
||||||
|
Reference in New Issue
Block a user