vsphere_volume_vsan_policy.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. /*
  2. Copyright 2017 The Kubernetes Authors.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package vsphere
  14. import (
  15. "context"
  16. "fmt"
  17. "hash/fnv"
  18. "time"
  19. "strings"
  20. "github.com/onsi/ginkgo"
  21. "github.com/onsi/gomega"
  22. v1 "k8s.io/api/core/v1"
  23. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  24. clientset "k8s.io/client-go/kubernetes"
  25. "k8s.io/kubernetes/test/e2e/framework"
  26. e2enode "k8s.io/kubernetes/test/e2e/framework/node"
  27. e2epod "k8s.io/kubernetes/test/e2e/framework/pod"
  28. e2epv "k8s.io/kubernetes/test/e2e/framework/pv"
  29. e2eskipper "k8s.io/kubernetes/test/e2e/framework/skipper"
  30. "k8s.io/kubernetes/test/e2e/storage/utils"
  31. )
  32. const (
  33. vmfsDatastore = "sharedVmfs-0"
  34. vsanDatastore = "vsanDatastore"
  35. dummyVMPrefixName = "vsphere-k8s"
  36. diskStripesCapabilityMaxVal = "11"
  37. )
  38. /*
  39. Test to verify the storage policy based management for dynamic volume provisioning inside kubernetes.
  40. There are 2 ways to achieve it:
  41. 1. Specify VSAN storage capabilities in the storage-class.
  42. 2. Use existing vCenter SPBM storage policies.
  43. Valid VSAN storage capabilities are mentioned below:
  44. 1. hostFailuresToTolerate
  45. 2. forceProvisioning
  46. 3. cacheReservation
  47. 4. diskStripes
  48. 5. objectSpaceReservation
  49. 6. iopsLimit
  50. Steps
  51. 1. Create StorageClass with.
  52. a. VSAN storage capabilities set to valid/invalid values (or)
  53. b. Use existing vCenter SPBM storage policies.
  54. 2. Create PVC which uses the StorageClass created in step 1.
  55. 3. Wait for PV to be provisioned.
  56. 4. Wait for PVC's status to become Bound
  57. 5. Create pod using PVC on specific node.
  58. 6. Wait for Disk to be attached to the node.
  59. 7. Delete pod and Wait for Volume Disk to be detached from the Node.
  60. 8. Delete PVC, PV and Storage Class
  61. */
  62. var _ = utils.SIGDescribe("Storage Policy Based Volume Provisioning [Feature:vsphere]", func() {
  63. f := framework.NewDefaultFramework("volume-vsan-policy")
  64. var (
  65. client clientset.Interface
  66. namespace string
  67. scParameters map[string]string
  68. policyName string
  69. tagPolicy string
  70. masterNode string
  71. )
  72. ginkgo.BeforeEach(func() {
  73. e2eskipper.SkipUnlessProviderIs("vsphere")
  74. Bootstrap(f)
  75. client = f.ClientSet
  76. namespace = f.Namespace.Name
  77. policyName = GetAndExpectStringEnvVar(SPBMPolicyName)
  78. tagPolicy = GetAndExpectStringEnvVar(SPBMTagPolicy)
  79. framework.Logf("framework: %+v", f)
  80. scParameters = make(map[string]string)
  81. _, err := e2enode.GetRandomReadySchedulableNode(f.ClientSet)
  82. framework.ExpectNoError(err)
  83. masternodes, _, err := e2enode.GetMasterAndWorkerNodes(client)
  84. framework.ExpectNoError(err)
  85. gomega.Expect(masternodes).NotTo(gomega.BeEmpty())
  86. masterNode = masternodes.List()[0]
  87. })
  88. // Valid policy.
  89. ginkgo.It("verify VSAN storage capability with valid hostFailuresToTolerate and cacheReservation values is honored for dynamically provisioned pvc using storageclass", func() {
  90. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy hostFailuresToTolerate: %s, cacheReservation: %s", HostFailuresToTolerateCapabilityVal, CacheReservationCapabilityVal))
  91. scParameters[PolicyHostFailuresToTolerate] = HostFailuresToTolerateCapabilityVal
  92. scParameters[PolicyCacheReservation] = CacheReservationCapabilityVal
  93. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  94. invokeValidPolicyTest(f, client, namespace, scParameters)
  95. })
  96. // Valid policy.
  97. ginkgo.It("verify VSAN storage capability with valid diskStripes and objectSpaceReservation values is honored for dynamically provisioned pvc using storageclass", func() {
  98. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy diskStripes: %s, objectSpaceReservation: %s", DiskStripesCapabilityVal, ObjectSpaceReservationCapabilityVal))
  99. scParameters[PolicyDiskStripes] = "1"
  100. scParameters[PolicyObjectSpaceReservation] = "30"
  101. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  102. invokeValidPolicyTest(f, client, namespace, scParameters)
  103. })
  104. // Valid policy.
  105. ginkgo.It("verify VSAN storage capability with valid diskStripes and objectSpaceReservation values and a VSAN datastore is honored for dynamically provisioned pvc using storageclass", func() {
  106. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy diskStripes: %s, objectSpaceReservation: %s", DiskStripesCapabilityVal, ObjectSpaceReservationCapabilityVal))
  107. scParameters[PolicyDiskStripes] = DiskStripesCapabilityVal
  108. scParameters[PolicyObjectSpaceReservation] = ObjectSpaceReservationCapabilityVal
  109. scParameters[Datastore] = vsanDatastore
  110. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  111. invokeValidPolicyTest(f, client, namespace, scParameters)
  112. })
  113. // Valid policy.
  114. ginkgo.It("verify VSAN storage capability with valid objectSpaceReservation and iopsLimit values is honored for dynamically provisioned pvc using storageclass", func() {
  115. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy objectSpaceReservation: %s, iopsLimit: %s", ObjectSpaceReservationCapabilityVal, IopsLimitCapabilityVal))
  116. scParameters[PolicyObjectSpaceReservation] = ObjectSpaceReservationCapabilityVal
  117. scParameters[PolicyIopsLimit] = IopsLimitCapabilityVal
  118. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  119. invokeValidPolicyTest(f, client, namespace, scParameters)
  120. })
  121. // Invalid VSAN storage capabilities parameters.
  122. ginkgo.It("verify VSAN storage capability with invalid capability name objectSpaceReserve is not honored for dynamically provisioned pvc using storageclass", func() {
  123. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy objectSpaceReserve: %s, stripeWidth: %s", ObjectSpaceReservationCapabilityVal, StripeWidthCapabilityVal))
  124. scParameters["objectSpaceReserve"] = ObjectSpaceReservationCapabilityVal
  125. scParameters[PolicyDiskStripes] = StripeWidthCapabilityVal
  126. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  127. err := invokeInvalidPolicyTestNeg(client, namespace, scParameters)
  128. framework.ExpectError(err)
  129. errorMsg := "invalid option \\\"objectSpaceReserve\\\" for volume plugin kubernetes.io/vsphere-volume"
  130. if !strings.Contains(err.Error(), errorMsg) {
  131. framework.ExpectNoError(err, errorMsg)
  132. }
  133. })
  134. // Invalid policy on a VSAN test bed.
  135. // diskStripes value has to be between 1 and 12.
  136. ginkgo.It("verify VSAN storage capability with invalid diskStripes value is not honored for dynamically provisioned pvc using storageclass", func() {
  137. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy diskStripes: %s, cacheReservation: %s", DiskStripesCapabilityInvalidVal, CacheReservationCapabilityVal))
  138. scParameters[PolicyDiskStripes] = DiskStripesCapabilityInvalidVal
  139. scParameters[PolicyCacheReservation] = CacheReservationCapabilityVal
  140. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  141. err := invokeInvalidPolicyTestNeg(client, namespace, scParameters)
  142. framework.ExpectError(err)
  143. errorMsg := "Invalid value for " + PolicyDiskStripes + "."
  144. if !strings.Contains(err.Error(), errorMsg) {
  145. framework.ExpectNoError(err, errorMsg)
  146. }
  147. })
  148. // Invalid policy on a VSAN test bed.
  149. // hostFailuresToTolerate value has to be between 0 and 3 including.
  150. ginkgo.It("verify VSAN storage capability with invalid hostFailuresToTolerate value is not honored for dynamically provisioned pvc using storageclass", func() {
  151. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy hostFailuresToTolerate: %s", HostFailuresToTolerateCapabilityInvalidVal))
  152. scParameters[PolicyHostFailuresToTolerate] = HostFailuresToTolerateCapabilityInvalidVal
  153. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  154. err := invokeInvalidPolicyTestNeg(client, namespace, scParameters)
  155. framework.ExpectError(err)
  156. errorMsg := "Invalid value for " + PolicyHostFailuresToTolerate + "."
  157. if !strings.Contains(err.Error(), errorMsg) {
  158. framework.ExpectNoError(err, errorMsg)
  159. }
  160. })
  161. // Specify a valid VSAN policy on a non-VSAN test bed.
  162. // The test should fail.
  163. ginkgo.It("verify VSAN storage capability with non-vsan datastore is not honored for dynamically provisioned pvc using storageclass", func() {
  164. ginkgo.By(fmt.Sprintf("Invoking test for VSAN policy diskStripes: %s, objectSpaceReservation: %s and a non-VSAN datastore: %s", DiskStripesCapabilityVal, ObjectSpaceReservationCapabilityVal, vmfsDatastore))
  165. scParameters[PolicyDiskStripes] = DiskStripesCapabilityVal
  166. scParameters[PolicyObjectSpaceReservation] = ObjectSpaceReservationCapabilityVal
  167. scParameters[Datastore] = vmfsDatastore
  168. framework.Logf("Invoking test for VSAN storage capabilities: %+v", scParameters)
  169. err := invokeInvalidPolicyTestNeg(client, namespace, scParameters)
  170. framework.ExpectError(err)
  171. errorMsg := "The specified datastore: \\\"" + vmfsDatastore + "\\\" is not a VSAN datastore. " +
  172. "The policy parameters will work only with VSAN Datastore."
  173. if !strings.Contains(err.Error(), errorMsg) {
  174. framework.ExpectNoError(err, errorMsg)
  175. }
  176. })
  177. ginkgo.It("verify an existing and compatible SPBM policy is honored for dynamically provisioned pvc using storageclass", func() {
  178. ginkgo.By(fmt.Sprintf("Invoking test for SPBM policy: %s", policyName))
  179. scParameters[SpbmStoragePolicy] = policyName
  180. scParameters[DiskFormat] = ThinDisk
  181. framework.Logf("Invoking test for SPBM storage policy: %+v", scParameters)
  182. invokeValidPolicyTest(f, client, namespace, scParameters)
  183. })
  184. ginkgo.It("verify clean up of stale dummy VM for dynamically provisioned pvc using SPBM policy", func() {
  185. scParameters[PolicyDiskStripes] = diskStripesCapabilityMaxVal
  186. scParameters[PolicyObjectSpaceReservation] = ObjectSpaceReservationCapabilityVal
  187. scParameters[Datastore] = vsanDatastore
  188. framework.Logf("Invoking test for SPBM storage policy: %+v", scParameters)
  189. kubernetesClusterName := GetAndExpectStringEnvVar(KubernetesClusterName)
  190. invokeStaleDummyVMTestWithStoragePolicy(client, masterNode, namespace, kubernetesClusterName, scParameters)
  191. })
  192. ginkgo.It("verify if a SPBM policy is not honored on a non-compatible datastore for dynamically provisioned pvc using storageclass", func() {
  193. ginkgo.By(fmt.Sprintf("Invoking test for SPBM policy: %s and datastore: %s", tagPolicy, vsanDatastore))
  194. scParameters[SpbmStoragePolicy] = tagPolicy
  195. scParameters[Datastore] = vsanDatastore
  196. scParameters[DiskFormat] = ThinDisk
  197. framework.Logf("Invoking test for SPBM storage policy on a non-compatible datastore: %+v", scParameters)
  198. err := invokeInvalidPolicyTestNeg(client, namespace, scParameters)
  199. framework.ExpectError(err)
  200. errorMsg := "User specified datastore is not compatible with the storagePolicy: \\\"" + tagPolicy + "\\\""
  201. if !strings.Contains(err.Error(), errorMsg) {
  202. framework.ExpectNoError(err, errorMsg)
  203. }
  204. })
  205. ginkgo.It("verify if a non-existing SPBM policy is not honored for dynamically provisioned pvc using storageclass", func() {
  206. ginkgo.By(fmt.Sprintf("Invoking test for SPBM policy: %s", BronzeStoragePolicy))
  207. scParameters[SpbmStoragePolicy] = BronzeStoragePolicy
  208. scParameters[DiskFormat] = ThinDisk
  209. framework.Logf("Invoking test for non-existing SPBM storage policy: %+v", scParameters)
  210. err := invokeInvalidPolicyTestNeg(client, namespace, scParameters)
  211. framework.ExpectError(err)
  212. errorMsg := "no pbm profile found with name: \\\"" + BronzeStoragePolicy + "\\"
  213. if !strings.Contains(err.Error(), errorMsg) {
  214. framework.ExpectNoError(err, errorMsg)
  215. }
  216. })
  217. ginkgo.It("verify an if a SPBM policy and VSAN capabilities cannot be honored for dynamically provisioned pvc using storageclass", func() {
  218. ginkgo.By(fmt.Sprintf("Invoking test for SPBM policy: %s with VSAN storage capabilities", policyName))
  219. scParameters[SpbmStoragePolicy] = policyName
  220. gomega.Expect(scParameters[SpbmStoragePolicy]).NotTo(gomega.BeEmpty())
  221. scParameters[PolicyDiskStripes] = DiskStripesCapabilityVal
  222. scParameters[DiskFormat] = ThinDisk
  223. framework.Logf("Invoking test for SPBM storage policy and VSAN capabilities together: %+v", scParameters)
  224. err := invokeInvalidPolicyTestNeg(client, namespace, scParameters)
  225. framework.ExpectError(err)
  226. errorMsg := "Cannot specify storage policy capabilities along with storage policy name. Please specify only one"
  227. if !strings.Contains(err.Error(), errorMsg) {
  228. framework.ExpectNoError(err, errorMsg)
  229. }
  230. })
  231. })
  232. func invokeValidPolicyTest(f *framework.Framework, client clientset.Interface, namespace string, scParameters map[string]string) {
  233. ginkgo.By("Creating Storage Class With storage policy params")
  234. storageclass, err := client.StorageV1().StorageClasses().Create(context.TODO(), getVSphereStorageClassSpec("storagepolicysc", scParameters, nil, ""), metav1.CreateOptions{})
  235. framework.ExpectNoError(err, fmt.Sprintf("Failed to create storage class with err: %v", err))
  236. defer client.StorageV1().StorageClasses().Delete(context.TODO(), storageclass.Name, nil)
  237. ginkgo.By("Creating PVC using the Storage Class")
  238. pvclaim, err := e2epv.CreatePVC(client, namespace, getVSphereClaimSpecWithStorageClass(namespace, "2Gi", storageclass))
  239. framework.ExpectNoError(err)
  240. defer e2epv.DeletePersistentVolumeClaim(client, pvclaim.Name, namespace)
  241. var pvclaims []*v1.PersistentVolumeClaim
  242. pvclaims = append(pvclaims, pvclaim)
  243. ginkgo.By("Waiting for claim to be in bound phase")
  244. persistentvolumes, err := e2epv.WaitForPVClaimBoundPhase(client, pvclaims, framework.ClaimProvisionTimeout)
  245. framework.ExpectNoError(err)
  246. ginkgo.By("Creating pod to attach PV to the node")
  247. // Create pod to attach Volume to Node
  248. pod, err := e2epod.CreatePod(client, namespace, nil, pvclaims, false, "")
  249. framework.ExpectNoError(err)
  250. ginkgo.By("Verify the volume is accessible and available in the pod")
  251. verifyVSphereVolumesAccessible(client, pod, persistentvolumes)
  252. ginkgo.By("Deleting pod")
  253. e2epod.DeletePodWithWait(client, pod)
  254. ginkgo.By("Waiting for volumes to be detached from the node")
  255. waitForVSphereDiskToDetach(persistentvolumes[0].Spec.VsphereVolume.VolumePath, pod.Spec.NodeName)
  256. }
  257. func invokeInvalidPolicyTestNeg(client clientset.Interface, namespace string, scParameters map[string]string) error {
  258. ginkgo.By("Creating Storage Class With storage policy params")
  259. storageclass, err := client.StorageV1().StorageClasses().Create(context.TODO(), getVSphereStorageClassSpec("storagepolicysc", scParameters, nil, ""), metav1.CreateOptions{})
  260. framework.ExpectNoError(err, fmt.Sprintf("Failed to create storage class with err: %v", err))
  261. defer client.StorageV1().StorageClasses().Delete(context.TODO(), storageclass.Name, nil)
  262. ginkgo.By("Creating PVC using the Storage Class")
  263. pvclaim, err := e2epv.CreatePVC(client, namespace, getVSphereClaimSpecWithStorageClass(namespace, "2Gi", storageclass))
  264. framework.ExpectNoError(err)
  265. defer e2epv.DeletePersistentVolumeClaim(client, pvclaim.Name, namespace)
  266. ginkgo.By("Waiting for claim to be in bound phase")
  267. err = e2epv.WaitForPersistentVolumeClaimPhase(v1.ClaimBound, client, pvclaim.Namespace, pvclaim.Name, framework.Poll, 2*time.Minute)
  268. framework.ExpectError(err)
  269. eventList, err := client.CoreV1().Events(pvclaim.Namespace).List(context.TODO(), metav1.ListOptions{})
  270. framework.ExpectNoError(err)
  271. return fmt.Errorf("Failure message: %+q", eventList.Items[0].Message)
  272. }
  273. func invokeStaleDummyVMTestWithStoragePolicy(client clientset.Interface, masterNode string, namespace string, clusterName string, scParameters map[string]string) {
  274. ginkgo.By("Creating Storage Class With storage policy params")
  275. storageclass, err := client.StorageV1().StorageClasses().Create(context.TODO(), getVSphereStorageClassSpec("storagepolicysc", scParameters, nil, ""), metav1.CreateOptions{})
  276. framework.ExpectNoError(err, fmt.Sprintf("Failed to create storage class with err: %v", err))
  277. defer client.StorageV1().StorageClasses().Delete(context.TODO(), storageclass.Name, nil)
  278. ginkgo.By("Creating PVC using the Storage Class")
  279. pvclaim, err := e2epv.CreatePVC(client, namespace, getVSphereClaimSpecWithStorageClass(namespace, "2Gi", storageclass))
  280. framework.ExpectNoError(err)
  281. var pvclaims []*v1.PersistentVolumeClaim
  282. pvclaims = append(pvclaims, pvclaim)
  283. ginkgo.By("Expect claim to fail provisioning volume")
  284. _, err = e2epv.WaitForPVClaimBoundPhase(client, pvclaims, 2*time.Minute)
  285. framework.ExpectError(err)
  286. updatedClaim, err := client.CoreV1().PersistentVolumeClaims(namespace).Get(context.TODO(), pvclaim.Name, metav1.GetOptions{})
  287. framework.ExpectNoError(err)
  288. vmName := clusterName + "-dynamic-pvc-" + string(updatedClaim.UID)
  289. e2epv.DeletePersistentVolumeClaim(client, pvclaim.Name, namespace)
  290. // Wait for 6 minutes to let the vSphere Cloud Provider clean up routine delete the dummy VM
  291. time.Sleep(6 * time.Minute)
  292. fnvHash := fnv.New32a()
  293. fnvHash.Write([]byte(vmName))
  294. dummyVMFullName := dummyVMPrefixName + "-" + fmt.Sprint(fnvHash.Sum32())
  295. errorMsg := "Dummy VM - " + vmName + "is still present. Failing the test.."
  296. nodeInfo := TestContext.NodeMapper.GetNodeInfo(masterNode)
  297. isVMPresentFlag, _ := nodeInfo.VSphere.IsVMPresent(dummyVMFullName, nodeInfo.DataCenterRef)
  298. framework.ExpectNotEqual(isVMPresentFlag, true, errorMsg)
  299. }