CSIStorageCapacity: CSIStorageCapacity API
This adds the CSIStorageCapacity API change for https://github.com/kubernetes/enhancements/tree/master/keps/sig-storage/1472-storage-capacity-tracking
This commit is contained in:
@@ -22,6 +22,7 @@ import (
|
||||
"strings"
|
||||
|
||||
apiequality "k8s.io/apimachinery/pkg/api/equality"
|
||||
apimachineryvalidation "k8s.io/apimachinery/pkg/api/validation"
|
||||
"k8s.io/apimachinery/pkg/util/sets"
|
||||
"k8s.io/apimachinery/pkg/util/validation"
|
||||
"k8s.io/apimachinery/pkg/util/validation/field"
|
||||
@@ -459,3 +460,36 @@ func validateVolumeLifecycleModes(modes []storage.VolumeLifecycleMode, fldPath *
|
||||
|
||||
return allErrs
|
||||
}
|
||||
|
||||
// ValidateStorageCapacityName checks that a name is appropriate for a
|
||||
// CSIStorageCapacity object.
|
||||
var ValidateStorageCapacityName = apimachineryvalidation.NameIsDNSSubdomain
|
||||
|
||||
// ValidateCSIStorageCapacity validates a CSIStorageCapacity.
|
||||
func ValidateCSIStorageCapacity(capacity *storage.CSIStorageCapacity) field.ErrorList {
|
||||
allErrs := apivalidation.ValidateObjectMeta(&capacity.ObjectMeta, true, ValidateStorageCapacityName, field.NewPath("metadata"))
|
||||
allErrs = append(allErrs, metav1validation.ValidateLabelSelector(capacity.NodeTopology, field.NewPath("nodeTopology"))...)
|
||||
for _, msg := range apivalidation.ValidateClassName(capacity.StorageClassName, false) {
|
||||
allErrs = append(allErrs, field.Invalid(field.NewPath("storageClassName"), capacity.StorageClassName, msg))
|
||||
}
|
||||
if capacity.Capacity != nil {
|
||||
allErrs = append(allErrs, apivalidation.ValidateNonnegativeQuantity(*capacity.Capacity, field.NewPath("capacity"))...)
|
||||
}
|
||||
return allErrs
|
||||
}
|
||||
|
||||
// ValidateCSIStorageCapacityUpdate tests if an update to CSIStorageCapacity is valid.
|
||||
func ValidateCSIStorageCapacityUpdate(capacity, oldCapacity *storage.CSIStorageCapacity) field.ErrorList {
|
||||
allErrs := apivalidation.ValidateObjectMetaUpdate(&capacity.ObjectMeta, &oldCapacity.ObjectMeta, field.NewPath("metadata"))
|
||||
|
||||
// Input fields for CSI GetCapacity are immutable.
|
||||
// If this ever relaxes in the future, make sure to increment the Generation number in PrepareForUpdate
|
||||
if !apiequality.Semantic.DeepEqual(capacity.NodeTopology, oldCapacity.NodeTopology) {
|
||||
allErrs = append(allErrs, field.Invalid(field.NewPath("nodeTopology"), capacity.NodeTopology, "field is immutable"))
|
||||
}
|
||||
if capacity.StorageClassName != oldCapacity.StorageClassName {
|
||||
allErrs = append(allErrs, field.Invalid(field.NewPath("storageClassName"), capacity.StorageClassName, "field is immutable"))
|
||||
}
|
||||
|
||||
return allErrs
|
||||
}
|
||||
|
@@ -1939,3 +1939,104 @@ func TestCSIDriverValidationUpdate(t *testing.T) {
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestValidateCSIStorageCapacity(t *testing.T) {
|
||||
storageClassName := "test-sc"
|
||||
invalidName := "-invalid-@#$%^&*()-"
|
||||
|
||||
goodCapacity := storage.CSIStorageCapacity{
|
||||
ObjectMeta: metav1.ObjectMeta{
|
||||
Name: "csc-329803da-fdd2-42e4-af6f-7b07e7ccc305",
|
||||
Namespace: metav1.NamespaceDefault,
|
||||
},
|
||||
StorageClassName: storageClassName,
|
||||
}
|
||||
goodTopology := metav1.LabelSelector{
|
||||
MatchLabels: map[string]string{"foo": "bar"},
|
||||
}
|
||||
|
||||
scenarios := map[string]struct {
|
||||
isExpectedFailure bool
|
||||
capacity *storage.CSIStorageCapacity
|
||||
}{
|
||||
"good-capacity": {
|
||||
capacity: &goodCapacity,
|
||||
},
|
||||
"missing-storage-class-name": {
|
||||
isExpectedFailure: true,
|
||||
capacity: func() *storage.CSIStorageCapacity {
|
||||
capacity := goodCapacity
|
||||
capacity.StorageClassName = ""
|
||||
return &capacity
|
||||
}(),
|
||||
},
|
||||
"bad-storage-class-name": {
|
||||
isExpectedFailure: true,
|
||||
capacity: func() *storage.CSIStorageCapacity {
|
||||
capacity := goodCapacity
|
||||
capacity.StorageClassName = invalidName
|
||||
return &capacity
|
||||
}(),
|
||||
},
|
||||
"good-capacity-value": {
|
||||
capacity: func() *storage.CSIStorageCapacity {
|
||||
capacity := goodCapacity
|
||||
capacity.Capacity = resource.NewQuantity(1, resource.BinarySI)
|
||||
return &capacity
|
||||
}(),
|
||||
},
|
||||
"bad-capacity-value": {
|
||||
isExpectedFailure: true,
|
||||
capacity: func() *storage.CSIStorageCapacity {
|
||||
capacity := goodCapacity
|
||||
capacity.Capacity = resource.NewQuantity(-11, resource.BinarySI)
|
||||
return &capacity
|
||||
}(),
|
||||
},
|
||||
"good-topology": {
|
||||
capacity: func() *storage.CSIStorageCapacity {
|
||||
capacity := goodCapacity
|
||||
capacity.NodeTopology = &goodTopology
|
||||
return &capacity
|
||||
}(),
|
||||
},
|
||||
"empty-topology": {
|
||||
capacity: func() *storage.CSIStorageCapacity {
|
||||
capacity := goodCapacity
|
||||
capacity.NodeTopology = &metav1.LabelSelector{}
|
||||
return &capacity
|
||||
}(),
|
||||
},
|
||||
"bad-topology-fields": {
|
||||
isExpectedFailure: true,
|
||||
capacity: func() *storage.CSIStorageCapacity {
|
||||
capacity := goodCapacity
|
||||
capacity.NodeTopology = &metav1.LabelSelector{
|
||||
MatchExpressions: []metav1.LabelSelectorRequirement{
|
||||
{
|
||||
Key: "foo",
|
||||
Operator: metav1.LabelSelectorOperator("no-such-operator"),
|
||||
Values: []string{
|
||||
"bar",
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
return &capacity
|
||||
}(),
|
||||
},
|
||||
}
|
||||
|
||||
for name, scenario := range scenarios {
|
||||
t.Run(name, func(t *testing.T) {
|
||||
errs := ValidateCSIStorageCapacity(scenario.capacity)
|
||||
if len(errs) == 0 && scenario.isExpectedFailure {
|
||||
t.Errorf("Unexpected success")
|
||||
}
|
||||
if len(errs) > 0 && !scenario.isExpectedFailure {
|
||||
t.Errorf("Unexpected failure: %+v", errs)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
}
|
||||
|
Reference in New Issue
Block a user