Use CSI translation lib in VolumeBinderPredicate
This commit is contained in:
parent
4e77a27908
commit
7cba40fb09
@ -14,18 +14,25 @@ go_library(
|
|||||||
"//pkg/apis/core/v1/helper:go_default_library",
|
"//pkg/apis/core/v1/helper:go_default_library",
|
||||||
"//pkg/controller/volume/persistentvolume/util:go_default_library",
|
"//pkg/controller/volume/persistentvolume/util:go_default_library",
|
||||||
"//pkg/controller/volume/scheduling/metrics:go_default_library",
|
"//pkg/controller/volume/scheduling/metrics:go_default_library",
|
||||||
|
"//pkg/features:go_default_library",
|
||||||
"//pkg/volume/util:go_default_library",
|
"//pkg/volume/util:go_default_library",
|
||||||
"//staging/src/k8s.io/api/core/v1:go_default_library",
|
"//staging/src/k8s.io/api/core/v1:go_default_library",
|
||||||
|
"//staging/src/k8s.io/api/storage/v1beta1:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/api/meta:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/api/meta:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/labels:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/labels:go_default_library",
|
||||||
|
"//staging/src/k8s.io/apimachinery/pkg/util/sets:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/util/wait:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/util/wait:go_default_library",
|
||||||
"//staging/src/k8s.io/apiserver/pkg/storage/etcd3:go_default_library",
|
"//staging/src/k8s.io/apiserver/pkg/storage/etcd3:go_default_library",
|
||||||
|
"//staging/src/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/informers/core/v1:go_default_library",
|
"//staging/src/k8s.io/client-go/informers/core/v1:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/informers/storage/v1:go_default_library",
|
"//staging/src/k8s.io/client-go/informers/storage/v1:go_default_library",
|
||||||
|
"//staging/src/k8s.io/client-go/informers/storage/v1beta1:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/kubernetes:go_default_library",
|
"//staging/src/k8s.io/client-go/kubernetes:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/listers/storage/v1:go_default_library",
|
"//staging/src/k8s.io/client-go/listers/storage/v1:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/tools/cache:go_default_library",
|
"//staging/src/k8s.io/client-go/tools/cache:go_default_library",
|
||||||
|
"//staging/src/k8s.io/csi-translation-lib:go_default_library",
|
||||||
|
"//staging/src/k8s.io/csi-translation-lib/plugins:go_default_library",
|
||||||
"//vendor/k8s.io/klog:go_default_library",
|
"//vendor/k8s.io/klog:go_default_library",
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
@ -43,19 +50,24 @@ go_test(
|
|||||||
"//pkg/controller:go_default_library",
|
"//pkg/controller:go_default_library",
|
||||||
"//pkg/controller/volume/persistentvolume/testing:go_default_library",
|
"//pkg/controller/volume/persistentvolume/testing:go_default_library",
|
||||||
"//pkg/controller/volume/persistentvolume/util:go_default_library",
|
"//pkg/controller/volume/persistentvolume/util:go_default_library",
|
||||||
|
"//pkg/features:go_default_library",
|
||||||
"//staging/src/k8s.io/api/core/v1:go_default_library",
|
"//staging/src/k8s.io/api/core/v1:go_default_library",
|
||||||
"//staging/src/k8s.io/api/storage/v1:go_default_library",
|
"//staging/src/k8s.io/api/storage/v1:go_default_library",
|
||||||
|
"//staging/src/k8s.io/api/storage/v1beta1:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/api/resource:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/api/resource:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/types:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/types:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/util/diff:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/util/diff:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/util/wait:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/util/wait:go_default_library",
|
||||||
"//staging/src/k8s.io/apimachinery/pkg/watch:go_default_library",
|
"//staging/src/k8s.io/apimachinery/pkg/watch:go_default_library",
|
||||||
|
"//staging/src/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/informers:go_default_library",
|
"//staging/src/k8s.io/client-go/informers:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/informers/core/v1:go_default_library",
|
"//staging/src/k8s.io/client-go/informers/core/v1:go_default_library",
|
||||||
|
"//staging/src/k8s.io/client-go/informers/storage/v1beta1:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/kubernetes:go_default_library",
|
"//staging/src/k8s.io/client-go/kubernetes:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/kubernetes/fake:go_default_library",
|
"//staging/src/k8s.io/client-go/kubernetes/fake:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/testing:go_default_library",
|
"//staging/src/k8s.io/client-go/testing:go_default_library",
|
||||||
|
"//staging/src/k8s.io/component-base/featuregate/testing:go_default_library",
|
||||||
"//vendor/k8s.io/klog:go_default_library",
|
"//vendor/k8s.io/klog:go_default_library",
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
@ -19,24 +19,40 @@ package scheduling
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"sort"
|
"sort"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
v1 "k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
|
storagev1beta1 "k8s.io/api/storage/v1beta1"
|
||||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
"k8s.io/apimachinery/pkg/labels"
|
"k8s.io/apimachinery/pkg/labels"
|
||||||
|
"k8s.io/apimachinery/pkg/util/sets"
|
||||||
"k8s.io/apimachinery/pkg/util/wait"
|
"k8s.io/apimachinery/pkg/util/wait"
|
||||||
"k8s.io/apiserver/pkg/storage/etcd3"
|
"k8s.io/apiserver/pkg/storage/etcd3"
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
coreinformers "k8s.io/client-go/informers/core/v1"
|
coreinformers "k8s.io/client-go/informers/core/v1"
|
||||||
storageinformers "k8s.io/client-go/informers/storage/v1"
|
storageinformers "k8s.io/client-go/informers/storage/v1"
|
||||||
|
storagev1beta1informers "k8s.io/client-go/informers/storage/v1beta1"
|
||||||
clientset "k8s.io/client-go/kubernetes"
|
clientset "k8s.io/client-go/kubernetes"
|
||||||
storagelisters "k8s.io/client-go/listers/storage/v1"
|
storagelisters "k8s.io/client-go/listers/storage/v1"
|
||||||
|
csitrans "k8s.io/csi-translation-lib"
|
||||||
|
csiplugins "k8s.io/csi-translation-lib/plugins"
|
||||||
"k8s.io/klog"
|
"k8s.io/klog"
|
||||||
v1helper "k8s.io/kubernetes/pkg/apis/core/v1/helper"
|
v1helper "k8s.io/kubernetes/pkg/apis/core/v1/helper"
|
||||||
pvutil "k8s.io/kubernetes/pkg/controller/volume/persistentvolume/util"
|
pvutil "k8s.io/kubernetes/pkg/controller/volume/persistentvolume/util"
|
||||||
"k8s.io/kubernetes/pkg/controller/volume/scheduling/metrics"
|
"k8s.io/kubernetes/pkg/controller/volume/scheduling/metrics"
|
||||||
|
"k8s.io/kubernetes/pkg/features"
|
||||||
volumeutil "k8s.io/kubernetes/pkg/volume/util"
|
volumeutil "k8s.io/kubernetes/pkg/volume/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// InTreeToCSITranslator contains methods required to check migratable status
|
||||||
|
// and perform translations from InTree PV's to CSI
|
||||||
|
type InTreeToCSITranslator interface {
|
||||||
|
IsPVMigratable(pv *v1.PersistentVolume) bool
|
||||||
|
GetInTreePluginNameFromSpec(pv *v1.PersistentVolume, vol *v1.Volume) (string, error)
|
||||||
|
TranslateInTreePVToCSI(pv *v1.PersistentVolume) (*v1.PersistentVolume, error)
|
||||||
|
}
|
||||||
|
|
||||||
// SchedulerVolumeBinder is used by the scheduler to handle PVC/PV binding
|
// SchedulerVolumeBinder is used by the scheduler to handle PVC/PV binding
|
||||||
// and dynamic provisioning. The binding decisions are integrated into the pod scheduling
|
// and dynamic provisioning. The binding decisions are integrated into the pod scheduling
|
||||||
// workflow so that the PV NodeAffinity is also considered along with the pod's other
|
// workflow so that the PV NodeAffinity is also considered along with the pod's other
|
||||||
@ -103,9 +119,10 @@ type volumeBinder struct {
|
|||||||
kubeClient clientset.Interface
|
kubeClient clientset.Interface
|
||||||
classLister storagelisters.StorageClassLister
|
classLister storagelisters.StorageClassLister
|
||||||
|
|
||||||
nodeInformer coreinformers.NodeInformer
|
nodeInformer coreinformers.NodeInformer
|
||||||
pvcCache PVCAssumeCache
|
csiNodeInformer storagev1beta1informers.CSINodeInformer
|
||||||
pvCache PVAssumeCache
|
pvcCache PVCAssumeCache
|
||||||
|
pvCache PVAssumeCache
|
||||||
|
|
||||||
// Stores binding decisions that were made in FindPodVolumes for use in AssumePodVolumes.
|
// Stores binding decisions that were made in FindPodVolumes for use in AssumePodVolumes.
|
||||||
// AssumePodVolumes modifies the bindings again for use in BindPodVolumes.
|
// AssumePodVolumes modifies the bindings again for use in BindPodVolumes.
|
||||||
@ -113,12 +130,15 @@ type volumeBinder struct {
|
|||||||
|
|
||||||
// Amount of time to wait for the bind operation to succeed
|
// Amount of time to wait for the bind operation to succeed
|
||||||
bindTimeout time.Duration
|
bindTimeout time.Duration
|
||||||
|
|
||||||
|
translator InTreeToCSITranslator
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewVolumeBinder sets up all the caches needed for the scheduler to make volume binding decisions.
|
// NewVolumeBinder sets up all the caches needed for the scheduler to make volume binding decisions.
|
||||||
func NewVolumeBinder(
|
func NewVolumeBinder(
|
||||||
kubeClient clientset.Interface,
|
kubeClient clientset.Interface,
|
||||||
nodeInformer coreinformers.NodeInformer,
|
nodeInformer coreinformers.NodeInformer,
|
||||||
|
csiNodeInformer storagev1beta1informers.CSINodeInformer,
|
||||||
pvcInformer coreinformers.PersistentVolumeClaimInformer,
|
pvcInformer coreinformers.PersistentVolumeClaimInformer,
|
||||||
pvInformer coreinformers.PersistentVolumeInformer,
|
pvInformer coreinformers.PersistentVolumeInformer,
|
||||||
storageClassInformer storageinformers.StorageClassInformer,
|
storageClassInformer storageinformers.StorageClassInformer,
|
||||||
@ -128,10 +148,12 @@ func NewVolumeBinder(
|
|||||||
kubeClient: kubeClient,
|
kubeClient: kubeClient,
|
||||||
classLister: storageClassInformer.Lister(),
|
classLister: storageClassInformer.Lister(),
|
||||||
nodeInformer: nodeInformer,
|
nodeInformer: nodeInformer,
|
||||||
|
csiNodeInformer: csiNodeInformer,
|
||||||
pvcCache: NewPVCAssumeCache(pvcInformer.Informer()),
|
pvcCache: NewPVCAssumeCache(pvcInformer.Informer()),
|
||||||
pvCache: NewPVAssumeCache(pvInformer.Informer()),
|
pvCache: NewPVAssumeCache(pvInformer.Informer()),
|
||||||
podBindingCache: NewPodBindingCache(),
|
podBindingCache: NewPodBindingCache(),
|
||||||
bindTimeout: bindTimeout,
|
bindTimeout: bindTimeout,
|
||||||
|
translator: csitrans.New(),
|
||||||
}
|
}
|
||||||
|
|
||||||
return b
|
return b
|
||||||
@ -457,6 +479,12 @@ func (b *volumeBinder) checkBindings(pod *v1.Pod, bindings []*bindingInfo, claim
|
|||||||
return false, fmt.Errorf("failed to get node %q: %v", pod.Spec.NodeName, err)
|
return false, fmt.Errorf("failed to get node %q: %v", pod.Spec.NodeName, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
csiNode, err := b.csiNodeInformer.Lister().Get(node.Name)
|
||||||
|
if err != nil {
|
||||||
|
// TODO: return the error once CSINode is created by default
|
||||||
|
klog.V(4).Infof("Could not get a CSINode object for the node %q: %v", node.Name, err)
|
||||||
|
}
|
||||||
|
|
||||||
// Check for any conditions that might require scheduling retry
|
// Check for any conditions that might require scheduling retry
|
||||||
|
|
||||||
// When pod is removed from scheduling queue because of deletion or any
|
// When pod is removed from scheduling queue because of deletion or any
|
||||||
@ -485,6 +513,11 @@ func (b *volumeBinder) checkBindings(pod *v1.Pod, bindings []*bindingInfo, claim
|
|||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pv, err = b.tryTranslatePVToCSI(pv, csiNode)
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("failed to translate pv to csi: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
// Check PV's node affinity (the node might not have the proper label)
|
// Check PV's node affinity (the node might not have the proper label)
|
||||||
if err := volumeutil.CheckNodeAffinity(pv, node.Labels); err != nil {
|
if err := volumeutil.CheckNodeAffinity(pv, node.Labels); err != nil {
|
||||||
return false, fmt.Errorf("pv %q node affinity doesn't match node %q: %v", pv.Name, node.Name, err)
|
return false, fmt.Errorf("pv %q node affinity doesn't match node %q: %v", pv.Name, node.Name, err)
|
||||||
@ -538,6 +571,12 @@ func (b *volumeBinder) checkBindings(pod *v1.Pod, bindings []*bindingInfo, claim
|
|||||||
}
|
}
|
||||||
return false, fmt.Errorf("failed to get pv %q from cache: %v", pvc.Spec.VolumeName, err)
|
return false, fmt.Errorf("failed to get pv %q from cache: %v", pvc.Spec.VolumeName, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pv, err = b.tryTranslatePVToCSI(pv, csiNode)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
if err := volumeutil.CheckNodeAffinity(pv, node.Labels); err != nil {
|
if err := volumeutil.CheckNodeAffinity(pv, node.Labels); err != nil {
|
||||||
return false, fmt.Errorf("pv %q node affinity doesn't match node %q: %v", pv.Name, node.Name, err)
|
return false, fmt.Errorf("pv %q node affinity doesn't match node %q: %v", pv.Name, node.Name, err)
|
||||||
}
|
}
|
||||||
@ -641,6 +680,12 @@ func (b *volumeBinder) getPodVolumes(pod *v1.Pod) (boundClaims []*v1.PersistentV
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (b *volumeBinder) checkBoundClaims(claims []*v1.PersistentVolumeClaim, node *v1.Node, podName string) (bool, error) {
|
func (b *volumeBinder) checkBoundClaims(claims []*v1.PersistentVolumeClaim, node *v1.Node, podName string) (bool, error) {
|
||||||
|
csiNode, err := b.csiNodeInformer.Lister().Get(node.Name)
|
||||||
|
if err != nil {
|
||||||
|
// TODO: return the error once CSINode is created by default
|
||||||
|
klog.V(4).Infof("Could not get a CSINode object for the node %q: %v", node.Name, err)
|
||||||
|
}
|
||||||
|
|
||||||
for _, pvc := range claims {
|
for _, pvc := range claims {
|
||||||
pvName := pvc.Spec.VolumeName
|
pvName := pvc.Spec.VolumeName
|
||||||
pv, err := b.pvCache.GetPV(pvName)
|
pv, err := b.pvCache.GetPV(pvName)
|
||||||
@ -648,6 +693,11 @@ func (b *volumeBinder) checkBoundClaims(claims []*v1.PersistentVolumeClaim, node
|
|||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pv, err = b.tryTranslatePVToCSI(pv, csiNode)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
err = volumeutil.CheckNodeAffinity(pv, node.Labels)
|
err = volumeutil.CheckNodeAffinity(pv, node.Labels)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
klog.V(4).Infof("PersistentVolume %q, Node %q mismatch for Pod %q: %v", pvName, node.Name, podName, err)
|
klog.V(4).Infof("PersistentVolume %q, Node %q mismatch for Pod %q: %v", pvName, node.Name, podName, err)
|
||||||
@ -783,3 +833,72 @@ func (a byPVCSize) Less(i, j int) bool {
|
|||||||
func claimToClaimKey(claim *v1.PersistentVolumeClaim) string {
|
func claimToClaimKey(claim *v1.PersistentVolumeClaim) string {
|
||||||
return fmt.Sprintf("%s/%s", claim.Namespace, claim.Name)
|
return fmt.Sprintf("%s/%s", claim.Namespace, claim.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// isCSIMigrationOnForPlugin checks if CSI migrartion is enabled for a given plugin.
|
||||||
|
func isCSIMigrationOnForPlugin(pluginName string) bool {
|
||||||
|
switch pluginName {
|
||||||
|
case csiplugins.AWSEBSInTreePluginName:
|
||||||
|
return utilfeature.DefaultFeatureGate.Enabled(features.CSIMigrationAWS)
|
||||||
|
case csiplugins.GCEPDInTreePluginName:
|
||||||
|
return utilfeature.DefaultFeatureGate.Enabled(features.CSIMigrationGCE)
|
||||||
|
case csiplugins.AzureDiskInTreePluginName:
|
||||||
|
return utilfeature.DefaultFeatureGate.Enabled(features.CSIMigrationAzureDisk)
|
||||||
|
case csiplugins.CinderInTreePluginName:
|
||||||
|
return utilfeature.DefaultFeatureGate.Enabled(features.CSIMigrationOpenStack)
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// isPluginMigratedToCSIOnNode checks if an in-tree plugin has been migrated to a CSI driver on the node.
|
||||||
|
func isPluginMigratedToCSIOnNode(pluginName string, csiNode *storagev1beta1.CSINode) bool {
|
||||||
|
if csiNode == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
csiNodeAnn := csiNode.GetAnnotations()
|
||||||
|
if csiNodeAnn == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
var mpaSet sets.String
|
||||||
|
mpa := csiNodeAnn[v1.MigratedPluginsAnnotationKey]
|
||||||
|
if len(mpa) == 0 {
|
||||||
|
mpaSet = sets.NewString()
|
||||||
|
} else {
|
||||||
|
tok := strings.Split(mpa, ",")
|
||||||
|
mpaSet = sets.NewString(tok...)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mpaSet.Has(pluginName)
|
||||||
|
}
|
||||||
|
|
||||||
|
// tryTranslatePVToCSI will translate the in-tree PV to CSI if it meets the criteria. If not, it returns the unmodified in-tree PV.
|
||||||
|
func (b *volumeBinder) tryTranslatePVToCSI(pv *v1.PersistentVolume, csiNode *storagev1beta1.CSINode) (*v1.PersistentVolume, error) {
|
||||||
|
if !b.translator.IsPVMigratable(pv) {
|
||||||
|
return pv, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !utilfeature.DefaultFeatureGate.Enabled(features.CSIMigration) {
|
||||||
|
return pv, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pluginName, err := b.translator.GetInTreePluginNameFromSpec(pv, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("could not get plugin name from pv: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isCSIMigrationOnForPlugin(pluginName) {
|
||||||
|
return pv, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isPluginMigratedToCSIOnNode(pluginName, csiNode) {
|
||||||
|
return pv, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
transPV, err := b.translator.TranslateInTreePVToCSI(pv)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("could not translate pv: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return transPV, nil
|
||||||
|
}
|
||||||
|
@ -25,22 +25,27 @@ import (
|
|||||||
|
|
||||||
v1 "k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
storagev1 "k8s.io/api/storage/v1"
|
storagev1 "k8s.io/api/storage/v1"
|
||||||
|
storagev1beta1 "k8s.io/api/storage/v1beta1"
|
||||||
"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/types"
|
"k8s.io/apimachinery/pkg/types"
|
||||||
"k8s.io/apimachinery/pkg/util/diff"
|
"k8s.io/apimachinery/pkg/util/diff"
|
||||||
"k8s.io/apimachinery/pkg/util/wait"
|
"k8s.io/apimachinery/pkg/util/wait"
|
||||||
"k8s.io/apimachinery/pkg/watch"
|
"k8s.io/apimachinery/pkg/watch"
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
"k8s.io/client-go/informers"
|
"k8s.io/client-go/informers"
|
||||||
coreinformers "k8s.io/client-go/informers/core/v1"
|
coreinformers "k8s.io/client-go/informers/core/v1"
|
||||||
|
storagev1beta1informers "k8s.io/client-go/informers/storage/v1beta1"
|
||||||
clientset "k8s.io/client-go/kubernetes"
|
clientset "k8s.io/client-go/kubernetes"
|
||||||
"k8s.io/client-go/kubernetes/fake"
|
"k8s.io/client-go/kubernetes/fake"
|
||||||
k8stesting "k8s.io/client-go/testing"
|
k8stesting "k8s.io/client-go/testing"
|
||||||
|
featuregatetesting "k8s.io/component-base/featuregate/testing"
|
||||||
"k8s.io/klog"
|
"k8s.io/klog"
|
||||||
"k8s.io/kubernetes/pkg/api/testapi"
|
"k8s.io/kubernetes/pkg/api/testapi"
|
||||||
"k8s.io/kubernetes/pkg/controller"
|
"k8s.io/kubernetes/pkg/controller"
|
||||||
pvtesting "k8s.io/kubernetes/pkg/controller/volume/persistentvolume/testing"
|
pvtesting "k8s.io/kubernetes/pkg/controller/volume/persistentvolume/testing"
|
||||||
pvutil "k8s.io/kubernetes/pkg/controller/volume/persistentvolume/util"
|
pvutil "k8s.io/kubernetes/pkg/controller/volume/persistentvolume/util"
|
||||||
|
"k8s.io/kubernetes/pkg/features"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -65,6 +70,10 @@ var (
|
|||||||
|
|
||||||
selectedNodePVC = makeTestPVC("provisioned-pvc", "1Gi", nodeLabelValue, pvcSelectedNode, "", "1", &waitClassWithProvisioner)
|
selectedNodePVC = makeTestPVC("provisioned-pvc", "1Gi", nodeLabelValue, pvcSelectedNode, "", "1", &waitClassWithProvisioner)
|
||||||
|
|
||||||
|
// PVCs for CSI migration
|
||||||
|
boundMigrationPVC = makeTestPVC("pvc-migration-bound", "1G", "", pvcBound, "pv-migration-bound", "1", &waitClass)
|
||||||
|
provMigrationPVCBound = makeTestPVC("pvc-migration-provisioned", "1Gi", "", pvcBound, "pv-migration-bound", "1", &waitClassWithProvisioner)
|
||||||
|
|
||||||
// PVs for manual binding
|
// PVs for manual binding
|
||||||
pvNode1a = makeTestPV("pv-node1a", "node1", "5G", "1", nil, waitClass)
|
pvNode1a = makeTestPV("pv-node1a", "node1", "5G", "1", nil, waitClass)
|
||||||
pvNode1b = makeTestPV("pv-node1b", "node1", "10G", "1", nil, waitClass)
|
pvNode1b = makeTestPV("pv-node1b", "node1", "10G", "1", nil, waitClass)
|
||||||
@ -77,6 +86,10 @@ var (
|
|||||||
pvBoundImmediate = makeTestPV("pv-bound-immediate", "node1", "1G", "1", immediateBoundPVC, immediateClass)
|
pvBoundImmediate = makeTestPV("pv-bound-immediate", "node1", "1G", "1", immediateBoundPVC, immediateClass)
|
||||||
pvBoundImmediateNode2 = makeTestPV("pv-bound-immediate", "node2", "1G", "1", immediateBoundPVC, immediateClass)
|
pvBoundImmediateNode2 = makeTestPV("pv-bound-immediate", "node2", "1G", "1", immediateBoundPVC, immediateClass)
|
||||||
|
|
||||||
|
// PVs for CSI migration
|
||||||
|
migrationPVBound = makeTestPVForCSIMigration(zone1Labels, boundMigrationPVC)
|
||||||
|
migrationPVBoundToUnbound = makeTestPVForCSIMigration(zone1Labels, unboundPVC)
|
||||||
|
|
||||||
// storage class names
|
// storage class names
|
||||||
waitClass = "waitClass"
|
waitClass = "waitClass"
|
||||||
immediateClass = "immediateClass"
|
immediateClass = "immediateClass"
|
||||||
@ -87,20 +100,30 @@ var (
|
|||||||
node1 = makeNode("node1", map[string]string{nodeLabelKey: "node1"})
|
node1 = makeNode("node1", map[string]string{nodeLabelKey: "node1"})
|
||||||
node2 = makeNode("node2", map[string]string{nodeLabelKey: "node2"})
|
node2 = makeNode("node2", map[string]string{nodeLabelKey: "node2"})
|
||||||
node1NoLabels = makeNode("node1", nil)
|
node1NoLabels = makeNode("node1", nil)
|
||||||
|
node1Zone1 = makeNode("node1", map[string]string{"topology.gke.io/zone": "us-east-1"})
|
||||||
|
node1Zone2 = makeNode("node1", map[string]string{"topology.gke.io/zone": "us-east-2"})
|
||||||
|
|
||||||
|
// csiNode objects
|
||||||
|
csiNode1Migrated = makeCSINode("node1", "kubernetes.io/gce-pd")
|
||||||
|
csiNode1NotMigrated = makeCSINode("node1", "")
|
||||||
|
|
||||||
// node topology
|
// node topology
|
||||||
nodeLabelKey = "nodeKey"
|
nodeLabelKey = "nodeKey"
|
||||||
nodeLabelValue = "node1"
|
nodeLabelValue = "node1"
|
||||||
|
|
||||||
|
// node topology for CSI migration
|
||||||
|
zone1Labels = map[string]string{v1.LabelZoneFailureDomain: "us-east-1", v1.LabelZoneRegion: "us-east-1a"}
|
||||||
)
|
)
|
||||||
|
|
||||||
type testEnv struct {
|
type testEnv struct {
|
||||||
client clientset.Interface
|
client clientset.Interface
|
||||||
reactor *pvtesting.VolumeReactor
|
reactor *pvtesting.VolumeReactor
|
||||||
binder SchedulerVolumeBinder
|
binder SchedulerVolumeBinder
|
||||||
internalBinder *volumeBinder
|
internalBinder *volumeBinder
|
||||||
internalNodeInformer coreinformers.NodeInformer
|
internalNodeInformer coreinformers.NodeInformer
|
||||||
internalPVCache *assumeCache
|
internalCSINodeInformer storagev1beta1informers.CSINodeInformer
|
||||||
internalPVCCache *assumeCache
|
internalPVCache *assumeCache
|
||||||
|
internalPVCCache *assumeCache
|
||||||
}
|
}
|
||||||
|
|
||||||
func newTestBinder(t *testing.T, stopCh <-chan struct{}) *testEnv {
|
func newTestBinder(t *testing.T, stopCh <-chan struct{}) *testEnv {
|
||||||
@ -119,11 +142,13 @@ func newTestBinder(t *testing.T, stopCh <-chan struct{}) *testEnv {
|
|||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
||||||
|
|
||||||
nodeInformer := informerFactory.Core().V1().Nodes()
|
nodeInformer := informerFactory.Core().V1().Nodes()
|
||||||
|
csiNodeInformer := informerFactory.Storage().V1beta1().CSINodes()
|
||||||
pvcInformer := informerFactory.Core().V1().PersistentVolumeClaims()
|
pvcInformer := informerFactory.Core().V1().PersistentVolumeClaims()
|
||||||
classInformer := informerFactory.Storage().V1().StorageClasses()
|
classInformer := informerFactory.Storage().V1().StorageClasses()
|
||||||
binder := NewVolumeBinder(
|
binder := NewVolumeBinder(
|
||||||
client,
|
client,
|
||||||
nodeInformer,
|
nodeInformer,
|
||||||
|
csiNodeInformer,
|
||||||
pvcInformer,
|
pvcInformer,
|
||||||
informerFactory.Core().V1().PersistentVolumes(),
|
informerFactory.Core().V1().PersistentVolumes(),
|
||||||
classInformer,
|
classInformer,
|
||||||
@ -214,13 +239,14 @@ func newTestBinder(t *testing.T, stopCh <-chan struct{}) *testEnv {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return &testEnv{
|
return &testEnv{
|
||||||
client: client,
|
client: client,
|
||||||
reactor: reactor,
|
reactor: reactor,
|
||||||
binder: binder,
|
binder: binder,
|
||||||
internalBinder: internalBinder,
|
internalBinder: internalBinder,
|
||||||
internalNodeInformer: nodeInformer,
|
internalNodeInformer: nodeInformer,
|
||||||
internalPVCache: internalPVCache,
|
internalCSINodeInformer: csiNodeInformer,
|
||||||
internalPVCCache: internalPVCCache,
|
internalPVCache: internalPVCache,
|
||||||
|
internalPVCCache: internalPVCCache,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -231,6 +257,13 @@ func (env *testEnv) initNodes(cachedNodes []*v1.Node) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (env *testEnv) initCSINodes(cachedCSINodes []*storagev1beta1.CSINode) {
|
||||||
|
csiNodeInformer := env.internalCSINodeInformer.Informer()
|
||||||
|
for _, csiNode := range cachedCSINodes {
|
||||||
|
csiNodeInformer.GetIndexer().Add(csiNode)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (env *testEnv) initClaims(cachedPVCs []*v1.PersistentVolumeClaim, apiPVCs []*v1.PersistentVolumeClaim) {
|
func (env *testEnv) initClaims(cachedPVCs []*v1.PersistentVolumeClaim, apiPVCs []*v1.PersistentVolumeClaim) {
|
||||||
internalPVCCache := env.internalPVCCache
|
internalPVCCache := env.internalPVCCache
|
||||||
for _, pvc := range cachedPVCs {
|
for _, pvc := range cachedPVCs {
|
||||||
@ -593,6 +626,21 @@ func makeTestPV(name, node, capacity, version string, boundToPVC *v1.PersistentV
|
|||||||
return pv
|
return pv
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func makeTestPVForCSIMigration(labels map[string]string, pvc *v1.PersistentVolumeClaim) *v1.PersistentVolume {
|
||||||
|
pv := makeTestPV("pv-migration-bound", "node1", "1G", "1", pvc, waitClass)
|
||||||
|
pv.Spec.NodeAffinity = nil // Will be written by the CSI translation lib
|
||||||
|
pv.ObjectMeta.Labels = labels
|
||||||
|
pv.Spec.PersistentVolumeSource = v1.PersistentVolumeSource{
|
||||||
|
GCEPersistentDisk: &v1.GCEPersistentDiskVolumeSource{
|
||||||
|
PDName: "test-disk",
|
||||||
|
FSType: "ext4",
|
||||||
|
Partition: 0,
|
||||||
|
ReadOnly: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
return pv
|
||||||
|
}
|
||||||
|
|
||||||
func pvcSetSelectedNode(pvc *v1.PersistentVolumeClaim, node string) *v1.PersistentVolumeClaim {
|
func pvcSetSelectedNode(pvc *v1.PersistentVolumeClaim, node string) *v1.PersistentVolumeClaim {
|
||||||
newPVC := pvc.DeepCopy()
|
newPVC := pvc.DeepCopy()
|
||||||
metav1.SetMetaDataAnnotation(&newPVC.ObjectMeta, pvutil.AnnSelectedNode, node)
|
metav1.SetMetaDataAnnotation(&newPVC.ObjectMeta, pvutil.AnnSelectedNode, node)
|
||||||
@ -620,6 +668,17 @@ func makeNode(name string, labels map[string]string) *v1.Node {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func makeCSINode(name, migratedPlugin string) *storagev1beta1.CSINode {
|
||||||
|
return &storagev1beta1.CSINode{
|
||||||
|
ObjectMeta: metav1.ObjectMeta{
|
||||||
|
Name: name,
|
||||||
|
Annotations: map[string]string{
|
||||||
|
v1.MigratedPluginsAnnotationKey: migratedPlugin,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func makePod(pvcs []*v1.PersistentVolumeClaim) *v1.Pod {
|
func makePod(pvcs []*v1.PersistentVolumeClaim) *v1.Pod {
|
||||||
pod := &v1.Pod{
|
pod := &v1.Pod{
|
||||||
ObjectMeta: metav1.ObjectMeta{
|
ObjectMeta: metav1.ObjectMeta{
|
||||||
@ -983,6 +1042,115 @@ func TestFindPodVolumesWithProvisioning(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TestFindPodVolumesWithCSIMigration aims to test the node affinity check procedure that's
|
||||||
|
// done in FindPodVolumes. In order to reach this code path, the given PVCs must be bound to a PV.
|
||||||
|
func TestFindPodVolumesWithCSIMigration(t *testing.T) {
|
||||||
|
scenarios := map[string]struct {
|
||||||
|
// Inputs
|
||||||
|
pvs []*v1.PersistentVolume
|
||||||
|
podPVCs []*v1.PersistentVolumeClaim
|
||||||
|
// If nil, use pod PVCs
|
||||||
|
cachePVCs []*v1.PersistentVolumeClaim
|
||||||
|
// If nil, makePod with podPVCs
|
||||||
|
pod *v1.Pod
|
||||||
|
|
||||||
|
// Setup
|
||||||
|
initNodes []*v1.Node
|
||||||
|
initCSINodes []*storagev1beta1.CSINode
|
||||||
|
|
||||||
|
// Expected return values
|
||||||
|
expectedUnbound bool
|
||||||
|
expectedBound bool
|
||||||
|
shouldFail bool
|
||||||
|
}{
|
||||||
|
"pvc-bound": {
|
||||||
|
podPVCs: []*v1.PersistentVolumeClaim{boundMigrationPVC},
|
||||||
|
pvs: []*v1.PersistentVolume{migrationPVBound},
|
||||||
|
initNodes: []*v1.Node{node1Zone1},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1Migrated},
|
||||||
|
expectedBound: true,
|
||||||
|
expectedUnbound: true,
|
||||||
|
},
|
||||||
|
"pvc-bound,csinode-not-migrated": {
|
||||||
|
podPVCs: []*v1.PersistentVolumeClaim{boundMigrationPVC},
|
||||||
|
pvs: []*v1.PersistentVolume{migrationPVBound},
|
||||||
|
initNodes: []*v1.Node{node1Zone1},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1NotMigrated},
|
||||||
|
expectedBound: true,
|
||||||
|
expectedUnbound: true,
|
||||||
|
},
|
||||||
|
"pvc-bound,missing-csinode": {
|
||||||
|
podPVCs: []*v1.PersistentVolumeClaim{boundMigrationPVC},
|
||||||
|
pvs: []*v1.PersistentVolume{migrationPVBound},
|
||||||
|
initNodes: []*v1.Node{node1Zone1},
|
||||||
|
expectedBound: true,
|
||||||
|
expectedUnbound: true,
|
||||||
|
},
|
||||||
|
"pvc-bound,node-different-zone": {
|
||||||
|
podPVCs: []*v1.PersistentVolumeClaim{boundMigrationPVC},
|
||||||
|
pvs: []*v1.PersistentVolume{migrationPVBound},
|
||||||
|
initNodes: []*v1.Node{node1Zone2},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1Migrated},
|
||||||
|
expectedBound: false,
|
||||||
|
expectedUnbound: true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.CSIMigration, true)()
|
||||||
|
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.CSIMigrationGCE, true)()
|
||||||
|
|
||||||
|
for name, scenario := range scenarios {
|
||||||
|
klog.V(5).Infof("Running test case %q", name)
|
||||||
|
|
||||||
|
// Setup
|
||||||
|
testEnv := newTestBinder(t, ctx.Done())
|
||||||
|
testEnv.initVolumes(scenario.pvs, scenario.pvs)
|
||||||
|
|
||||||
|
var node *v1.Node
|
||||||
|
if len(scenario.initNodes) > 0 {
|
||||||
|
testEnv.initNodes(scenario.initNodes)
|
||||||
|
node = scenario.initNodes[0]
|
||||||
|
} else {
|
||||||
|
node = node1
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(scenario.initCSINodes) > 0 {
|
||||||
|
testEnv.initCSINodes(scenario.initCSINodes)
|
||||||
|
}
|
||||||
|
|
||||||
|
// a. Init pvc cache
|
||||||
|
if scenario.cachePVCs == nil {
|
||||||
|
scenario.cachePVCs = scenario.podPVCs
|
||||||
|
}
|
||||||
|
testEnv.initClaims(scenario.cachePVCs, scenario.cachePVCs)
|
||||||
|
|
||||||
|
// b. Generate pod with given claims
|
||||||
|
if scenario.pod == nil {
|
||||||
|
scenario.pod = makePod(scenario.podPVCs)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Execute
|
||||||
|
unboundSatisfied, boundSatisfied, err := testEnv.binder.FindPodVolumes(scenario.pod, node)
|
||||||
|
|
||||||
|
// Validate
|
||||||
|
if !scenario.shouldFail && err != nil {
|
||||||
|
t.Errorf("Test %q failed: returned error: %v", name, err)
|
||||||
|
}
|
||||||
|
if scenario.shouldFail && err == nil {
|
||||||
|
t.Errorf("Test %q failed: returned success but expected error", name)
|
||||||
|
}
|
||||||
|
if boundSatisfied != scenario.expectedBound {
|
||||||
|
t.Errorf("Test %q failed: expected boundSatsified %v, got %v", name, scenario.expectedBound, boundSatisfied)
|
||||||
|
}
|
||||||
|
if unboundSatisfied != scenario.expectedUnbound {
|
||||||
|
t.Errorf("Test %q failed: expected unboundSatsified %v, got %v", name, scenario.expectedUnbound, unboundSatisfied)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestAssumePodVolumes(t *testing.T) {
|
func TestAssumePodVolumes(t *testing.T) {
|
||||||
scenarios := map[string]struct {
|
scenarios := map[string]struct {
|
||||||
// Inputs
|
// Inputs
|
||||||
@ -1414,6 +1582,122 @@ func TestCheckBindings(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestCheckBindingsWithCSIMigration(t *testing.T) {
|
||||||
|
scenarios := map[string]struct {
|
||||||
|
// Inputs
|
||||||
|
initPVs []*v1.PersistentVolume
|
||||||
|
initPVCs []*v1.PersistentVolumeClaim
|
||||||
|
initNodes []*v1.Node
|
||||||
|
initCSINodes []*storagev1beta1.CSINode
|
||||||
|
|
||||||
|
bindings []*bindingInfo
|
||||||
|
provisionedPVCs []*v1.PersistentVolumeClaim
|
||||||
|
|
||||||
|
// API updates before checking
|
||||||
|
apiPVs []*v1.PersistentVolume
|
||||||
|
apiPVCs []*v1.PersistentVolumeClaim
|
||||||
|
|
||||||
|
// Expected return values
|
||||||
|
shouldFail bool
|
||||||
|
expectedBound bool
|
||||||
|
migrationEnabled bool
|
||||||
|
}{
|
||||||
|
"provisioning-pvc-bound": {
|
||||||
|
bindings: []*bindingInfo{},
|
||||||
|
provisionedPVCs: []*v1.PersistentVolumeClaim{addProvisionAnn(provMigrationPVCBound)},
|
||||||
|
initPVs: []*v1.PersistentVolume{migrationPVBound},
|
||||||
|
initPVCs: []*v1.PersistentVolumeClaim{provMigrationPVCBound},
|
||||||
|
initNodes: []*v1.Node{node1Zone1},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1Migrated},
|
||||||
|
apiPVCs: []*v1.PersistentVolumeClaim{addProvisionAnn(provMigrationPVCBound)},
|
||||||
|
expectedBound: true,
|
||||||
|
},
|
||||||
|
"binding-node-pv-same-zone": {
|
||||||
|
bindings: []*bindingInfo{makeBinding(unboundPVC, migrationPVBoundToUnbound)},
|
||||||
|
provisionedPVCs: []*v1.PersistentVolumeClaim{},
|
||||||
|
initPVs: []*v1.PersistentVolume{migrationPVBoundToUnbound},
|
||||||
|
initPVCs: []*v1.PersistentVolumeClaim{unboundPVC},
|
||||||
|
initNodes: []*v1.Node{node1Zone1},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1Migrated},
|
||||||
|
migrationEnabled: true,
|
||||||
|
},
|
||||||
|
"binding-without-csinode": {
|
||||||
|
bindings: []*bindingInfo{makeBinding(unboundPVC, migrationPVBoundToUnbound)},
|
||||||
|
provisionedPVCs: []*v1.PersistentVolumeClaim{},
|
||||||
|
initPVs: []*v1.PersistentVolume{migrationPVBoundToUnbound},
|
||||||
|
initPVCs: []*v1.PersistentVolumeClaim{unboundPVC},
|
||||||
|
initNodes: []*v1.Node{node1Zone1},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{},
|
||||||
|
migrationEnabled: true,
|
||||||
|
},
|
||||||
|
"binding-non-migrated-plugin": {
|
||||||
|
bindings: []*bindingInfo{makeBinding(unboundPVC, migrationPVBoundToUnbound)},
|
||||||
|
provisionedPVCs: []*v1.PersistentVolumeClaim{},
|
||||||
|
initPVs: []*v1.PersistentVolume{migrationPVBoundToUnbound},
|
||||||
|
initPVCs: []*v1.PersistentVolumeClaim{unboundPVC},
|
||||||
|
initNodes: []*v1.Node{node1Zone1},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1NotMigrated},
|
||||||
|
migrationEnabled: true,
|
||||||
|
},
|
||||||
|
"binding-node-pv-in-different-zones": {
|
||||||
|
bindings: []*bindingInfo{makeBinding(unboundPVC, migrationPVBoundToUnbound)},
|
||||||
|
provisionedPVCs: []*v1.PersistentVolumeClaim{},
|
||||||
|
initPVs: []*v1.PersistentVolume{migrationPVBoundToUnbound},
|
||||||
|
initPVCs: []*v1.PersistentVolumeClaim{unboundPVC},
|
||||||
|
initNodes: []*v1.Node{node1Zone2},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1Migrated},
|
||||||
|
migrationEnabled: true,
|
||||||
|
shouldFail: true,
|
||||||
|
},
|
||||||
|
"binding-node-pv-different-zones-migration-off": {
|
||||||
|
bindings: []*bindingInfo{makeBinding(unboundPVC, migrationPVBoundToUnbound)},
|
||||||
|
provisionedPVCs: []*v1.PersistentVolumeClaim{},
|
||||||
|
initPVs: []*v1.PersistentVolume{migrationPVBoundToUnbound},
|
||||||
|
initPVCs: []*v1.PersistentVolumeClaim{unboundPVC},
|
||||||
|
initNodes: []*v1.Node{node1Zone2},
|
||||||
|
initCSINodes: []*storagev1beta1.CSINode{csiNode1Migrated},
|
||||||
|
migrationEnabled: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
for name, scenario := range scenarios {
|
||||||
|
t.Run(name, func(t *testing.T) {
|
||||||
|
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.CSIMigration, scenario.migrationEnabled)()
|
||||||
|
defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.CSIMigrationGCE, scenario.migrationEnabled)()
|
||||||
|
|
||||||
|
// Setup
|
||||||
|
pod := makePod(nil)
|
||||||
|
testEnv := newTestBinder(t, ctx.Done())
|
||||||
|
testEnv.initNodes(scenario.initNodes)
|
||||||
|
testEnv.initCSINodes(scenario.initCSINodes)
|
||||||
|
testEnv.initVolumes(scenario.initPVs, nil)
|
||||||
|
testEnv.initClaims(scenario.initPVCs, nil)
|
||||||
|
testEnv.assumeVolumes(t, name, "node1", pod, scenario.bindings, scenario.provisionedPVCs)
|
||||||
|
|
||||||
|
// Before execute
|
||||||
|
testEnv.updateVolumes(t, scenario.apiPVs, true)
|
||||||
|
testEnv.updateClaims(t, scenario.apiPVCs, true)
|
||||||
|
|
||||||
|
// Execute
|
||||||
|
allBound, err := testEnv.internalBinder.checkBindings(pod, scenario.bindings, scenario.provisionedPVCs)
|
||||||
|
|
||||||
|
// Validate
|
||||||
|
if !scenario.shouldFail && err != nil {
|
||||||
|
t.Errorf("Test %q failed: returned error: %v", name, err)
|
||||||
|
}
|
||||||
|
if scenario.shouldFail && err == nil {
|
||||||
|
t.Errorf("Test %q failed: returned success but expected error", name)
|
||||||
|
}
|
||||||
|
if scenario.expectedBound != allBound {
|
||||||
|
t.Errorf("Test %q failed: returned bound %v", name, allBound)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestBindPodVolumes(t *testing.T) {
|
func TestBindPodVolumes(t *testing.T) {
|
||||||
type scenarioType struct {
|
type scenarioType struct {
|
||||||
// Inputs
|
// Inputs
|
||||||
|
@ -276,6 +276,7 @@ func New(client clientset.Interface,
|
|||||||
volumeBinder := volumebinder.NewVolumeBinder(
|
volumeBinder := volumebinder.NewVolumeBinder(
|
||||||
client,
|
client,
|
||||||
informerFactory.Core().V1().Nodes(),
|
informerFactory.Core().V1().Nodes(),
|
||||||
|
informerFactory.Storage().V1beta1().CSINodes(),
|
||||||
informerFactory.Core().V1().PersistentVolumeClaims(),
|
informerFactory.Core().V1().PersistentVolumeClaims(),
|
||||||
informerFactory.Core().V1().PersistentVolumes(),
|
informerFactory.Core().V1().PersistentVolumes(),
|
||||||
informerFactory.Storage().V1().StorageClasses(),
|
informerFactory.Storage().V1().StorageClasses(),
|
||||||
|
@ -10,6 +10,7 @@ go_library(
|
|||||||
"//staging/src/k8s.io/api/core/v1:go_default_library",
|
"//staging/src/k8s.io/api/core/v1:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/informers/core/v1:go_default_library",
|
"//staging/src/k8s.io/client-go/informers/core/v1:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/informers/storage/v1:go_default_library",
|
"//staging/src/k8s.io/client-go/informers/storage/v1:go_default_library",
|
||||||
|
"//staging/src/k8s.io/client-go/informers/storage/v1beta1:go_default_library",
|
||||||
"//staging/src/k8s.io/client-go/kubernetes:go_default_library",
|
"//staging/src/k8s.io/client-go/kubernetes:go_default_library",
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
@ -19,9 +19,10 @@ package volumebinder
|
|||||||
import (
|
import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"k8s.io/api/core/v1"
|
v1 "k8s.io/api/core/v1"
|
||||||
coreinformers "k8s.io/client-go/informers/core/v1"
|
coreinformers "k8s.io/client-go/informers/core/v1"
|
||||||
storageinformers "k8s.io/client-go/informers/storage/v1"
|
storageinformers "k8s.io/client-go/informers/storage/v1"
|
||||||
|
storagev1beta1informers "k8s.io/client-go/informers/storage/v1beta1"
|
||||||
clientset "k8s.io/client-go/kubernetes"
|
clientset "k8s.io/client-go/kubernetes"
|
||||||
volumescheduling "k8s.io/kubernetes/pkg/controller/volume/scheduling"
|
volumescheduling "k8s.io/kubernetes/pkg/controller/volume/scheduling"
|
||||||
)
|
)
|
||||||
@ -35,13 +36,14 @@ type VolumeBinder struct {
|
|||||||
func NewVolumeBinder(
|
func NewVolumeBinder(
|
||||||
client clientset.Interface,
|
client clientset.Interface,
|
||||||
nodeInformer coreinformers.NodeInformer,
|
nodeInformer coreinformers.NodeInformer,
|
||||||
|
csiNodeInformer storagev1beta1informers.CSINodeInformer,
|
||||||
pvcInformer coreinformers.PersistentVolumeClaimInformer,
|
pvcInformer coreinformers.PersistentVolumeClaimInformer,
|
||||||
pvInformer coreinformers.PersistentVolumeInformer,
|
pvInformer coreinformers.PersistentVolumeInformer,
|
||||||
storageClassInformer storageinformers.StorageClassInformer,
|
storageClassInformer storageinformers.StorageClassInformer,
|
||||||
bindTimeout time.Duration) *VolumeBinder {
|
bindTimeout time.Duration) *VolumeBinder {
|
||||||
|
|
||||||
return &VolumeBinder{
|
return &VolumeBinder{
|
||||||
Binder: volumescheduling.NewVolumeBinder(client, nodeInformer, pvcInformer, pvInformer, storageClassInformer, bindTimeout),
|
Binder: volumescheduling.NewVolumeBinder(client, nodeInformer, csiNodeInformer, pvcInformer, pvInformer, storageClassInformer, bindTimeout),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user