Merge pull request #106322 from amacaskill/volume-source-cloning-e2e
Fix failing volume cloning e2e test for GCE PD CSI Driver
This commit is contained in:
		| @@ -26,6 +26,7 @@ const ( | |||||||
| 	podDeleteTimeout                 = 5 * time.Minute | 	podDeleteTimeout                 = 5 * time.Minute | ||||||
| 	claimProvisionTimeout            = 5 * time.Minute | 	claimProvisionTimeout            = 5 * time.Minute | ||||||
| 	claimProvisionShortTimeout       = 1 * time.Minute | 	claimProvisionShortTimeout       = 1 * time.Minute | ||||||
|  | 	dataSourceProvisionTimeout       = 5 * time.Minute | ||||||
| 	claimBoundTimeout                = 3 * time.Minute | 	claimBoundTimeout                = 3 * time.Minute | ||||||
| 	pvReclaimTimeout                 = 3 * time.Minute | 	pvReclaimTimeout                 = 3 * time.Minute | ||||||
| 	pvBoundTimeout                   = 3 * time.Minute | 	pvBoundTimeout                   = 3 * time.Minute | ||||||
| @@ -56,6 +57,9 @@ type TimeoutContext struct { | |||||||
| 	// ClaimProvision is how long claims have to become dynamically provisioned. | 	// ClaimProvision is how long claims have to become dynamically provisioned. | ||||||
| 	ClaimProvision time.Duration | 	ClaimProvision time.Duration | ||||||
|  |  | ||||||
|  | 	// DataSourceProvision is how long claims have to become dynamically provisioned from source claim. | ||||||
|  | 	DataSourceProvision time.Duration | ||||||
|  |  | ||||||
| 	// ClaimProvisionShort is the same as `ClaimProvision`, but shorter. | 	// ClaimProvisionShort is the same as `ClaimProvision`, but shorter. | ||||||
| 	ClaimProvisionShort time.Duration | 	ClaimProvisionShort time.Duration | ||||||
|  |  | ||||||
| @@ -96,6 +100,7 @@ func NewTimeoutContextWithDefaults() *TimeoutContext { | |||||||
| 		PodDelete:                 podDeleteTimeout, | 		PodDelete:                 podDeleteTimeout, | ||||||
| 		ClaimProvision:            claimProvisionTimeout, | 		ClaimProvision:            claimProvisionTimeout, | ||||||
| 		ClaimProvisionShort:       claimProvisionShortTimeout, | 		ClaimProvisionShort:       claimProvisionShortTimeout, | ||||||
|  | 		DataSourceProvision:       dataSourceProvisionTimeout, | ||||||
| 		ClaimBound:                claimBoundTimeout, | 		ClaimBound:                claimBoundTimeout, | ||||||
| 		PVReclaim:                 pvReclaimTimeout, | 		PVReclaim:                 pvReclaimTimeout, | ||||||
| 		PVBound:                   pvBoundTimeout, | 		PVBound:                   pvBoundTimeout, | ||||||
|   | |||||||
| @@ -41,6 +41,7 @@ package volume | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"crypto/sha256" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| @@ -51,6 +52,7 @@ import ( | |||||||
| 	apierrors "k8s.io/apimachinery/pkg/api/errors" | 	apierrors "k8s.io/apimachinery/pkg/api/errors" | ||||||
| 	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/wait" | ||||||
| 	clientset "k8s.io/client-go/kubernetes" | 	clientset "k8s.io/client-go/kubernetes" | ||||||
| 	clientexec "k8s.io/client-go/util/exec" | 	clientexec "k8s.io/client-go/util/exec" | ||||||
| 	"k8s.io/kubernetes/test/e2e/framework" | 	"k8s.io/kubernetes/test/e2e/framework" | ||||||
| @@ -235,6 +237,73 @@ func CreateStorageServer(cs clientset.Interface, config TestConfig) (pod *v1.Pod | |||||||
| 	return pod, ip | 	return pod, ip | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // GetVolumeAttachmentName returns the hash value of the provisioner, the config ClientNodeSelection name, | ||||||
|  | // and the VolumeAttachment name of the PV that is bound to the PVC with the passed in claimName and claimNamespace. | ||||||
|  | func GetVolumeAttachmentName(cs clientset.Interface, config TestConfig, provisioner string, claimName string, claimNamespace string) string { | ||||||
|  | 	var nodeName string | ||||||
|  | 	// For provisioning tests, ClientNodeSelection is not set so we do not know the NodeName of the VolumeAttachment of the PV that is | ||||||
|  | 	// bound to the PVC with the passed in claimName and claimNamespace. We need this NodeName because it is used to generate the | ||||||
|  | 	// attachmentName that is returned, and used to look up a certain VolumeAttachment in WaitForVolumeAttachmentTerminated. | ||||||
|  | 	// To get the nodeName of the VolumeAttachment, we get all the VolumeAttachments, look for the VolumeAttachment with a | ||||||
|  | 	// PersistentVolumeName equal to the PV that is bound to the passed in PVC, and then we get the NodeName from that VolumeAttachment. | ||||||
|  | 	if config.ClientNodeSelection.Name == "" { | ||||||
|  | 		claim, _ := cs.CoreV1().PersistentVolumeClaims(claimNamespace).Get(context.TODO(), claimName, metav1.GetOptions{}) | ||||||
|  | 		pvName := claim.Spec.VolumeName | ||||||
|  | 		volumeAttachments, _ := cs.StorageV1().VolumeAttachments().List(context.TODO(), metav1.ListOptions{}) | ||||||
|  | 		for _, volumeAttachment := range volumeAttachments.Items { | ||||||
|  | 			if *volumeAttachment.Spec.Source.PersistentVolumeName == pvName { | ||||||
|  | 				nodeName = volumeAttachment.Spec.NodeName | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		nodeName = config.ClientNodeSelection.Name | ||||||
|  | 	} | ||||||
|  | 	handle := getVolumeHandle(cs, claimName, claimNamespace) | ||||||
|  | 	attachmentHash := sha256.Sum256([]byte(fmt.Sprintf("%s%s%s", handle, provisioner, nodeName))) | ||||||
|  | 	return fmt.Sprintf("csi-%x", attachmentHash) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // getVolumeHandle returns the VolumeHandle of the PV that is bound to the PVC with the passed in claimName and claimNamespace. | ||||||
|  | func getVolumeHandle(cs clientset.Interface, claimName string, claimNamespace string) string { | ||||||
|  | 	// re-get the claim to the latest state with bound volume | ||||||
|  | 	claim, err := cs.CoreV1().PersistentVolumeClaims(claimNamespace).Get(context.TODO(), claimName, metav1.GetOptions{}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		framework.ExpectNoError(err, "Cannot get PVC") | ||||||
|  | 		return "" | ||||||
|  | 	} | ||||||
|  | 	pvName := claim.Spec.VolumeName | ||||||
|  | 	pv, err := cs.CoreV1().PersistentVolumes().Get(context.TODO(), pvName, metav1.GetOptions{}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		framework.ExpectNoError(err, "Cannot get PV") | ||||||
|  | 		return "" | ||||||
|  | 	} | ||||||
|  | 	if pv.Spec.CSI == nil { | ||||||
|  | 		gomega.Expect(pv.Spec.CSI).NotTo(gomega.BeNil()) | ||||||
|  | 		return "" | ||||||
|  | 	} | ||||||
|  | 	return pv.Spec.CSI.VolumeHandle | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // WaitForVolumeAttachmentTerminated waits for the VolumeAttachment with the passed in attachmentName to be terminated. | ||||||
|  | func WaitForVolumeAttachmentTerminated(attachmentName string, cs clientset.Interface, timeout time.Duration) error { | ||||||
|  | 	waitErr := wait.PollImmediate(10*time.Second, timeout, func() (bool, error) { | ||||||
|  | 		_, err := cs.StorageV1().VolumeAttachments().Get(context.TODO(), attachmentName, metav1.GetOptions{}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			// if the volumeattachment object is not found, it means it has been terminated. | ||||||
|  | 			if apierrors.IsNotFound(err) { | ||||||
|  | 				return true, nil | ||||||
|  | 			} | ||||||
|  | 			return false, err | ||||||
|  | 		} | ||||||
|  | 		return false, nil | ||||||
|  | 	}) | ||||||
|  | 	if waitErr != nil { | ||||||
|  | 		return fmt.Errorf("error waiting volume attachment %v to terminate: %v", attachmentName, waitErr) | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| // startVolumeServer starts a container specified by config.serverImage and exports all | // startVolumeServer starts a container specified by config.serverImage and exports all | ||||||
| // config.serverPorts from it. The returned pod should be used to get the server | // config.serverPorts from it. The returned pod should be used to get the server | ||||||
| // IP address and create appropriate VolumeSource. | // IP address and create appropriate VolumeSource. | ||||||
|   | |||||||
| @@ -18,7 +18,6 @@ package storage | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"crypto/sha256" |  | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"math/rand" | 	"math/rand" | ||||||
| @@ -374,9 +373,8 @@ var _ = utils.SIGDescribe("CSI mock volume", func() { | |||||||
| 				framework.ExpectNoError(err, "Failed to start pod: %v", err) | 				framework.ExpectNoError(err, "Failed to start pod: %v", err) | ||||||
|  |  | ||||||
| 				ginkgo.By("Checking if VolumeAttachment was created for the pod") | 				ginkgo.By("Checking if VolumeAttachment was created for the pod") | ||||||
| 				handle := getVolumeHandle(m.cs, claim) | 				testConfig := storageframework.ConvertTestConfig(m.config) | ||||||
| 				attachmentHash := sha256.Sum256([]byte(fmt.Sprintf("%s%s%s", handle, m.provisioner, m.config.ClientNodeSelection.Name))) | 				attachmentName := e2evolume.GetVolumeAttachmentName(m.cs, testConfig, m.provisioner, claim.Name, claim.Namespace) | ||||||
| 				attachmentName := fmt.Sprintf("csi-%x", attachmentHash) |  | ||||||
| 				_, err = m.cs.StorageV1().VolumeAttachments().Get(context.TODO(), attachmentName, metav1.GetOptions{}) | 				_, err = m.cs.StorageV1().VolumeAttachments().Get(context.TODO(), attachmentName, metav1.GetOptions{}) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					if apierrors.IsNotFound(err) { | 					if apierrors.IsNotFound(err) { | ||||||
| @@ -425,9 +423,8 @@ var _ = utils.SIGDescribe("CSI mock volume", func() { | |||||||
|  |  | ||||||
| 			// VolumeAttachment should be created because the default value for CSI attachable is true | 			// VolumeAttachment should be created because the default value for CSI attachable is true | ||||||
| 			ginkgo.By("Checking if VolumeAttachment was created for the pod") | 			ginkgo.By("Checking if VolumeAttachment was created for the pod") | ||||||
| 			handle := getVolumeHandle(m.cs, claim) | 			testConfig := storageframework.ConvertTestConfig(m.config) | ||||||
| 			attachmentHash := sha256.Sum256([]byte(fmt.Sprintf("%s%s%s", handle, m.provisioner, m.config.ClientNodeSelection.Name))) | 			attachmentName := e2evolume.GetVolumeAttachmentName(m.cs, testConfig, m.provisioner, claim.Name, claim.Namespace) | ||||||
| 			attachmentName := fmt.Sprintf("csi-%x", attachmentHash) |  | ||||||
| 			_, err = m.cs.StorageV1().VolumeAttachments().Get(context.TODO(), attachmentName, metav1.GetOptions{}) | 			_, err = m.cs.StorageV1().VolumeAttachments().Get(context.TODO(), attachmentName, metav1.GetOptions{}) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				if apierrors.IsNotFound(err) { | 				if apierrors.IsNotFound(err) { | ||||||
| @@ -461,7 +458,7 @@ var _ = utils.SIGDescribe("CSI mock volume", func() { | |||||||
| 			ginkgo.By(fmt.Sprintf("Wait for the volumeattachment to be deleted up to %v", csiVolumeAttachmentTimeout)) | 			ginkgo.By(fmt.Sprintf("Wait for the volumeattachment to be deleted up to %v", csiVolumeAttachmentTimeout)) | ||||||
| 			// This step can be slow because we have to wait either a NodeUpdate event happens or | 			// This step can be slow because we have to wait either a NodeUpdate event happens or | ||||||
| 			// the detachment for this volume timeout so that we can do a force detach. | 			// the detachment for this volume timeout so that we can do a force detach. | ||||||
| 			err = waitForVolumeAttachmentTerminated(attachmentName, m.cs) | 			err = e2evolume.WaitForVolumeAttachmentTerminated(attachmentName, m.cs, csiVolumeAttachmentTimeout) | ||||||
| 			framework.ExpectNoError(err, "Failed to delete VolumeAttachment: %v", err) | 			framework.ExpectNoError(err, "Failed to delete VolumeAttachment: %v", err) | ||||||
| 		}) | 		}) | ||||||
| 	}) | 	}) | ||||||
| @@ -2084,24 +2081,6 @@ func waitForMaxVolumeCondition(pod *v1.Pod, cs clientset.Interface) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func waitForVolumeAttachmentTerminated(attachmentName string, cs clientset.Interface) error { |  | ||||||
| 	waitErr := wait.PollImmediate(10*time.Second, csiVolumeAttachmentTimeout, func() (bool, error) { |  | ||||||
| 		_, err := cs.StorageV1().VolumeAttachments().Get(context.TODO(), attachmentName, metav1.GetOptions{}) |  | ||||||
| 		if err != nil { |  | ||||||
| 			// if the volumeattachment object is not found, it means it has been terminated. |  | ||||||
| 			if apierrors.IsNotFound(err) { |  | ||||||
| 				return true, nil |  | ||||||
| 			} |  | ||||||
| 			return false, err |  | ||||||
| 		} |  | ||||||
| 		return false, nil |  | ||||||
| 	}) |  | ||||||
| 	if waitErr != nil { |  | ||||||
| 		return fmt.Errorf("error waiting volume attachment %v to terminate: %v", attachmentName, waitErr) |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func checkCSINodeForLimits(nodeName string, driverName string, cs clientset.Interface) (int32, error) { | func checkCSINodeForLimits(nodeName string, driverName string, cs clientset.Interface) (int32, error) { | ||||||
| 	var attachLimit int32 | 	var attachLimit int32 | ||||||
|  |  | ||||||
| @@ -2426,26 +2405,6 @@ func destroyCSIDriver(cs clientset.Interface, driverName string) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func getVolumeHandle(cs clientset.Interface, claim *v1.PersistentVolumeClaim) string { |  | ||||||
| 	// re-get the claim to the latest state with bound volume |  | ||||||
| 	claim, err := cs.CoreV1().PersistentVolumeClaims(claim.Namespace).Get(context.TODO(), claim.Name, metav1.GetOptions{}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		framework.ExpectNoError(err, "Cannot get PVC") |  | ||||||
| 		return "" |  | ||||||
| 	} |  | ||||||
| 	pvName := claim.Spec.VolumeName |  | ||||||
| 	pv, err := cs.CoreV1().PersistentVolumes().Get(context.TODO(), pvName, metav1.GetOptions{}) |  | ||||||
| 	if err != nil { |  | ||||||
| 		framework.ExpectNoError(err, "Cannot get PV") |  | ||||||
| 		return "" |  | ||||||
| 	} |  | ||||||
| 	if pv.Spec.CSI == nil { |  | ||||||
| 		gomega.Expect(pv.Spec.CSI).NotTo(gomega.BeNil()) |  | ||||||
| 		return "" |  | ||||||
| 	} |  | ||||||
| 	return pv.Spec.CSI.VolumeHandle |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func getVolumeLimitFromCSINode(csiNode *storagev1.CSINode, driverName string) int32 { | func getVolumeLimitFromCSINode(csiNode *storagev1.CSINode, driverName string) int32 { | ||||||
| 	for _, d := range csiNode.Spec.Drivers { | 	for _, d := range csiNode.Spec.Drivers { | ||||||
| 		if d.Name != driverName { | 		if d.Name != driverName { | ||||||
|   | |||||||
| @@ -163,6 +163,7 @@ func (p *provisioningTestSuite) DefineTests(driver storageframework.TestDriver, | |||||||
| 			Claim:        l.pvc, | 			Claim:        l.pvc, | ||||||
| 			SourceClaim:  l.sourcePVC, | 			SourceClaim:  l.sourcePVC, | ||||||
| 			Class:        l.sc, | 			Class:        l.sc, | ||||||
|  | 			Provisioner:  l.sc.Provisioner, | ||||||
| 			ClaimSize:    claimSize, | 			ClaimSize:    claimSize, | ||||||
| 			ExpectedSize: claimSize, | 			ExpectedSize: claimSize, | ||||||
| 			VolumeMode:   pattern.VolMode, | 			VolumeMode:   pattern.VolMode, | ||||||
| @@ -254,7 +255,6 @@ func (p *provisioningTestSuite) DefineTests(driver storageframework.TestDriver, | |||||||
| 		expectedContent := fmt.Sprintf("Hello from namespace %s", f.Namespace.Name) | 		expectedContent := fmt.Sprintf("Hello from namespace %s", f.Namespace.Name) | ||||||
| 		dataSource, dataSourceCleanup := preparePVCDataSourceForProvisioning(f, testConfig, l.cs, l.sourcePVC, l.sc, pattern.VolMode, expectedContent) | 		dataSource, dataSourceCleanup := preparePVCDataSourceForProvisioning(f, testConfig, l.cs, l.sourcePVC, l.sc, pattern.VolMode, expectedContent) | ||||||
| 		defer dataSourceCleanup() | 		defer dataSourceCleanup() | ||||||
|  |  | ||||||
| 		l.pvc.Spec.DataSource = dataSource | 		l.pvc.Spec.DataSource = dataSource | ||||||
| 		l.testCase.NodeSelection = testConfig.ClientNodeSelection | 		l.testCase.NodeSelection = testConfig.ClientNodeSelection | ||||||
| 		l.testCase.PvCheck = func(claim *v1.PersistentVolumeClaim) { | 		l.testCase.PvCheck = func(claim *v1.PersistentVolumeClaim) { | ||||||
| @@ -269,6 +269,9 @@ func (p *provisioningTestSuite) DefineTests(driver storageframework.TestDriver, | |||||||
| 			} | 			} | ||||||
| 			e2evolume.TestVolumeClientSlow(f, testConfig, nil, "", tests) | 			e2evolume.TestVolumeClientSlow(f, testConfig, nil, "", tests) | ||||||
| 		} | 		} | ||||||
|  | 		// Cloning fails if the source disk is still in the process of detaching, so we wait for the VolumeAttachment to be removed before cloning. | ||||||
|  | 		volumeAttachment := e2evolume.GetVolumeAttachmentName(f.ClientSet, testConfig, l.testCase.Provisioner, dataSource.Name, l.sourcePVC.Namespace) | ||||||
|  | 		e2evolume.WaitForVolumeAttachmentTerminated(volumeAttachment, f.ClientSet, f.Timeouts.DataSourceProvision) | ||||||
| 		l.testCase.TestDynamicProvisioning() | 		l.testCase.TestDynamicProvisioning() | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| @@ -322,6 +325,9 @@ func (p *provisioningTestSuite) DefineTests(driver storageframework.TestDriver, | |||||||
| 					} | 					} | ||||||
| 					e2evolume.TestVolumeClientSlow(f, myTestConfig, nil, "", tests) | 					e2evolume.TestVolumeClientSlow(f, myTestConfig, nil, "", tests) | ||||||
| 				} | 				} | ||||||
|  | 				// Cloning fails if the source disk is still in the process of detaching, so we wait for the VolumeAttachment to be removed before cloning. | ||||||
|  | 				volumeAttachment := e2evolume.GetVolumeAttachmentName(f.ClientSet, testConfig, l.testCase.Provisioner, dataSource.Name, l.sourcePVC.Namespace) | ||||||
|  | 				e2evolume.WaitForVolumeAttachmentTerminated(volumeAttachment, f.ClientSet, f.Timeouts.DataSourceProvision) | ||||||
| 				t.TestDynamicProvisioning() | 				t.TestDynamicProvisioning() | ||||||
| 			}(i) | 			}(i) | ||||||
| 		} | 		} | ||||||
| @@ -377,7 +383,6 @@ func SetupStorageClass( | |||||||
| // see #ProvisionStorageClass | // see #ProvisionStorageClass | ||||||
| func (t StorageClassTest) TestDynamicProvisioning() *v1.PersistentVolume { | func (t StorageClassTest) TestDynamicProvisioning() *v1.PersistentVolume { | ||||||
| 	var err error | 	var err error | ||||||
|  |  | ||||||
| 	client := t.Client | 	client := t.Client | ||||||
| 	gomega.Expect(client).NotTo(gomega.BeNil(), "StorageClassTest.Client is required") | 	gomega.Expect(client).NotTo(gomega.BeNil(), "StorageClassTest.Client is required") | ||||||
| 	claim := t.Claim | 	claim := t.Claim | ||||||
| @@ -413,7 +418,7 @@ func (t StorageClassTest) TestDynamicProvisioning() *v1.PersistentVolume { | |||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		var pod *v1.Pod | 		var pod *v1.Pod | ||||||
| 		pod, err := e2epod.CreateSecPod(client, podConfig, framework.PodStartTimeout) | 		pod, err := e2epod.CreateSecPod(client, podConfig, t.Timeouts.DataSourceProvision) | ||||||
| 		// Delete pod now, otherwise PV can't be deleted below | 		// Delete pod now, otherwise PV can't be deleted below | ||||||
| 		framework.ExpectNoError(err) | 		framework.ExpectNoError(err) | ||||||
| 		e2epod.DeletePodOrFail(client, pod.Namespace, pod.Name) | 		e2epod.DeletePodOrFail(client, pod.Namespace, pod.Name) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Kubernetes Prow Robot
					Kubernetes Prow Robot