kubernetes/pkg/registry/core/persistentvolumeclaim/strategy_test.go
Ben Swartzlander 00dba76918 Add DataSourceRef field to PVC spec
Modify the behavior of the AnyVolumeDataSource alpha feature gate to enable
a new field, DataSourceRef, rather than modifying the behavior of the
existing DataSource field. This allows addition Volume Populators in a way
that doesn't risk breaking backwards compatibility, although it will
result in eventually deprecating the DataSource field.
2021-07-06 21:17:41 -04:00

257 lines
7.8 KiB
Go

/*
Copyright 2015 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 persistentvolumeclaim
import (
"fmt"
"reflect"
"testing"
"k8s.io/apimachinery/pkg/util/diff"
genericapirequest "k8s.io/apiserver/pkg/endpoints/request"
utilfeature "k8s.io/apiserver/pkg/util/feature"
featuregatetesting "k8s.io/component-base/featuregate/testing"
apitesting "k8s.io/kubernetes/pkg/api/testing"
api "k8s.io/kubernetes/pkg/apis/core"
"k8s.io/kubernetes/pkg/features"
// ensure types are installed
_ "k8s.io/kubernetes/pkg/apis/core/install"
)
func TestSelectableFieldLabelConversions(t *testing.T) {
apitesting.TestSelectableFieldLabelConversionsOfKind(t,
"v1",
"PersistentVolumeClaim",
PersistentVolumeClaimToSelectableFields(&api.PersistentVolumeClaim{}),
map[string]string{"name": "metadata.name"},
)
}
func TestDropConditions(t *testing.T) {
ctx := genericapirequest.NewDefaultContext()
pvcWithConditions := func() *api.PersistentVolumeClaim {
return &api.PersistentVolumeClaim{
Status: api.PersistentVolumeClaimStatus{
Conditions: []api.PersistentVolumeClaimCondition{
{Type: api.PersistentVolumeClaimResizing, Status: api.ConditionTrue},
},
},
}
}
pvcWithoutConditions := func() *api.PersistentVolumeClaim {
return &api.PersistentVolumeClaim{
Status: api.PersistentVolumeClaimStatus{},
}
}
pvcInfo := []struct {
description string
hasConditions bool
pvc func() *api.PersistentVolumeClaim
}{
{
description: "has Conditions",
hasConditions: true,
pvc: pvcWithConditions,
},
{
description: "does not have Conditions",
hasConditions: false,
pvc: pvcWithoutConditions,
},
}
for _, enabled := range []bool{true, false} {
for _, oldPvcInfo := range pvcInfo {
for _, newPvcInfo := range pvcInfo {
oldPvcHasConditins, oldPvc := oldPvcInfo.hasConditions, oldPvcInfo.pvc()
newPvcHasConditions, newPvc := newPvcInfo.hasConditions, newPvcInfo.pvc()
t.Run(fmt.Sprintf("feature enabled=%v, old pvc %v, new pvc %v", enabled, oldPvcInfo.description, newPvcInfo.description), func(t *testing.T) {
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.ExpandPersistentVolumes, enabled)()
StatusStrategy.PrepareForUpdate(ctx, newPvc, oldPvc)
// old pvc should never be changed
if !reflect.DeepEqual(oldPvc, oldPvcInfo.pvc()) {
t.Errorf("old pvc changed: %v", diff.ObjectReflectDiff(oldPvc, oldPvcInfo.pvc()))
}
switch {
case enabled || oldPvcHasConditins:
// new pvc should not be changed if the feature is enabled, or if the old pvc had Conditions
if !reflect.DeepEqual(newPvc, newPvcInfo.pvc()) {
t.Errorf("new pvc changed: %v", diff.ObjectReflectDiff(newPvc, newPvcInfo.pvc()))
}
case newPvcHasConditions:
// new pvc should be changed
if reflect.DeepEqual(newPvc, newPvcInfo.pvc()) {
t.Errorf("new pvc was not changed")
}
// new pvc should not have Conditions
if !reflect.DeepEqual(newPvc, pvcWithoutConditions()) {
t.Errorf("new pvc had Conditions: %v", diff.ObjectReflectDiff(newPvc, pvcWithoutConditions()))
}
default:
// new pvc should not need to be changed
if !reflect.DeepEqual(newPvc, newPvcInfo.pvc()) {
t.Errorf("new pvc changed: %v", diff.ObjectReflectDiff(newPvc, newPvcInfo.pvc()))
}
}
})
}
}
}
}
func TestPrepareForCreate(t *testing.T) {
ctx := genericapirequest.NewDefaultContext()
makeDataSource := func(apiGroup, kind, name string) *api.TypedLocalObjectReference {
return &api.TypedLocalObjectReference{
APIGroup: &apiGroup,
Kind: kind,
Name: name,
}
}
volumeDataSource := makeDataSource("", "PersistentVolumeClaim", "my-vol")
snapshotDataSource := makeDataSource("snapshot.storage.k8s.io", "VolumeSnapshot", "my-snap")
genericDataSource := makeDataSource("generic.storage.k8s.io", "Generic", "my-foo")
coreDataSource := makeDataSource("", "Pod", "my-pod")
var tests = map[string]struct {
anyEnabled bool
dataSource *api.TypedLocalObjectReference
dataSourceRef *api.TypedLocalObjectReference
want *api.TypedLocalObjectReference
wantRef *api.TypedLocalObjectReference
}{
"any disabled with empty ds": {
want: nil,
},
"any disabled with volume ds": {
dataSource: volumeDataSource,
want: volumeDataSource,
},
"any disabled with snapshot ds": {
dataSource: snapshotDataSource,
want: snapshotDataSource,
},
"any disabled with generic ds": {
dataSource: genericDataSource,
want: nil,
},
"any disabled with invalid ds": {
dataSource: coreDataSource,
want: nil,
},
"any disabled with volume ds ref": {
dataSourceRef: volumeDataSource,
},
"any disabled with snapshot ds ref": {
dataSourceRef: snapshotDataSource,
},
"any disabled with generic ds ref": {
dataSourceRef: genericDataSource,
},
"any disabled with invalid ds ref": {
dataSourceRef: coreDataSource,
},
"any enabled with empty ds": {
anyEnabled: true,
want: nil,
},
"any enabled with volume ds": {
dataSource: volumeDataSource,
anyEnabled: true,
want: volumeDataSource,
wantRef: volumeDataSource,
},
"any enabled with snapshot ds": {
dataSource: snapshotDataSource,
anyEnabled: true,
want: snapshotDataSource,
wantRef: snapshotDataSource,
},
"any enabled with generic ds": {
dataSource: genericDataSource,
anyEnabled: true,
},
"any enabled with invalid ds": {
dataSource: coreDataSource,
anyEnabled: true,
},
"any enabled with volume ds ref": {
dataSourceRef: volumeDataSource,
anyEnabled: true,
want: volumeDataSource,
wantRef: volumeDataSource,
},
"any enabled with snapshot ds ref": {
dataSourceRef: snapshotDataSource,
anyEnabled: true,
want: snapshotDataSource,
wantRef: snapshotDataSource,
},
"any enabled with generic ds ref": {
dataSourceRef: genericDataSource,
anyEnabled: true,
want: genericDataSource,
wantRef: genericDataSource,
},
"any enabled with invalid ds ref": {
dataSourceRef: coreDataSource,
anyEnabled: true,
want: coreDataSource,
wantRef: coreDataSource,
},
"any enabled with mismatched data sources": {
dataSource: volumeDataSource,
dataSourceRef: snapshotDataSource,
anyEnabled: true,
want: volumeDataSource,
wantRef: snapshotDataSource,
},
}
for testName, test := range tests {
t.Run(testName, func(t *testing.T) {
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.AnyVolumeDataSource, test.anyEnabled)()
pvc := api.PersistentVolumeClaim{
Spec: api.PersistentVolumeClaimSpec{
DataSource: test.dataSource,
DataSourceRef: test.dataSourceRef,
},
}
// Method under test
Strategy.PrepareForCreate(ctx, &pvc)
if !reflect.DeepEqual(pvc.Spec.DataSource, test.want) {
t.Errorf("data source does not match, test: %s, anyEnabled: %v, dataSource: %v, expected: %v",
testName, test.anyEnabled, test.dataSource, test.want)
}
if !reflect.DeepEqual(pvc.Spec.DataSourceRef, test.wantRef) {
t.Errorf("data source ref does not match, test: %s, anyEnabled: %v, dataSourceRef: %v, expected: %v",
testName, test.anyEnabled, test.dataSourceRef, test.wantRef)
}
})
}
}