Skip to content

Commit

Permalink
Test delete VRG with an older csi-addons
Browse files Browse the repository at this point in the history
In csi-addons < 0.10.0 we could not detect failed validation since the
Validated condition is missing.

Add 2 tests:

- VR failed validation: the VRG should block deletion until the VR is
  deleted manually.
- VR completed: the VRG should not block deletion because the Validated
  condition is missing.

Signed-off-by: Nir Soffer <[email protected]>
  • Loading branch information
nirs committed Oct 1, 2024
1 parent aace41f commit 5735835
Showing 1 changed file with 112 additions and 0 deletions.
112 changes: 112 additions & 0 deletions internal/controller/vrg_volrep_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -659,6 +659,105 @@ var _ = Describe("VolumeReplicationGroupVolRepController", func() {
})
})

// Test VRG deletion when VR failed validation and Validated condition is missing (csi-addons < 0.10.0)
var vrgDeleteIncompleteVR *vrgTest
//nolint:dupl
Context("VR failed validation in primary state and Validated condition is missing", func() {
createTestTemplate := &template{
ClaimBindInfo: corev1.ClaimBound,
VolumeBindInfo: corev1.VolumeBound,
schedulingInterval: "1h",
storageClassName: "manual",
replicationClassName: "test-replicationclass",
vrcProvisioner: "manual.storage.com",
scProvisioner: "manual.storage.com",
replicationClassLabels: map[string]string{"protection": "ramen"},
}
It("sets up PVCs, PVs and VRGs (with s3 stores that fail uploads)", func() {
createTestTemplate.s3Profiles = []string{s3Profiles[vrgS3ProfileNumber].S3ProfileName}
vrgDeleteIncompleteVR = newVRGTestCaseCreateAndStart(1, createTestTemplate, true, false)
})
It("waits for VRG to create a VR for each PVC", func() {
expectedVRCount := len(vrgDeleteFailedVR.pvcNames)
vrgDeleteIncompleteVR.waitForVRCountToMatch(expectedVRCount)
})
It("simulate incomplete VR", func() {
vrgDeleteIncompleteVR.promoteVolRepsWithOptions(promoteOptions{ValidatedFailed: true, ValidatedMissing: true})
})
It("VRG can not be deleted", func() {
By("deleting the VRG")
vrg := vrgDeleteIncompleteVR.getVRG()
Expect(k8sClient.Delete(context.TODO(), vrg)).To(Succeed())

By("ensuring VRG cannot be deleted")
Consistently(func() error {
return apiReader.Get(context.TODO(), vrgDeleteIncompleteVR.vrgNamespacedName(), vrg)
}, vrgtimeout, vrginterval).
Should(Succeed(), "VRG %s was deleted when VR is incomplete", vrgDeleteIncompleteVR.vrgName)

By("deleting the VRs")
vrgDeleteIncompleteVR.deleteVolReps()

By("ensuring the VRG is deleted")
Eventually(func() error {
return apiReader.Get(context.TODO(), vrgDeleteFailedVR.vrgNamespacedName(), vrg)
}, vrgtimeout, vrginterval).
Should(MatchError(errors.NewNotFound(schema.GroupResource{
Group: ramendrv1alpha1.GroupVersion.Group,
Resource: "volumereplicationgroups",
}, vrgDeleteFailedVR.vrgName)))

vrgDeleteIncompleteVR.cleanupNamespace()
vrgDeleteIncompleteVR.cleanupSC()
vrgDeleteIncompleteVR.cleanupVRC()
})
})

// Test VRG deletion when VR completed and Validated condition is missing (csi-addons < 0.10.0)
var vrgDeleteCompletedVR *vrgTest
//nolint:dupl
Context("VR failed validation in primary state and Validated condition is missing", func() {
createTestTemplate := &template{
ClaimBindInfo: corev1.ClaimBound,
VolumeBindInfo: corev1.VolumeBound,
schedulingInterval: "1h",
storageClassName: "manual",
replicationClassName: "test-replicationclass",
vrcProvisioner: "manual.storage.com",
scProvisioner: "manual.storage.com",
replicationClassLabels: map[string]string{"protection": "ramen"},
}
It("sets up PVCs, PVs and VRGs (with s3 stores that fail uploads)", func() {
createTestTemplate.s3Profiles = []string{s3Profiles[vrgS3ProfileNumber].S3ProfileName}
vrgDeleteCompletedVR = newVRGTestCaseCreateAndStart(1, createTestTemplate, true, false)
})
It("waits for VRG to create a VR for each PVC", func() {
expectedVRCount := len(vrgDeleteFailedVR.pvcNames)
vrgDeleteCompletedVR.waitForVRCountToMatch(expectedVRCount)
})
It("simulate completed VR", func() {
vrgDeleteCompletedVR.promoteVolRepsWithOptions(promoteOptions{ValidatedMissing: true})
})
It("VRG can be deleted", func() {
By("deleting the VRG")
vrg := vrgDeleteCompletedVR.getVRG()
Expect(k8sClient.Delete(context.TODO(), vrg)).To(Succeed())

By("ensuring the VRG is deleted")
Eventually(func() error {
return apiReader.Get(context.TODO(), vrgDeleteFailedVR.vrgNamespacedName(), vrg)
}, vrgtimeout, vrginterval).
Should(MatchError(errors.NewNotFound(schema.GroupResource{
Group: ramendrv1alpha1.GroupVersion.Group,
Resource: "volumereplicationgroups",
}, vrgDeleteFailedVR.vrgName)))

vrgDeleteCompletedVR.cleanupNamespace()
vrgDeleteCompletedVR.cleanupSC()
vrgDeleteCompletedVR.cleanupVRC()
})
})

// Try the simple case of creating VRG, PVC, PV and
// check whether VolRep resources are created or not
var vrgTestCases []*vrgTest
Expand Down Expand Up @@ -2331,6 +2430,19 @@ func (v *vrgTest) generateVRConditions(generation int64, options promoteOptions)
return append(conditions, completed, degraded, resyncing)
}

func (v *vrgTest) deleteVolReps() {
vrList := &volrep.VolumeReplicationList{}
err := k8sClient.List(context.TODO(), vrList, &client.ListOptions{Namespace: v.namespace})
Expect(err).NotTo(HaveOccurred(), "failed to get a list of VRs in namespace %s", v.namespace)

for i := range vrList.Items {
vr := vrList.Items[i]

err := k8sClient.Delete(context.TODO(), &vr)
Expect(err).NotTo(HaveOccurred(), "failed to delete volRep %v/%s", vr.Namespace, vr.Name)
}
}

func (v *vrgTest) protectDeletionOfVolReps() {
By("Adding a finalizer to protect VolumeReplication resources being deleted " + v.namespace)

Expand Down

0 comments on commit 5735835

Please sign in to comment.