vsphere_volume_vsan_policy.go 18 KB

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