vsphere_volume_perf.go 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  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. "time"
  17. "github.com/onsi/ginkgo"
  18. "github.com/onsi/gomega"
  19. v1 "k8s.io/api/core/v1"
  20. storageV1 "k8s.io/api/storage/v1"
  21. clientset "k8s.io/client-go/kubernetes"
  22. "k8s.io/kubernetes/test/e2e/framework"
  23. e2elog "k8s.io/kubernetes/test/e2e/framework/log"
  24. "k8s.io/kubernetes/test/e2e/storage/utils"
  25. )
  26. /* This test calculates latency numbers for volume lifecycle operations
  27. 1. Create 4 type of storage classes
  28. 2. Read the total number of volumes to be created and volumes per pod
  29. 3. Create total PVCs (number of volumes)
  30. 4. Create Pods with attached volumes per pod
  31. 5. Verify access to the volumes
  32. 6. Delete pods and wait for volumes to detach
  33. 7. Delete the PVCs
  34. */
  35. const (
  36. SCSIUnitsAvailablePerNode = 55
  37. CreateOp = "CreateOp"
  38. AttachOp = "AttachOp"
  39. DetachOp = "DetachOp"
  40. DeleteOp = "DeleteOp"
  41. )
  42. var _ = utils.SIGDescribe("vcp-performance [Feature:vsphere]", func() {
  43. f := framework.NewDefaultFramework("vcp-performance")
  44. var (
  45. client clientset.Interface
  46. namespace string
  47. nodeSelectorList []*NodeSelector
  48. policyName string
  49. datastoreName string
  50. volumeCount int
  51. volumesPerPod int
  52. iterations int
  53. )
  54. ginkgo.BeforeEach(func() {
  55. framework.SkipUnlessProviderIs("vsphere")
  56. Bootstrap(f)
  57. client = f.ClientSet
  58. namespace = f.Namespace.Name
  59. // Read the environment variables
  60. volumeCount = GetAndExpectIntEnvVar(VCPPerfVolumeCount)
  61. volumesPerPod = GetAndExpectIntEnvVar(VCPPerfVolumesPerPod)
  62. iterations = GetAndExpectIntEnvVar(VCPPerfIterations)
  63. policyName = GetAndExpectStringEnvVar(SPBMPolicyName)
  64. datastoreName = GetAndExpectStringEnvVar(StorageClassDatastoreName)
  65. nodes := framework.GetReadySchedulableNodesOrDie(client)
  66. gomega.Expect(len(nodes.Items)).To(gomega.BeNumerically(">=", 1), "Requires at least %d nodes (not %d)", 2, len(nodes.Items))
  67. msg := fmt.Sprintf("Cannot attach %d volumes to %d nodes. Maximum volumes that can be attached on %d nodes is %d", volumeCount, len(nodes.Items), len(nodes.Items), SCSIUnitsAvailablePerNode*len(nodes.Items))
  68. gomega.Expect(volumeCount).To(gomega.BeNumerically("<=", SCSIUnitsAvailablePerNode*len(nodes.Items)), msg)
  69. msg = fmt.Sprintf("Cannot attach %d volumes per pod. Maximum volumes that can be attached per pod is %d", volumesPerPod, SCSIUnitsAvailablePerNode)
  70. gomega.Expect(volumesPerPod).To(gomega.BeNumerically("<=", SCSIUnitsAvailablePerNode), msg)
  71. nodeSelectorList = createNodeLabels(client, namespace, nodes)
  72. })
  73. ginkgo.It("vcp performance tests", func() {
  74. scList := getTestStorageClasses(client, policyName, datastoreName)
  75. defer func(scList []*storageV1.StorageClass) {
  76. for _, sc := range scList {
  77. client.StorageV1().StorageClasses().Delete(sc.Name, nil)
  78. }
  79. }(scList)
  80. sumLatency := make(map[string]float64)
  81. for i := 0; i < iterations; i++ {
  82. latency := invokeVolumeLifeCyclePerformance(f, client, namespace, scList, volumesPerPod, volumeCount, nodeSelectorList)
  83. for key, val := range latency {
  84. sumLatency[key] += val
  85. }
  86. }
  87. iterations64 := float64(iterations)
  88. e2elog.Logf("Average latency for below operations")
  89. e2elog.Logf("Creating %d PVCs and waiting for bound phase: %v seconds", volumeCount, sumLatency[CreateOp]/iterations64)
  90. e2elog.Logf("Creating %v Pod: %v seconds", volumeCount/volumesPerPod, sumLatency[AttachOp]/iterations64)
  91. e2elog.Logf("Deleting %v Pod and waiting for disk to be detached: %v seconds", volumeCount/volumesPerPod, sumLatency[DetachOp]/iterations64)
  92. e2elog.Logf("Deleting %v PVCs: %v seconds", volumeCount, sumLatency[DeleteOp]/iterations64)
  93. })
  94. })
  95. func getTestStorageClasses(client clientset.Interface, policyName, datastoreName string) []*storageV1.StorageClass {
  96. const (
  97. storageclass1 = "sc-default"
  98. storageclass2 = "sc-vsan"
  99. storageclass3 = "sc-spbm"
  100. storageclass4 = "sc-user-specified-ds"
  101. )
  102. scNames := []string{storageclass1, storageclass2, storageclass3, storageclass4}
  103. scArrays := make([]*storageV1.StorageClass, len(scNames))
  104. for index, scname := range scNames {
  105. // Create vSphere Storage Class
  106. ginkgo.By(fmt.Sprintf("Creating Storage Class : %v", scname))
  107. var sc *storageV1.StorageClass
  108. var err error
  109. switch scname {
  110. case storageclass1:
  111. sc, err = client.StorageV1().StorageClasses().Create(getVSphereStorageClassSpec(storageclass1, nil, nil))
  112. case storageclass2:
  113. var scVSanParameters map[string]string
  114. scVSanParameters = make(map[string]string)
  115. scVSanParameters[Policy_HostFailuresToTolerate] = "1"
  116. sc, err = client.StorageV1().StorageClasses().Create(getVSphereStorageClassSpec(storageclass2, scVSanParameters, nil))
  117. case storageclass3:
  118. var scSPBMPolicyParameters map[string]string
  119. scSPBMPolicyParameters = make(map[string]string)
  120. scSPBMPolicyParameters[SpbmStoragePolicy] = policyName
  121. sc, err = client.StorageV1().StorageClasses().Create(getVSphereStorageClassSpec(storageclass3, scSPBMPolicyParameters, nil))
  122. case storageclass4:
  123. var scWithDSParameters map[string]string
  124. scWithDSParameters = make(map[string]string)
  125. scWithDSParameters[Datastore] = datastoreName
  126. scWithDatastoreSpec := getVSphereStorageClassSpec(storageclass4, scWithDSParameters, nil)
  127. sc, err = client.StorageV1().StorageClasses().Create(scWithDatastoreSpec)
  128. }
  129. gomega.Expect(sc).NotTo(gomega.BeNil())
  130. framework.ExpectNoError(err)
  131. scArrays[index] = sc
  132. }
  133. return scArrays
  134. }
  135. // invokeVolumeLifeCyclePerformance peforms full volume life cycle management and records latency for each operation
  136. func invokeVolumeLifeCyclePerformance(f *framework.Framework, client clientset.Interface, namespace string, sc []*storageV1.StorageClass, volumesPerPod int, volumeCount int, nodeSelectorList []*NodeSelector) (latency map[string]float64) {
  137. var (
  138. totalpvclaims [][]*v1.PersistentVolumeClaim
  139. totalpvs [][]*v1.PersistentVolume
  140. totalpods []*v1.Pod
  141. )
  142. nodeVolumeMap := make(map[string][]string)
  143. latency = make(map[string]float64)
  144. numPods := volumeCount / volumesPerPod
  145. ginkgo.By(fmt.Sprintf("Creating %d PVCs", volumeCount))
  146. start := time.Now()
  147. for i := 0; i < numPods; i++ {
  148. var pvclaims []*v1.PersistentVolumeClaim
  149. for j := 0; j < volumesPerPod; j++ {
  150. currsc := sc[((i*numPods)+j)%len(sc)]
  151. pvclaim, err := framework.CreatePVC(client, namespace, getVSphereClaimSpecWithStorageClass(namespace, "2Gi", currsc))
  152. framework.ExpectNoError(err)
  153. pvclaims = append(pvclaims, pvclaim)
  154. }
  155. totalpvclaims = append(totalpvclaims, pvclaims)
  156. }
  157. for _, pvclaims := range totalpvclaims {
  158. persistentvolumes, err := framework.WaitForPVClaimBoundPhase(client, pvclaims, framework.ClaimProvisionTimeout)
  159. framework.ExpectNoError(err)
  160. totalpvs = append(totalpvs, persistentvolumes)
  161. }
  162. elapsed := time.Since(start)
  163. latency[CreateOp] = elapsed.Seconds()
  164. ginkgo.By("Creating pod to attach PVs to the node")
  165. start = time.Now()
  166. for i, pvclaims := range totalpvclaims {
  167. nodeSelector := nodeSelectorList[i%len(nodeSelectorList)]
  168. pod, err := framework.CreatePod(client, namespace, map[string]string{nodeSelector.labelKey: nodeSelector.labelValue}, pvclaims, false, "")
  169. framework.ExpectNoError(err)
  170. totalpods = append(totalpods, pod)
  171. defer framework.DeletePodWithWait(f, client, pod)
  172. }
  173. elapsed = time.Since(start)
  174. latency[AttachOp] = elapsed.Seconds()
  175. for i, pod := range totalpods {
  176. verifyVSphereVolumesAccessible(client, pod, totalpvs[i])
  177. }
  178. ginkgo.By("Deleting pods")
  179. start = time.Now()
  180. for _, pod := range totalpods {
  181. err := framework.DeletePodWithWait(f, client, pod)
  182. framework.ExpectNoError(err)
  183. }
  184. elapsed = time.Since(start)
  185. latency[DetachOp] = elapsed.Seconds()
  186. for i, pod := range totalpods {
  187. for _, pv := range totalpvs[i] {
  188. nodeVolumeMap[pod.Spec.NodeName] = append(nodeVolumeMap[pod.Spec.NodeName], pv.Spec.VsphereVolume.VolumePath)
  189. }
  190. }
  191. err := waitForVSphereDisksToDetach(nodeVolumeMap)
  192. framework.ExpectNoError(err)
  193. ginkgo.By("Deleting the PVCs")
  194. start = time.Now()
  195. for _, pvclaims := range totalpvclaims {
  196. for _, pvc := range pvclaims {
  197. err = framework.DeletePersistentVolumeClaim(client, pvc.Name, namespace)
  198. framework.ExpectNoError(err)
  199. }
  200. }
  201. elapsed = time.Since(start)
  202. latency[DeleteOp] = elapsed.Seconds()
  203. return latency
  204. }